InvokeAI/invokeai/app/services/invocation_queue.py

69 lines
2.1 KiB
Python
Raw Normal View History

# Copyright (c) 2022 Kyle Schouviller (https://github.com/kyle0654)
2023-04-13 04:23:15 +00:00
import time
from abc import ABC, abstractmethod
from queue import Queue
from pydantic import BaseModel, Field
class InvocationQueueItem(BaseModel):
graph_execution_state_id: str = Field(description="The ID of the graph execution state")
invocation_id: str = Field(description="The ID of the node being invoked")
invoke_all: bool = Field(default=False)
timestamp: float = Field(default_factory=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