1
0
Fork 0
mirror of https://github.com/deepfakes/faceswap synced 2025-06-07 10:43:27 -04:00
faceswap/lib/queue_manager.py
2024-04-03 14:03:54 +01:00

177 lines
6.3 KiB
Python

#!/usr/bin/env python3
""" Queue Manager for faceswap
NB: Keep this in it's own module! If it gets loaded from
a multiprocess on a Windows System it will break Faceswap"""
import logging
import threading
from queue import Queue, Empty as QueueEmpty # pylint: disable=unused-import; # noqa
from time import sleep
logger = logging.getLogger(__name__)
class EventQueue(Queue):
""" Standard Queue object with a separate global shutdown parameter indicating that the main
process, and by extension this queue, should be shut down.
Parameters
----------
shutdown_event: :class:`threading.Event`
The global shutdown event common to all managed queues
maxsize: int, Optional
Upperbound limit on the number of items that can be placed in the queue. Default: `0`
"""
def __init__(self, shutdown_event: threading.Event, maxsize: int = 0) -> None:
super().__init__(maxsize=maxsize)
self._shutdown = shutdown_event
@property
def shutdown(self) -> threading.Event:
""" :class:`threading.Event`: The global shutdown event """
return self._shutdown
class _QueueManager():
""" Manage :class:`EventQueue` objects for availabilty across processes.
Notes
-----
Don't import this class directly, instead import via :func:`queue_manager` """
def __init__(self) -> None:
logger.debug("Initializing %s", self.__class__.__name__)
self.shutdown = threading.Event()
self.queues: dict[str, EventQueue] = {}
logger.debug("Initialized %s", self.__class__.__name__)
def add_queue(self, name: str, maxsize: int = 0, create_new: bool = False) -> str:
""" Add a :class:`EventQueue` to the manager.
Parameters
----------
name: str
The name of the queue to create
maxsize: int, optional
The maximum queue size. Set to `0` for unlimited. Default: `0`
create_new: bool, optional
If a queue of the given name exists, and this value is ``False``, then an error is
raised preventing the creation of duplicate queues. If this value is ``True`` and
the given name exists then an integer is appended to the end of the queue name and
incremented until the given name is unique. Default: ``False``
Returns
-------
str
The final generated name for the queue
"""
logger.debug("QueueManager adding: (name: '%s', maxsize: %s, create_new: %s)",
name, maxsize, create_new)
if not create_new and name in self.queues:
raise ValueError(f"Queue '{name}' already exists.")
if create_new and name in self.queues:
i = 0
while name in self.queues:
name = f"{name}{i}"
logger.debug("Duplicate queue name. Updated to: '%s'", name)
self.queues[name] = EventQueue(self.shutdown, maxsize=maxsize)
logger.debug("QueueManager added: (name: '%s')", name)
return name
def del_queue(self, name: str) -> None:
""" Remove a queue from the manager
Parameters
----------
name: str
The name of the queue to be deleted. Must exist within the queue manager.
"""
logger.debug("QueueManager deleting: '%s'", name)
del self.queues[name]
logger.debug("QueueManager deleted: '%s'", name)
def get_queue(self, name: str, maxsize: int = 0) -> EventQueue:
""" Return a :class:`EventQueue` from the manager. If it doesn't exist, create it.
Parameters
----------
name: str
The name of the queue to obtain
maxsize: int, Optional
The maximum queue size. Set to `0` for unlimited. Only used if the requested queue
does not already exist. Default: `0`
"""
logger.debug("QueueManager getting: '%s'", name)
queue = self.queues.get(name)
if not queue:
self.add_queue(name, maxsize)
queue = self.queues[name]
logger.debug("QueueManager got: '%s'", name)
return queue
def terminate_queues(self) -> None:
""" Terminates all managed queues.
Sets the global shutdown event, clears and send EOF to all queues. To be called if there
is an error """
logger.debug("QueueManager terminating all queues")
self.shutdown.set()
self._flush_queues()
for q_name, queue in self.queues.items():
logger.debug("QueueManager terminating: '%s'", q_name)
queue.put("EOF")
logger.debug("QueueManager terminated all queues")
def _flush_queues(self):
""" Empty out the contents of every managed queue. """
for q_name in self.queues:
self.flush_queue(q_name)
logger.debug("QueueManager flushed all queues")
def flush_queue(self, name: str) -> None:
""" Flush the contents from a managed queue.
Parameters
----------
name: str
The name of the managed :class:`EventQueue` to flush
"""
logger.debug("QueueManager flushing: '%s'", name)
queue = self.queues[name]
while not queue.empty():
queue.get(True, 1)
def debug_monitor(self, update_interval: int = 2) -> None:
""" A debug tool for monitoring managed :class:`EventQueues`.
Prints queue sizes to the console for all managed queues.
Parameters
----------
update_interval: int, Optional
The number of seconds between printing information to the console. Default: 2
"""
thread = threading.Thread(target=self._debug_queue_sizes,
args=(update_interval, ))
thread.daemon = True
thread.start()
def _debug_queue_sizes(self, update_interval) -> None:
""" Print the queue size for each managed queue to console.
Parameters
----------
update_interval: int
The number of seconds between printing information to the console
"""
while True:
logger.info("====================================================")
for name in sorted(self.queues.keys()):
logger.info("%s: %s", name, self.queues[name].qsize())
sleep(update_interval)
queue_manager = _QueueManager() # pylint: disable=invalid-name