2023-08-10 15:38:28 +00:00
|
|
|
from abc import ABC, abstractmethod
|
|
|
|
from typing import cast
|
|
|
|
import uuid
|
|
|
|
import sqlite3
|
|
|
|
import threading
|
|
|
|
from typing import (
|
|
|
|
Any,
|
|
|
|
List,
|
|
|
|
Literal,
|
|
|
|
Optional,
|
|
|
|
Union,
|
|
|
|
)
|
|
|
|
import json
|
|
|
|
|
|
|
|
from invokeai.app.invocations.baseinvocation import (
|
|
|
|
BaseInvocation,
|
|
|
|
)
|
2023-08-10 18:09:00 +00:00
|
|
|
from invokeai.app.services.graph import Graph
|
2023-08-10 15:38:28 +00:00
|
|
|
from invokeai.app.models.image import ImageField
|
|
|
|
|
|
|
|
from pydantic import BaseModel, Field, Extra, parse_raw_as
|
|
|
|
|
|
|
|
invocations = BaseInvocation.get_invocations()
|
|
|
|
InvocationsUnion = Union[invocations] # type: ignore
|
|
|
|
|
|
|
|
BatchDataType = Union[str, int, float, ImageField]
|
|
|
|
|
|
|
|
class Batch(BaseModel):
|
|
|
|
data: list[dict[str, BatchDataType]] = Field(description="Mapping of node field to data value")
|
|
|
|
node_id: str = Field(description="ID of the node to batch")
|
|
|
|
|
|
|
|
|
|
|
|
class BatchSession(BaseModel):
|
|
|
|
batch_id: str = Field(description="Identifier for which batch this Index belongs to")
|
|
|
|
session_id: str = Field(description="Session ID Created for this Batch Index")
|
|
|
|
state: Literal["created", "completed", "inprogress", "error"] = Field(
|
|
|
|
description="Is this session created, completed, in progress, or errored?"
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
class BatchProcess(BaseModel):
|
|
|
|
batch_id: Optional[str] = Field(default_factory=uuid.uuid4().__str__, description="Identifier for this batch")
|
|
|
|
batches: List[Batch] = Field(
|
|
|
|
description="List of batch configs to apply to this session",
|
|
|
|
default_factory=list,
|
|
|
|
)
|
|
|
|
graph: Graph = Field(description="The graph being executed")
|
|
|
|
|
|
|
|
|
|
|
|
class BatchSessionChanges(BaseModel, extra=Extra.forbid):
|
|
|
|
state: Literal["created", "completed", "inprogress", "error"] = Field(
|
|
|
|
description="Is this session created, completed, in progress, or errored?"
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
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 Batch Process record."""
|
|
|
|
pass
|
|
|
|
|
|
|
|
@abstractmethod
|
|
|
|
def save(
|
|
|
|
self,
|
|
|
|
batch_process: BatchProcess,
|
|
|
|
) -> BatchProcess:
|
|
|
|
"""Saves a Batch Process record."""
|
|
|
|
pass
|
|
|
|
|
|
|
|
@abstractmethod
|
|
|
|
def get(
|
|
|
|
self,
|
|
|
|
batch_id: str,
|
|
|
|
) -> BatchProcess:
|
|
|
|
"""Gets a Batch Process record."""
|
|
|
|
pass
|
|
|
|
|
|
|
|
@abstractmethod
|
|
|
|
def create_session(
|
|
|
|
self,
|
|
|
|
session: BatchSession,
|
|
|
|
) -> BatchSession:
|
|
|
|
"""Creates a Batch Session attached to a Batch Process."""
|
|
|
|
pass
|
|
|
|
|
|
|
|
@abstractmethod
|
|
|
|
def get_session(
|
|
|
|
self,
|
|
|
|
session_id: str
|
|
|
|
) -> BatchSession:
|
|
|
|
"""Gets session by session_id"""
|
|
|
|
pass
|
|
|
|
|
|
|
|
@abstractmethod
|
|
|
|
def get_created_session(
|
|
|
|
self,
|
|
|
|
batch_id: str
|
|
|
|
) -> BatchSession:
|
|
|
|
"""Gets all created Batch Sessions for a given Batch Process id."""
|
|
|
|
pass
|
|
|
|
|
|
|
|
@abstractmethod
|
|
|
|
def get_created_sessions(
|
|
|
|
self,
|
|
|
|
batch_id: str
|
|
|
|
) -> List[BatchSession]:
|
|
|
|
"""Gets all created Batch Sessions for a given Batch Process id."""
|
|
|
|
pass
|
|
|
|
|
|
|
|
@abstractmethod
|
|
|
|
def update_session_state(
|
|
|
|
self,
|
|
|
|
batch_id: str,
|
|
|
|
session_id: str,
|
|
|
|
changes: BatchSessionChanges,
|
|
|
|
) -> BatchSession:
|
|
|
|
"""Updates the state of a Batch Session record."""
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
|
|
|
class SqliteBatchProcessStorage(BatchProcessStorageBase):
|
|
|
|
_filename: str
|
|
|
|
_conn: sqlite3.Connection
|
|
|
|
_cursor: sqlite3.Cursor
|
|
|
|
_lock: threading.Lock
|
|
|
|
|
2023-08-10 18:09:00 +00:00
|
|
|
def __init__(self, filename: str, lock: threading.Lock = threading.Lock()) -> None:
|
2023-08-10 15:38:28 +00:00
|
|
|
super().__init__()
|
|
|
|
self._filename = filename
|
|
|
|
self._conn = sqlite3.connect(filename, check_same_thread=False)
|
|
|
|
# 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()
|
2023-08-10 18:09:00 +00:00
|
|
|
self._lock = lock
|
2023-08-10 15:38:28 +00:00
|
|
|
|
|
|
|
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,
|
|
|
|
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,
|
2023-08-10 18:09:00 +00:00
|
|
|
state TEXT NOT NULL,
|
2023-08-10 15:38:28 +00:00
|
|
|
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')
|
2023-08-10 18:09:00 +00:00
|
|
|
WHERE batch_id = old.batch_id AND session_id = old.session_id;
|
2023-08-10 15:38:28 +00:00
|
|
|
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()
|
2023-08-10 18:09:00 +00:00
|
|
|
batches = [batch.json() for batch in batch_process.batches]
|
2023-08-10 15:38:28 +00:00
|
|
|
self._cursor.execute(
|
|
|
|
"""--sql
|
|
|
|
INSERT OR IGNORE INTO batch_process (batch_id, batches, graph)
|
|
|
|
VALUES (?, ?, ?);
|
|
|
|
""",
|
2023-08-10 18:09:00 +00:00
|
|
|
(batch_process.batch_id, json.dumps(batches), batch_process.graph.json()),
|
2023-08-10 15:38:28 +00:00
|
|
|
)
|
|
|
|
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")
|
|
|
|
batches_raw = session_dict.get("batches", "unknown")
|
|
|
|
graph_raw = session_dict.get("graph", "unknown")
|
2023-08-10 18:09:00 +00:00
|
|
|
batches = json.loads(batches_raw)
|
|
|
|
batches = [parse_raw_as(Batch, batch) for batch in batches]
|
2023-08-10 15:38:28 +00:00
|
|
|
return BatchProcess(
|
|
|
|
batch_id=batch_id,
|
2023-08-10 18:09:00 +00:00
|
|
|
batches=batches,
|
2023-08-10 15:38:28 +00:00
|
|
|
graph=parse_raw_as(Graph, graph_raw),
|
|
|
|
)
|
|
|
|
|
|
|
|
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 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
|
2023-08-10 18:09:00 +00:00
|
|
|
return self._deserialize_batch_session(dict(result))
|
2023-08-10 15:38:28 +00:00
|
|
|
|
|
|
|
def _deserialize_batch_session(self, session_dict: dict) -> BatchSession:
|
|
|
|
"""Deserializes a batch session."""
|
|
|
|
|
|
|
|
# Retrieve all the values, setting "reasonable" defaults if they are not present.
|
|
|
|
|
|
|
|
batch_id = session_dict.get("batch_id", "unknown")
|
|
|
|
session_id = session_dict.get("session_id", "unknown")
|
|
|
|
state = session_dict.get("state", "unknown")
|
|
|
|
|
|
|
|
return BatchSession(
|
|
|
|
batch_id=batch_id,
|
|
|
|
session_id=session_id,
|
|
|
|
state=state,
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
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(list[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 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)
|