2023-06-29 06:01:17 +00:00
|
|
|
from .test_nodes import (
|
|
|
|
TestEventService,
|
|
|
|
ErrorInvocation,
|
|
|
|
TextToImageTestInvocation,
|
|
|
|
PromptTestInvocation,
|
|
|
|
create_edge,
|
|
|
|
wait_until,
|
|
|
|
)
|
|
|
|
from invokeai.app.services.invocation_queue import MemoryInvocationQueue
|
|
|
|
from invokeai.app.services.processor import DefaultInvocationProcessor
|
|
|
|
from invokeai.app.services.sqlite import SqliteItemStorage, sqlite_memory
|
2023-06-26 15:55:24 +00:00
|
|
|
from invokeai.app.services.invoker import Invoker
|
2023-06-29 06:01:17 +00:00
|
|
|
from invokeai.app.services.invocation_services import InvocationServices
|
2023-08-02 22:31:10 +00:00
|
|
|
from invokeai.app.services.invocation_stats import InvocationStatsService
|
2023-08-16 19:33:15 +00:00
|
|
|
from invokeai.app.services.batch_manager_storage import BatchData, SqliteBatchProcessStorage
|
2023-08-14 14:57:18 +00:00
|
|
|
from invokeai.app.services.batch_manager import (
|
|
|
|
Batch,
|
|
|
|
BatchManager,
|
|
|
|
)
|
2023-06-29 06:01:17 +00:00
|
|
|
from invokeai.app.services.graph import (
|
|
|
|
Graph,
|
|
|
|
GraphExecutionState,
|
|
|
|
LibraryGraph,
|
|
|
|
)
|
|
|
|
import pytest
|
2023-08-16 18:35:49 +00:00
|
|
|
import sqlite3
|
2023-06-26 15:55:24 +00:00
|
|
|
|
2022-12-01 05:33:20 +00:00
|
|
|
|
|
|
|
@pytest.fixture
|
|
|
|
def simple_graph():
|
|
|
|
g = Graph()
|
2023-06-29 06:01:17 +00:00
|
|
|
g.add_node(PromptTestInvocation(id="1", prompt="Banana sushi"))
|
|
|
|
g.add_node(TextToImageTestInvocation(id="2"))
|
2022-12-01 05:33:20 +00:00
|
|
|
g.add_edge(create_edge("1", "prompt", "2", "prompt"))
|
|
|
|
return g
|
|
|
|
|
2023-06-29 06:01:17 +00:00
|
|
|
|
2023-08-14 14:57:18 +00:00
|
|
|
@pytest.fixture
|
2023-08-16 19:33:15 +00:00
|
|
|
def simple_batch():
|
|
|
|
return Batch(
|
|
|
|
data=[
|
|
|
|
[
|
|
|
|
BatchData(
|
|
|
|
node_id="1",
|
|
|
|
field_name="prompt",
|
|
|
|
items=[
|
|
|
|
"Tomato sushi",
|
|
|
|
"Strawberry sushi",
|
|
|
|
"Broccoli sushi",
|
|
|
|
"Asparagus sushi",
|
|
|
|
"Tea sushi",
|
|
|
|
],
|
|
|
|
)
|
2023-08-14 15:01:31 +00:00
|
|
|
],
|
2023-08-16 19:33:15 +00:00
|
|
|
[
|
|
|
|
BatchData(
|
|
|
|
node_id="2",
|
|
|
|
field_name="prompt",
|
|
|
|
items=[
|
|
|
|
"Ume sushi",
|
|
|
|
"Ichigo sushi",
|
|
|
|
"Momo sushi",
|
|
|
|
"Mikan sushi",
|
|
|
|
"Cha sushi",
|
|
|
|
],
|
|
|
|
)
|
2023-08-14 15:01:31 +00:00
|
|
|
],
|
2023-08-16 19:33:15 +00:00
|
|
|
]
|
|
|
|
)
|
2023-08-14 14:57:18 +00:00
|
|
|
|
|
|
|
|
2023-06-29 06:01:17 +00:00
|
|
|
# This must be defined here to avoid issues with the dynamic creation of the union of all invocation types
|
|
|
|
# Defining it in a separate module will cause the union to be incomplete, and pydantic will not validate
|
|
|
|
# the test invocations.
|
|
|
|
@pytest.fixture
|
|
|
|
def mock_services() -> InvocationServices:
|
|
|
|
# NOTE: none of these are actually called by the test invocations
|
2023-08-16 18:35:49 +00:00
|
|
|
db_conn = sqlite3.connect(sqlite_memory, check_same_thread=False)
|
2023-08-16 19:33:15 +00:00
|
|
|
graph_execution_manager = SqliteItemStorage[GraphExecutionState](conn=db_conn, table_name="graph_executions")
|
2023-08-16 18:35:49 +00:00
|
|
|
batch_manager_storage = SqliteBatchProcessStorage(conn=db_conn)
|
2023-06-29 06:01:17 +00:00
|
|
|
return InvocationServices(
|
|
|
|
model_manager=None, # type: ignore
|
|
|
|
events=TestEventService(),
|
|
|
|
logger=None, # type: ignore
|
|
|
|
images=None, # type: ignore
|
|
|
|
latents=None, # type: ignore
|
2023-08-14 14:57:18 +00:00
|
|
|
batch_manager=BatchManager(batch_manager_storage),
|
2023-06-29 06:01:17 +00:00
|
|
|
boards=None, # type: ignore
|
|
|
|
board_images=None, # type: ignore
|
|
|
|
queue=MemoryInvocationQueue(),
|
2023-08-16 18:35:49 +00:00
|
|
|
graph_library=SqliteItemStorage[LibraryGraph](conn=db_conn, table_name="graphs"),
|
2023-08-02 22:31:10 +00:00
|
|
|
graph_execution_manager=graph_execution_manager,
|
2023-06-29 06:01:17 +00:00
|
|
|
processor=DefaultInvocationProcessor(),
|
2023-08-02 22:31:10 +00:00
|
|
|
performance_statistics=InvocationStatsService(graph_execution_manager),
|
2023-06-29 06:01:17 +00:00
|
|
|
configuration=None, # type: ignore
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2022-12-01 05:33:20 +00:00
|
|
|
@pytest.fixture()
|
2023-02-25 04:11:28 +00:00
|
|
|
def mock_invoker(mock_services: InvocationServices) -> Invoker:
|
2023-06-29 06:01:17 +00:00
|
|
|
return Invoker(services=mock_services)
|
|
|
|
|
2022-12-01 05:33:20 +00:00
|
|
|
|
|
|
|
def test_can_create_graph_state(mock_invoker: Invoker):
|
|
|
|
g = mock_invoker.create_execution_state()
|
|
|
|
mock_invoker.stop()
|
|
|
|
|
|
|
|
assert g is not None
|
|
|
|
assert isinstance(g, GraphExecutionState)
|
|
|
|
|
2023-06-29 06:01:17 +00:00
|
|
|
|
2022-12-01 05:33:20 +00:00
|
|
|
def test_can_create_graph_state_from_graph(mock_invoker: Invoker, simple_graph):
|
2023-06-29 06:01:17 +00:00
|
|
|
g = mock_invoker.create_execution_state(graph=simple_graph)
|
2022-12-01 05:33:20 +00:00
|
|
|
mock_invoker.stop()
|
|
|
|
|
|
|
|
assert g is not None
|
|
|
|
assert isinstance(g, GraphExecutionState)
|
|
|
|
assert g.graph == simple_graph
|
|
|
|
|
2023-06-29 06:01:17 +00:00
|
|
|
|
|
|
|
# @pytest.mark.xfail(reason = "Requires fixing following the model manager refactor")
|
2022-12-01 05:33:20 +00:00
|
|
|
def test_can_invoke(mock_invoker: Invoker, simple_graph):
|
2023-06-29 06:01:17 +00:00
|
|
|
g = mock_invoker.create_execution_state(graph=simple_graph)
|
2022-12-01 05:33:20 +00:00
|
|
|
invocation_id = mock_invoker.invoke(g)
|
|
|
|
assert invocation_id is not None
|
|
|
|
|
|
|
|
def has_executed_any(g: GraphExecutionState):
|
2023-02-25 04:11:28 +00:00
|
|
|
g = mock_invoker.services.graph_execution_manager.get(g.id)
|
2022-12-01 05:33:20 +00:00
|
|
|
return len(g.executed) > 0
|
|
|
|
|
2023-06-29 06:01:17 +00:00
|
|
|
wait_until(lambda: has_executed_any(g), timeout=5, interval=1)
|
2022-12-01 05:33:20 +00:00
|
|
|
mock_invoker.stop()
|
|
|
|
|
2023-02-25 04:11:28 +00:00
|
|
|
g = mock_invoker.services.graph_execution_manager.get(g.id)
|
2022-12-01 05:33:20 +00:00
|
|
|
assert len(g.executed) > 0
|
|
|
|
|
2023-06-29 06:01:17 +00:00
|
|
|
|
|
|
|
# @pytest.mark.xfail(reason = "Requires fixing following the model manager refactor")
|
2022-12-01 05:33:20 +00:00
|
|
|
def test_can_invoke_all(mock_invoker: Invoker, simple_graph):
|
2023-06-29 06:01:17 +00:00
|
|
|
g = mock_invoker.create_execution_state(graph=simple_graph)
|
|
|
|
invocation_id = mock_invoker.invoke(g, invoke_all=True)
|
2022-12-01 05:33:20 +00:00
|
|
|
assert invocation_id is not None
|
|
|
|
|
|
|
|
def has_executed_all(g: GraphExecutionState):
|
2023-02-25 04:11:28 +00:00
|
|
|
g = mock_invoker.services.graph_execution_manager.get(g.id)
|
2022-12-01 05:33:20 +00:00
|
|
|
return g.is_complete()
|
|
|
|
|
2023-06-29 06:01:17 +00:00
|
|
|
wait_until(lambda: has_executed_all(g), timeout=5, interval=1)
|
2022-12-01 05:33:20 +00:00
|
|
|
mock_invoker.stop()
|
|
|
|
|
2023-02-25 04:11:28 +00:00
|
|
|
g = mock_invoker.services.graph_execution_manager.get(g.id)
|
2022-12-01 05:33:20 +00:00
|
|
|
assert g.is_complete()
|
2023-02-27 18:01:07 +00:00
|
|
|
|
2023-06-29 06:01:17 +00:00
|
|
|
|
|
|
|
# @pytest.mark.xfail(reason = "Requires fixing following the model manager refactor")
|
2023-02-27 18:01:07 +00:00
|
|
|
def test_handles_errors(mock_invoker: Invoker):
|
|
|
|
g = mock_invoker.create_execution_state()
|
2023-06-29 06:01:17 +00:00
|
|
|
g.graph.add_node(ErrorInvocation(id="1"))
|
2023-02-27 18:01:07 +00:00
|
|
|
|
|
|
|
mock_invoker.invoke(g, invoke_all=True)
|
|
|
|
|
|
|
|
def has_executed_all(g: GraphExecutionState):
|
|
|
|
g = mock_invoker.services.graph_execution_manager.get(g.id)
|
|
|
|
return g.is_complete()
|
|
|
|
|
2023-06-29 06:01:17 +00:00
|
|
|
wait_until(lambda: has_executed_all(g), timeout=5, interval=1)
|
2023-02-27 18:01:07 +00:00
|
|
|
mock_invoker.stop()
|
|
|
|
|
|
|
|
g = mock_invoker.services.graph_execution_manager.get(g.id)
|
|
|
|
assert g.has_error()
|
|
|
|
assert g.is_complete()
|
|
|
|
|
2023-06-29 06:01:17 +00:00
|
|
|
assert all((i in g.errors for i in g.source_prepared_mapping["1"]))
|
2023-08-14 14:57:18 +00:00
|
|
|
|
2023-08-14 15:01:31 +00:00
|
|
|
|
2023-08-16 19:33:15 +00:00
|
|
|
def test_can_create_batch(mock_invoker: Invoker, simple_graph, simple_batch):
|
2023-08-14 14:57:18 +00:00
|
|
|
batch_process_res = mock_invoker.services.batch_manager.create_batch_process(
|
2023-08-16 19:33:15 +00:00
|
|
|
batch=simple_batch,
|
2023-08-14 14:57:18 +00:00
|
|
|
graph=simple_graph,
|
|
|
|
)
|
|
|
|
assert batch_process_res.batch_id
|
|
|
|
assert len(batch_process_res.session_ids) == 25
|
2023-08-21 15:48:53 +00:00
|
|
|
# TODO: without the mock events service emitting the `graph_execution_state` events,
|
|
|
|
# the batch sessions do not know when they have finished, so this logic will fail
|
|
|
|
|
|
|
|
# mock_invoker.services.batch_manager.run_batch_process(batch_process_res.batch_id)
|
|
|
|
|
|
|
|
# def has_executed_all_batches(batch_id: str):
|
|
|
|
# batch_sessions = mock_invoker.services.batch_manager.get_sessions(batch_id)
|
|
|
|
# print(batch_sessions)
|
|
|
|
# return all((s.state == "completed" for s in batch_sessions))
|
|
|
|
|
|
|
|
# wait_until(lambda: has_executed_all_batches(batch_process_res.batch_id), timeout=10, interval=1)
|
2023-08-16 19:42:45 +00:00
|
|
|
|
|
|
|
|
|
|
|
def test_can_create_bad_batches():
|
|
|
|
batch = None
|
|
|
|
try:
|
|
|
|
batch = Batch( # This batch has a duplicate node_id|fieldname combo
|
|
|
|
data=[
|
|
|
|
[
|
|
|
|
BatchData(
|
|
|
|
node_id="1",
|
|
|
|
field_name="prompt",
|
|
|
|
items=[
|
|
|
|
"Tomato sushi",
|
|
|
|
],
|
|
|
|
)
|
|
|
|
],
|
|
|
|
[
|
|
|
|
BatchData(
|
|
|
|
node_id="1",
|
|
|
|
field_name="prompt",
|
|
|
|
items=[
|
|
|
|
"Ume sushi",
|
|
|
|
],
|
|
|
|
)
|
|
|
|
],
|
|
|
|
]
|
|
|
|
)
|
|
|
|
except Exception as e:
|
|
|
|
assert e
|
|
|
|
try:
|
|
|
|
batch = Batch( # This batch has different item list lengths in the same group
|
|
|
|
data=[
|
|
|
|
[
|
|
|
|
BatchData(
|
|
|
|
node_id="1",
|
|
|
|
field_name="prompt",
|
|
|
|
items=[
|
|
|
|
"Tomato sushi",
|
|
|
|
],
|
|
|
|
),
|
|
|
|
BatchData(
|
|
|
|
node_id="1",
|
|
|
|
field_name="prompt",
|
|
|
|
items=[
|
|
|
|
"Tomato sushi",
|
|
|
|
"Courgette sushi",
|
|
|
|
],
|
|
|
|
),
|
|
|
|
],
|
|
|
|
[
|
|
|
|
BatchData(
|
|
|
|
node_id="1",
|
|
|
|
field_name="prompt",
|
|
|
|
items=[
|
|
|
|
"Ume sushi",
|
|
|
|
],
|
|
|
|
)
|
|
|
|
],
|
|
|
|
]
|
|
|
|
)
|
|
|
|
except Exception as e:
|
|
|
|
assert e
|
|
|
|
try:
|
|
|
|
batch = Batch( # This batch has a type mismatch in single items list
|
|
|
|
data=[
|
|
|
|
[
|
|
|
|
BatchData(
|
|
|
|
node_id="1",
|
|
|
|
field_name="prompt",
|
|
|
|
items=["Tomato sushi", 5],
|
|
|
|
),
|
|
|
|
],
|
|
|
|
]
|
|
|
|
)
|
|
|
|
except Exception as e:
|
|
|
|
assert e
|
|
|
|
assert not batch
|