InvokeAI/invokeai/app/services/invocation_queue.py

82 lines
2.3 KiB
Python
Raw Normal View History

# Copyright (c) 2022 Kyle Schouviller (https://github.com/kyle0654)
from abc import ABC, abstractmethod
from queue import Queue
2023-03-17 03:05:36 +00:00
import time
# TODO: make this serializable
class InvocationQueueItem:
2023-03-03 06:02:00 +00:00
# session_id: str
graph_execution_state_id: str
invocation_id: str
invoke_all: bool
2023-03-17 03:05:36 +00:00
timestamp: float
2023-03-03 06:02:00 +00:00
def __init__(
self,
# session_id: str,
graph_execution_state_id: str,
invocation_id: str,
2023-03-03 06:02:00 +00:00
invoke_all: bool = False,
):
# self.session_id = session_id
self.graph_execution_state_id = graph_execution_state_id
self.invocation_id = invocation_id
self.invoke_all = invoke_all
2023-03-17 03:05:36 +00:00
self.timestamp = time.time()
class InvocationQueueABC(ABC):
"""Abstract base class for all invocation queues"""
2023-03-03 06:02:00 +00:00
@abstractmethod
def get(self) -> InvocationQueueItem:
pass
2023-03-03 06:02:00 +00:00
@abstractmethod
2023-03-03 06:02:00 +00:00
def put(self, item: InvocationQueueItem | None) -> None:
pass
2023-03-17 03:05:36 +00:00
@abstractmethod
def cancel(self, graph_execution_state_id: str) -> None:
pass
@abstractmethod
def is_canceled(self, graph_execution_state_id: str) -> bool:
pass
class MemoryInvocationQueue(InvocationQueueABC):
__queue: Queue
2023-03-17 03:05:36 +00:00
__cancellations: dict[str, float]
def __init__(self):
self.__queue = Queue()
2023-03-17 03:05:36 +00:00
self.__cancellations = dict()
2023-03-03 06:02:00 +00:00
def get(self) -> InvocationQueueItem:
2023-03-17 03:05:36 +00:00
item = self.__queue.get()
while isinstance(item, InvocationQueueItem) \
and item.graph_execution_state_id in self.__cancellations \
and self.__cancellations[item.graph_execution_state_id] > item.timestamp:
item = self.__queue.get()
# Clear old items
for graph_execution_state_id in list(self.__cancellations.keys()):
if self.__cancellations[graph_execution_state_id] < item.timestamp:
del self.__cancellations[graph_execution_state_id]
return item
2023-03-03 06:02:00 +00:00
def put(self, item: InvocationQueueItem | None) -> None:
self.__queue.put(item)
2023-03-17 03:05:36 +00:00
def cancel(self, graph_execution_state_id: str) -> None:
if graph_execution_state_id not in self.__cancellations:
self.__cancellations[graph_execution_state_id] = time.time()
def is_canceled(self, graph_execution_state_id: str) -> bool:
return graph_execution_state_id in self.__cancellations