2023-09-24 05:12:51 +00:00
|
|
|
import sqlite3
|
|
|
|
import threading
|
|
|
|
from logging import Logger
|
2023-11-30 09:07:53 +00:00
|
|
|
from pathlib import Path
|
2023-09-24 05:12:51 +00:00
|
|
|
|
|
|
|
from invokeai.app.services.config import InvokeAIAppConfig
|
|
|
|
|
feat: refactor services folder/module structure
Refactor services folder/module structure.
**Motivation**
While working on our services I've repeatedly encountered circular imports and a general lack of clarity regarding where to put things. The structure introduced goes a long way towards resolving those issues, setting us up for a clean structure going forward.
**Services**
Services are now in their own folder with a few files:
- `services/{service_name}/__init__.py`: init as needed, mostly empty now
- `services/{service_name}/{service_name}_base.py`: the base class for the service
- `services/{service_name}/{service_name}_{impl_type}.py`: the default concrete implementation of the service - typically one of `sqlite`, `default`, or `memory`
- `services/{service_name}/{service_name}_common.py`: any common items - models, exceptions, utilities, etc
Though it's a bit verbose to have the service name both as the folder name and the prefix for files, I found it is _extremely_ confusing to have all of the base classes just be named `base.py`. So, at the cost of some verbosity when importing things, I've included the service name in the filename.
There are some minor logic changes. For example, in `InvocationProcessor`, instead of assigning the model manager service to a variable to be used later in the file, the service is used directly via the `Invoker`.
**Shared**
Things that are used across disparate services are in `services/shared/`:
- `default_graphs.py`: previously in `services/`
- `graphs.py`: previously in `services/`
- `paginatation`: generic pagination models used in a few services
- `sqlite`: the `SqliteDatabase` class, other sqlite-specific things
2023-09-24 08:11:07 +00:00
|
|
|
sqlite_memory = ":memory:"
|
|
|
|
|
2023-09-24 05:12:51 +00:00
|
|
|
|
|
|
|
class SqliteDatabase:
|
|
|
|
def __init__(self, config: InvokeAIAppConfig, logger: Logger):
|
|
|
|
self._logger = logger
|
|
|
|
self._config = config
|
|
|
|
|
|
|
|
if self._config.use_memory_db:
|
2023-11-30 09:07:53 +00:00
|
|
|
self.db_path = sqlite_memory
|
2023-09-24 05:12:51 +00:00
|
|
|
logger.info("Using in-memory database")
|
|
|
|
else:
|
|
|
|
db_path = self._config.db_path
|
|
|
|
db_path.parent.mkdir(parents=True, exist_ok=True)
|
2023-11-30 09:07:53 +00:00
|
|
|
self.db_path = str(db_path)
|
|
|
|
self._logger.info(f"Using database at {self.db_path}")
|
2023-09-24 05:12:51 +00:00
|
|
|
|
2023-11-30 09:07:53 +00:00
|
|
|
self.conn = sqlite3.connect(self.db_path, check_same_thread=False)
|
2023-10-16 00:16:41 +00:00
|
|
|
self.lock = threading.RLock()
|
2023-09-24 05:12:51 +00:00
|
|
|
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 clean(self) -> None:
|
|
|
|
try:
|
2023-11-30 09:07:53 +00:00
|
|
|
if self.db_path == sqlite_memory:
|
|
|
|
return
|
|
|
|
initial_db_size = Path(self.db_path).stat().st_size
|
2023-09-24 05:12:51 +00:00
|
|
|
self.lock.acquire()
|
|
|
|
self.conn.execute("VACUUM;")
|
|
|
|
self.conn.commit()
|
2023-11-30 09:07:53 +00:00
|
|
|
final_db_size = Path(self.db_path).stat().st_size
|
|
|
|
freed_space_in_mb = round((initial_db_size - final_db_size) / 1024 / 1024, 2)
|
2023-11-30 22:26:44 +00:00
|
|
|
if freed_space_in_mb > 0:
|
|
|
|
self._logger.info(f"Cleaned database (freed {freed_space_in_mb}MB)")
|
2023-09-24 05:12:51 +00:00
|
|
|
except Exception as e:
|
|
|
|
self._logger.error(f"Error cleaning database: {e}")
|
|
|
|
raise e
|
|
|
|
finally:
|
|
|
|
self.lock.release()
|