mirror of
https://github.com/invoke-ai/InvokeAI
synced 2024-08-30 20:32:17 +00:00
e461f9925e
Do the migration on a temp copy of the db, then back up the original and move the temp into its file.
60 lines
2.2 KiB
Python
60 lines
2.2 KiB
Python
import sqlite3
|
|
import threading
|
|
from logging import Logger
|
|
from pathlib import Path
|
|
|
|
from invokeai.app.services.config import InvokeAIAppConfig
|
|
from invokeai.app.services.shared.sqlite.sqlite_common import sqlite_memory
|
|
|
|
|
|
class SqliteDatabase:
|
|
database: Path | str # Must declare this here to satisfy type checker
|
|
|
|
def __init__(self, config: InvokeAIAppConfig, logger: Logger) -> None:
|
|
self.initialize(config, logger)
|
|
|
|
def initialize(self, config: InvokeAIAppConfig, logger: Logger) -> None:
|
|
self._logger = logger
|
|
self._config = config
|
|
self.is_memory = False
|
|
if self._config.use_memory_db:
|
|
self.database = sqlite_memory
|
|
self.is_memory = True
|
|
logger.info("Initializing in-memory database")
|
|
else:
|
|
self.database = self._config.db_path
|
|
self.database.parent.mkdir(parents=True, exist_ok=True)
|
|
self._logger.info(f"Initializing database at {self.database}")
|
|
|
|
self.conn = sqlite3.connect(database=self.database, check_same_thread=False)
|
|
self.lock = threading.RLock()
|
|
self.conn.row_factory = sqlite3.Row
|
|
|
|
if self._config.log_sql:
|
|
self.conn.set_trace_callback(self._logger.debug)
|
|
|
|
self.conn.execute("PRAGMA foreign_keys = ON;")
|
|
|
|
def reinitialize(self) -> None:
|
|
"""Reinitializes the database. Needed after migration."""
|
|
self.initialize(self._config, self._logger)
|
|
|
|
def close(self) -> None:
|
|
self.conn.close()
|
|
|
|
def clean(self) -> None:
|
|
with self.lock:
|
|
try:
|
|
if self.database == sqlite_memory:
|
|
return
|
|
initial_db_size = Path(self.database).stat().st_size
|
|
self.conn.execute("VACUUM;")
|
|
self.conn.commit()
|
|
final_db_size = Path(self.database).stat().st_size
|
|
freed_space_in_mb = round((initial_db_size - final_db_size) / 1024 / 1024, 2)
|
|
if freed_space_in_mb > 0:
|
|
self._logger.info(f"Cleaned database (freed {freed_space_in_mb}MB)")
|
|
except Exception as e:
|
|
self._logger.error(f"Error cleaning database: {e}")
|
|
raise
|