✨ feat(manager.py): add support for initializing alembic if not already initialized before running migrations
240 lines
9.4 KiB
Python
240 lines
9.4 KiB
Python
from pathlib import Path
|
|
from typing import TYPE_CHECKING
|
|
from langflow.services.base import Service
|
|
from langflow.services.database.models.user.crud import get_user_by_username
|
|
from langflow.services.database.utils import Result, TableResults
|
|
from langflow.services.getters import get_settings_service
|
|
from sqlalchemy import inspect
|
|
import sqlalchemy as sa
|
|
from sqlalchemy.exc import OperationalError
|
|
from sqlmodel import SQLModel, Session, create_engine
|
|
from loguru import logger
|
|
from alembic.config import Config
|
|
from alembic import command
|
|
from langflow.services.database import models # noqa
|
|
|
|
if TYPE_CHECKING:
|
|
from sqlalchemy.engine import Engine
|
|
|
|
|
|
class DatabaseService(Service):
|
|
name = "database_service"
|
|
|
|
def __init__(self, database_url: str):
|
|
self.database_url = database_url
|
|
# This file is in langflow.services.database.manager.py
|
|
# the ini is in langflow
|
|
langflow_dir = Path(__file__).parent.parent.parent
|
|
self.script_location = langflow_dir / "alembic"
|
|
self.alembic_cfg_path = langflow_dir / "alembic.ini"
|
|
self.engine = self._create_engine()
|
|
|
|
def _create_engine(self) -> "Engine":
|
|
"""Create the engine for the database."""
|
|
settings_service = get_settings_service()
|
|
if (
|
|
settings_service.settings.DATABASE_URL
|
|
and settings_service.settings.DATABASE_URL.startswith("sqlite")
|
|
):
|
|
connect_args = {"check_same_thread": False}
|
|
else:
|
|
connect_args = {}
|
|
return create_engine(self.database_url, connect_args=connect_args)
|
|
|
|
def __enter__(self):
|
|
self._session = Session(self.engine)
|
|
return self._session
|
|
|
|
def __exit__(self, exc_type, exc_value, traceback):
|
|
if exc_type is not None: # If an exception has been raised
|
|
logger.error(
|
|
f"Session rollback because of exception: {exc_type.__name__} {exc_value}"
|
|
)
|
|
self._session.rollback()
|
|
else:
|
|
self._session.commit()
|
|
self._session.close()
|
|
|
|
def get_session(self):
|
|
with Session(self.engine) as session:
|
|
yield session
|
|
|
|
def migrate_flows_if_auto_login(self):
|
|
# if auto_login is enabled, we need to migrate the flows
|
|
# to the default superuser if they don't have a user id
|
|
# associated with them
|
|
settings_service = get_settings_service()
|
|
if settings_service.auth_settings.AUTO_LOGIN:
|
|
with Session(self.engine) as session:
|
|
flows = (
|
|
session.query(models.Flow)
|
|
.filter(models.Flow.user_id == None) # noqa
|
|
.all()
|
|
)
|
|
if flows:
|
|
logger.debug("Migrating flows to default superuser")
|
|
username = settings_service.auth_settings.SUPERUSER
|
|
user = get_user_by_username(session, username)
|
|
for flow in flows:
|
|
flow.user_id = user.id
|
|
session.commit()
|
|
logger.debug("Flows migrated successfully")
|
|
|
|
def check_schema_health(self) -> bool:
|
|
inspector = inspect(self.engine)
|
|
|
|
model_mapping = {
|
|
"flow": models.Flow,
|
|
"user": models.User,
|
|
"apikey": models.ApiKey,
|
|
# Add other SQLModel classes here
|
|
}
|
|
|
|
# To account for tables that existed in older versions
|
|
legacy_tables = ["flowstyle"]
|
|
|
|
for table, model in model_mapping.items():
|
|
expected_columns = list(model.__fields__.keys())
|
|
|
|
try:
|
|
available_columns = [
|
|
col["name"] for col in inspector.get_columns(table)
|
|
]
|
|
except sa.exc.NoSuchTableError:
|
|
logger.error(f"Missing table: {table}")
|
|
return False
|
|
|
|
for column in expected_columns:
|
|
if column not in available_columns:
|
|
logger.error(f"Missing column: {column} in table {table}")
|
|
return False
|
|
|
|
for table in legacy_tables:
|
|
if table in inspector.get_table_names():
|
|
logger.warning(f"Legacy table exists: {table}")
|
|
|
|
return True
|
|
|
|
def init_alembic(self):
|
|
logger.info("Initializing alembic")
|
|
alembic_cfg = Config()
|
|
alembic_cfg.set_main_option("script_location", str(self.script_location))
|
|
alembic_cfg.set_main_option("sqlalchemy.url", self.database_url)
|
|
command.stamp(alembic_cfg, "head")
|
|
logger.info("Alembic initialized")
|
|
|
|
def run_migrations(self):
|
|
# First we need to check if alembic has been initialized
|
|
# If not, we need to initialize it
|
|
# if not self.script_location.exists(): # this is not the correct way to check if alembic has been initialized
|
|
# We need to check if the alembic_version table exists
|
|
# if not, we need to initialize alembic
|
|
with Session(self.engine) as session:
|
|
# If the table does not exist it throws an error
|
|
# so we need to catch it
|
|
try:
|
|
session.execute("SELECT * FROM alembic_version")
|
|
except Exception:
|
|
logger.info("Alembic not initialized")
|
|
try:
|
|
self.init_alembic()
|
|
except Exception as exc:
|
|
logger.error(f"Error initializing alembic: {exc}")
|
|
raise RuntimeError("Error initializing alembic") from exc
|
|
else:
|
|
logger.info("Alembic already initialized")
|
|
|
|
logger.info(f"Running DB migrations in {self.script_location}")
|
|
alembic_cfg = Config()
|
|
alembic_cfg.set_main_option("script_location", str(self.script_location))
|
|
alembic_cfg.set_main_option("sqlalchemy.url", self.database_url)
|
|
command.upgrade(alembic_cfg, "head")
|
|
|
|
def run_migrations_test(self):
|
|
# This method is used for testing purposes only
|
|
# We will check that all models are in the database
|
|
# and that the database is up to date with all columns
|
|
sql_models = [models.Flow, models.User, models.ApiKey]
|
|
return [
|
|
TableResults(sql_model.__tablename__, self.check_table(sql_model))
|
|
for sql_model in sql_models
|
|
]
|
|
|
|
def check_table(self, model):
|
|
results = []
|
|
inspector = inspect(self.engine)
|
|
table_name = model.__tablename__
|
|
expected_columns = list(model.__fields__.keys())
|
|
try:
|
|
available_columns = [
|
|
col["name"] for col in inspector.get_columns(table_name)
|
|
]
|
|
results.append(Result(name=table_name, type="table", success=True))
|
|
except sa.exc.NoSuchTableError:
|
|
logger.error(f"Missing table: {table_name}")
|
|
results.append(Result(name=table_name, type="table", success=False))
|
|
|
|
for column in expected_columns:
|
|
if column not in available_columns:
|
|
logger.error(f"Missing column: {column} in table {table_name}")
|
|
results.append(Result(name=column, type="column", success=False))
|
|
else:
|
|
results.append(Result(name=column, type="column", success=True))
|
|
return results
|
|
|
|
def create_db_and_tables(self):
|
|
from sqlalchemy import inspect
|
|
|
|
inspector = inspect(self.engine)
|
|
table_names = inspector.get_table_names()
|
|
current_tables = ["flow", "user", "apikey"]
|
|
|
|
if table_names and all(table in table_names for table in current_tables):
|
|
logger.debug("Database and tables already exist")
|
|
return
|
|
|
|
logger.debug("Creating database and tables")
|
|
|
|
for table in SQLModel.metadata.sorted_tables:
|
|
try:
|
|
table.create(self.engine, checkfirst=True)
|
|
except OperationalError as oe:
|
|
logger.warning(
|
|
f"Table {table} already exists, skipping. Exception: {oe}"
|
|
)
|
|
except Exception as exc:
|
|
logger.error(f"Error creating table {table}: {exc}")
|
|
raise RuntimeError(f"Error creating table {table}") from exc
|
|
|
|
# Now check if the required tables exist, if not, something went wrong.
|
|
inspector = inspect(self.engine)
|
|
table_names = inspector.get_table_names()
|
|
for table in current_tables:
|
|
if table not in table_names:
|
|
logger.error("Something went wrong creating the database and tables.")
|
|
logger.error("Please check your database settings.")
|
|
raise RuntimeError(
|
|
"Something went wrong creating the database and tables."
|
|
)
|
|
|
|
logger.debug("Database and tables created successfully")
|
|
|
|
def teardown(self):
|
|
logger.debug("Tearing down database")
|
|
try:
|
|
settings_service = get_settings_service()
|
|
# remove the default superuser if auto_login is enabled
|
|
# using the SUPERUSER to get the user
|
|
if settings_service.auth_settings.AUTO_LOGIN:
|
|
logger.debug("Removing default superuser")
|
|
username = settings_service.auth_settings.SUPERUSER
|
|
with Session(self.engine) as session:
|
|
user = get_user_by_username(session, username)
|
|
session.delete(user)
|
|
session.commit()
|
|
logger.debug("Default superuser removed")
|
|
|
|
except Exception as exc:
|
|
logger.error(f"Error tearing down database: {exc}")
|
|
|
|
self.engine.dispose()
|