import sqlite3 import threading import uuid from abc import ABC, abstractmethod from typing import List, Literal, Optional, Union, cast from pydantic import BaseModel, Extra, Field, StrictFloat, StrictInt, StrictStr, parse_raw_as, validator from invokeai.app.services.image_record_storage import OffsetPaginatedResults from invokeai.app.invocations.primitives import ImageField from invokeai.app.services.graph import Graph BatchDataType = Union[StrictStr, StrictInt, StrictFloat, ImageField] class BatchData(BaseModel): """ A batch data collection. """ node_id: str = Field(description="The node into which this batch data collection will be substituted.") field_name: str = Field(description="The field into which this batch data collection will be substituted.") items: list[BatchDataType] = Field( default_factory=list, description="The list of items to substitute into the node/field." ) class Batch(BaseModel): """ A batch, consisting of a list of a list of batch data collections. First, each inner list[BatchData] is zipped into a single batch data collection. Then, the final batch collection is created by taking the Cartesian product of all batch data collections. """ data: list[list[BatchData]] = Field(default_factory=list, description="The list of batch data collections.") runs: int = Field(default=1, description="Int stating how many times to iterate through all possible batch indices") @validator("runs") def validate_positive_runs(cls, r: int): if r < 1: raise ValueError("runs must be a positive integer") return r @validator("data") def validate_len(cls, v: list[list[BatchData]]): for batch_data in v: if any(len(batch_data[0].items) != len(i.items) for i in batch_data): raise ValueError("Zipped batch items must have all have same length") return v @validator("data") def validate_types(cls, v: list[list[BatchData]]): for batch_data in v: for datum in batch_data: for item in datum.items: if not all(isinstance(item, type(i)) for i in datum.items): raise TypeError("All items in a batch must have have same type") return v @validator("data") def validate_unique_field_mappings(cls, v: list[list[BatchData]]): paths: set[tuple[str, str]] = set() count: int = 0 for batch_data in v: for datum in batch_data: paths.add((datum.node_id, datum.field_name)) count += 1 if len(paths) != count: raise ValueError("Each batch data must have unique node_id and field_name") return v class BatchSession(BaseModel): batch_id: str = Field(description="The Batch to which this BatchSession is attached.") session_id: str = Field(description="The Session to which this BatchSession is attached.") state: Literal["created", "completed", "inprogress", "error"] = Field(description="The state of this BatchSession") def uuid_string(): res = uuid.uuid4() return str(res) class BatchProcess(BaseModel): batch_id: str = Field(default_factory=uuid_string, description="Identifier for this batch.") batch: Batch = Field(description="The Batch to apply to this session.") canceled: bool = Field(description="Whether or not to run sessions from this batch.", default=False) graph: Graph = Field(description="The graph into which batch data will be inserted before being executed.") class BatchSessionChanges(BaseModel, extra=Extra.forbid): state: Literal["created", "completed", "inprogress", "error"] = Field(description="The state of this BatchSession") class BatchProcessNotFoundException(Exception): """Raised when an Batch Process record is not found.""" def __init__(self, message="BatchProcess record not found"): super().__init__(message) class BatchProcessSaveException(Exception): """Raised when an Batch Process record cannot be saved.""" def __init__(self, message="BatchProcess record not saved"): super().__init__(message) class BatchProcessDeleteException(Exception): """Raised when an Batch Process record cannot be deleted.""" def __init__(self, message="BatchProcess record not deleted"): super().__init__(message) class BatchSessionNotFoundException(Exception): """Raised when an Batch Session record is not found.""" def __init__(self, message="BatchSession record not found"): super().__init__(message) class BatchSessionSaveException(Exception): """Raised when an Batch Session record cannot be saved.""" def __init__(self, message="BatchSession record not saved"): super().__init__(message) class BatchSessionDeleteException(Exception): """Raised when an Batch Session record cannot be deleted.""" def __init__(self, message="BatchSession record not deleted"): super().__init__(message) class BatchProcessStorageBase(ABC): """Low-level service responsible for interfacing with the Batch Process record store.""" @abstractmethod def delete(self, batch_id: str) -> None: """Deletes a BatchProcess record.""" pass @abstractmethod def save( self, batch_process: BatchProcess, ) -> BatchProcess: """Saves a BatchProcess record.""" pass @abstractmethod def get( self, batch_id: str, ) -> BatchProcess: """Gets a BatchProcess record.""" pass @abstractmethod def get_all( self, ) -> list[BatchProcess]: """Gets a BatchProcess record.""" pass @abstractmethod def get_incomplete( self, ) -> list[BatchProcess]: """Gets a BatchProcess record.""" pass @abstractmethod def start( self, batch_id: str, ) -> None: """'Starts' a BatchProcess record by marking its `canceled` attribute to False.""" pass @abstractmethod def cancel( self, batch_id: str, ) -> None: """'Cancels' a BatchProcess record by setting its `canceled` attribute to True.""" pass @abstractmethod def create_session( self, session: BatchSession, ) -> BatchSession: """Creates a BatchSession attached to a BatchProcess.""" pass @abstractmethod def get_session(self, session_id: str) -> BatchSession: """Gets a BatchSession by session_id""" pass @abstractmethod def get_sessions(self, batch_id: str) -> List[BatchSession]: """Gets all BatchSession's for a given BatchProcess id.""" pass @abstractmethod def get_created_session(self, batch_id: str) -> BatchSession: """Gets the latest BatchSession with state `created`, for a given BatchProcess id.""" pass @abstractmethod def get_created_sessions(self, batch_id: str) -> List[BatchSession]: """Gets all BatchSession's with state `created`, for a given BatchProcess id.""" pass @abstractmethod def update_session_state( self, batch_id: str, session_id: str, changes: BatchSessionChanges, ) -> BatchSession: """Updates the state of a BatchSession record.""" pass class SqliteBatchProcessStorage(BatchProcessStorageBase): _conn: sqlite3.Connection _cursor: sqlite3.Cursor _lock: threading.Lock def __init__(self, conn: sqlite3.Connection) -> None: super().__init__() self._conn = conn # Enable row factory to get rows as dictionaries (must be done before making the cursor!) self._conn.row_factory = sqlite3.Row self._cursor = self._conn.cursor() self._lock = threading.Lock() try: self._lock.acquire() # Enable foreign keys self._conn.execute("PRAGMA foreign_keys = ON;") self._create_tables() self._conn.commit() finally: self._lock.release() def _create_tables(self) -> None: """Creates the `batch_process` table and `batch_session` junction table.""" # Create the `batch_process` table. self._cursor.execute( """--sql CREATE TABLE IF NOT EXISTS batch_process ( batch_id TEXT NOT NULL PRIMARY KEY, batches TEXT NOT NULL, graph TEXT NOT NULL, canceled BOOLEAN NOT NULL DEFAULT(0), created_at DATETIME NOT NULL DEFAULT(STRFTIME('%Y-%m-%d %H:%M:%f', 'NOW')), -- Updated via trigger updated_at DATETIME NOT NULL DEFAULT(STRFTIME('%Y-%m-%d %H:%M:%f', 'NOW')), -- Soft delete, currently unused deleted_at DATETIME ); """ ) self._cursor.execute( """--sql CREATE INDEX IF NOT EXISTS idx_batch_process_created_at ON batch_process (created_at); """ ) # Add trigger for `updated_at`. self._cursor.execute( """--sql CREATE TRIGGER IF NOT EXISTS tg_batch_process_updated_at AFTER UPDATE ON batch_process FOR EACH ROW BEGIN UPDATE batch_process SET updated_at = current_timestamp WHERE batch_id = old.batch_id; END; """ ) # Create the `batch_session` junction table. self._cursor.execute( """--sql CREATE TABLE IF NOT EXISTS batch_session ( batch_id TEXT NOT NULL, session_id TEXT NOT NULL, state TEXT NOT NULL, created_at DATETIME NOT NULL DEFAULT(STRFTIME('%Y-%m-%d %H:%M:%f', 'NOW')), -- updated via trigger updated_at DATETIME NOT NULL DEFAULT(STRFTIME('%Y-%m-%d %H:%M:%f', 'NOW')), -- Soft delete, currently unused deleted_at DATETIME, -- enforce one-to-many relationship between batch_process and batch_session using PK -- (we can extend this to many-to-many later) PRIMARY KEY (batch_id,session_id), FOREIGN KEY (batch_id) REFERENCES batch_process (batch_id) ON DELETE CASCADE ); """ ) # Add index for batch id self._cursor.execute( """--sql CREATE INDEX IF NOT EXISTS idx_batch_session_batch_id ON batch_session (batch_id); """ ) # Add index for batch id, sorted by created_at self._cursor.execute( """--sql CREATE INDEX IF NOT EXISTS idx_batch_session_batch_id_created_at ON batch_session (batch_id,created_at); """ ) # Add trigger for `updated_at`. self._cursor.execute( """--sql CREATE TRIGGER IF NOT EXISTS tg_batch_session_updated_at AFTER UPDATE ON batch_session FOR EACH ROW BEGIN UPDATE batch_session SET updated_at = STRFTIME('%Y-%m-%d %H:%M:%f', 'NOW') WHERE batch_id = old.batch_id AND session_id = old.session_id; END; """ ) def delete(self, batch_id: str) -> None: try: self._lock.acquire() self._cursor.execute( """--sql DELETE FROM batch_process WHERE batch_id = ?; """, (batch_id,), ) self._conn.commit() except sqlite3.Error as e: self._conn.rollback() raise BatchProcessDeleteException from e except Exception as e: self._conn.rollback() raise BatchProcessDeleteException from e finally: self._lock.release() def save( self, batch_process: BatchProcess, ) -> BatchProcess: try: self._lock.acquire() self._cursor.execute( """--sql INSERT OR IGNORE INTO batch_process (batch_id, batches, graph) VALUES (?, ?, ?); """, (batch_process.batch_id, batch_process.batch.json(), batch_process.graph.json()), ) self._conn.commit() except sqlite3.Error as e: self._conn.rollback() raise BatchProcessSaveException from e finally: self._lock.release() return self.get(batch_process.batch_id) def _deserialize_batch_process(self, session_dict: dict) -> BatchProcess: """Deserializes a batch session.""" # Retrieve all the values, setting "reasonable" defaults if they are not present. batch_id = session_dict.get("batch_id", "unknown") batch_raw = session_dict.get("batches", "unknown") graph_raw = session_dict.get("graph", "unknown") canceled = session_dict.get("canceled", 0) return BatchProcess( batch_id=batch_id, batch=parse_raw_as(Batch, batch_raw), graph=parse_raw_as(Graph, graph_raw), canceled=canceled == 1, ) def get( self, batch_id: str, ) -> BatchProcess: try: self._lock.acquire() self._cursor.execute( """--sql SELECT * FROM batch_process WHERE batch_id = ?; """, (batch_id,), ) result = cast(Union[sqlite3.Row, None], self._cursor.fetchone()) except sqlite3.Error as e: self._conn.rollback() raise BatchProcessNotFoundException from e finally: self._lock.release() if result is None: raise BatchProcessNotFoundException return self._deserialize_batch_process(dict(result)) def get_all( self, ) -> list[BatchProcess]: try: self._lock.acquire() self._cursor.execute( """--sql SELECT * FROM batch_process """ ) result = cast(list[sqlite3.Row], self._cursor.fetchall()) except sqlite3.Error as e: self._conn.rollback() raise BatchProcessNotFoundException from e finally: self._lock.release() if result is None: return list() return list(map(lambda r: self._deserialize_batch_process(dict(r)), result)) def get_incomplete( self, ) -> list[BatchProcess]: try: self._lock.acquire() self._cursor.execute( """--sql SELECT bp.* FROM batch_process bp WHERE bp.batch_id IN ( SELECT batch_id FROM batch_session bs WHERE state = 'created' ); """ ) result = cast(list[sqlite3.Row], self._cursor.fetchall()) except sqlite3.Error as e: self._conn.rollback() raise BatchProcessNotFoundException from e finally: self._lock.release() if result is None: return list() return list(map(lambda r: self._deserialize_batch_process(dict(r)), result)) def start( self, batch_id: str, ) -> None: try: self._lock.acquire() self._cursor.execute( f"""--sql UPDATE batch_process SET canceled = 0 WHERE batch_id = ?; """, (batch_id,), ) self._conn.commit() except sqlite3.Error as e: self._conn.rollback() raise BatchSessionSaveException from e finally: self._lock.release() def cancel( self, batch_id: str, ) -> None: try: self._lock.acquire() self._cursor.execute( f"""--sql UPDATE batch_process SET canceled = 1 WHERE batch_id = ?; """, (batch_id,), ) self._conn.commit() except sqlite3.Error as e: self._conn.rollback() raise BatchSessionSaveException from e finally: self._lock.release() def create_session( self, session: BatchSession, ) -> BatchSession: try: self._lock.acquire() self._cursor.execute( """--sql INSERT OR IGNORE INTO batch_session (batch_id, session_id, state) VALUES (?, ?, ?); """, (session.batch_id, session.session_id, session.state), ) self._conn.commit() except sqlite3.Error as e: self._conn.rollback() raise BatchSessionSaveException from e finally: self._lock.release() return self.get_session(session.session_id) def get_session(self, session_id: str) -> BatchSession: try: self._lock.acquire() self._cursor.execute( """--sql SELECT * FROM batch_session WHERE session_id= ?; """, (session_id,), ) result = cast(Union[sqlite3.Row, None], self._cursor.fetchone()) except sqlite3.Error as e: self._conn.rollback() raise BatchSessionNotFoundException from e finally: self._lock.release() if result is None: raise BatchSessionNotFoundException return self._deserialize_batch_session(dict(result)) def _deserialize_batch_session(self, session_dict: dict) -> BatchSession: """Deserializes a batch session.""" return BatchSession.parse_obj(session_dict) def get_created_session(self, batch_id: str) -> BatchSession: try: self._lock.acquire() self._cursor.execute( """--sql SELECT * FROM batch_session WHERE batch_id = ? AND state = 'created'; """, (batch_id,), ) result = cast(Optional[sqlite3.Row], self._cursor.fetchone()) except sqlite3.Error as e: self._conn.rollback() raise BatchSessionNotFoundException from e finally: self._lock.release() if result is None: raise BatchSessionNotFoundException session = self._deserialize_batch_session(dict(result)) return session def get_created_sessions(self, batch_id: str) -> List[BatchSession]: try: self._lock.acquire() self._cursor.execute( """--sql SELECT * FROM batch_session WHERE batch_id = ? AND state = created; """, (batch_id,), ) result = cast(list[sqlite3.Row], self._cursor.fetchall()) except sqlite3.Error as e: self._conn.rollback() raise BatchSessionNotFoundException from e finally: self._lock.release() if result is None: raise BatchSessionNotFoundException sessions = list(map(lambda r: self._deserialize_batch_session(dict(r)), result)) return sessions def get_sessions(self, batch_id: str) -> List[BatchSession]: try: self._lock.acquire() self._cursor.execute( """--sql SELECT * FROM batch_session WHERE batch_id = ?; """, (batch_id,), ) result = cast(list[sqlite3.Row], self._cursor.fetchall()) except sqlite3.Error as e: self._conn.rollback() raise BatchSessionNotFoundException from e finally: self._lock.release() if result is None: raise BatchSessionNotFoundException sessions = list(map(lambda r: self._deserialize_batch_session(dict(r)), result)) return sessions def update_session_state( self, batch_id: str, session_id: str, changes: BatchSessionChanges, ) -> BatchSession: try: self._lock.acquire() # Change the state of a batch session if changes.state is not None: self._cursor.execute( f"""--sql UPDATE batch_session SET state = ? WHERE batch_id = ? AND session_id = ?; """, (changes.state, batch_id, session_id), ) self._conn.commit() except sqlite3.Error as e: self._conn.rollback() raise BatchSessionSaveException from e finally: self._lock.release() return self.get_session(session_id)