Newer
Older
# Copyright 2022-2023 ETSI TeraFlowSDN - TFS OSG (https://tfs.etsi.org/)
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
# MutexQueues:
# ------------
# This class enables to schedule and serialize operations concurrently issued
# over a number of resources. For instance, when multiple components want to
# configure devices through the Device component, configuration operations
# have to be serialized to prevent data corruptions, and race conditions, etc.
# Usage Example:
# class Servicer():
# def __init__(self):
# # init other stuff
# self.drivers = dict()
# self.mutex_queues = MutexQueues()
#
# def configure_device(self, device_uuid, settings):
# self.mutex_queues.wait_my_turn(device_uuid)
# driver = self.drivers.get(device_uuid)
# if driver is None:
# driver = Driver(device_uuid)
# self.drivers[device_uuid] = driver
# driver.configure(settings)
# self.mutex_queues.signal_done(device_uuid)
class MutexQueues:
def __init__(self) -> None:
# lock to protect dictionary updates
self.lock = threading.Lock()
# dictionaty of queues of mutexes: queue_name => queue[mutex]
# first mutex is the running one
self.mutex_queues : Dict[str, Queue[threading.Event]] = dict()
def add_alias(self, queue_name_a : str, queue_name_b : str) -> None:
with self.lock:
if queue_name_a in self.mutex_queues and queue_name_b not in self.mutex_queues:
self.mutex_queues[queue_name_b] = self.mutex_queues[queue_name_a]
elif queue_name_b in self.mutex_queues and queue_name_a not in self.mutex_queues:
self.mutex_queues[queue_name_a] = self.mutex_queues[queue_name_b]
LOGGER.warning('[wait_my_turn] begin queue_name={:s}'.format(str(queue_name)))
# create my mutex and enqueue it
mutex = threading.Event()
LOGGER.warning('[wait_my_turn] [lock] queue_name={:s} mutex={:s}'.format(str(queue_name), str(mutex)))
LOGGER.warning('[wait_my_turn] [lock] queue_name={:s} mutex_queues={:s}'.format(str(queue_name), str(self.mutex_queues)))
queue : Queue = self.mutex_queues.setdefault(queue_name, Queue())
first_in_queue = (queue.qsize() == 0)
LOGGER.warning('[wait_my_turn] [lock] queue_name={:s} first_in_queue={:s}'.format(str(queue_name), str(first_in_queue)))
queue.put_nowait(mutex)
# if I'm the first in the queue upon addition, means there are no running tasks
# directly return without waiting
if first_in_queue:
LOGGER.warning('[wait_my_turn] end first_in_queue queue_name={:s}'.format(str(queue_name)))
return
LOGGER.warning('[wait_my_turn] waiting queue_name={:s}'.format(str(queue_name)))
LOGGER.warning('[wait_my_turn] end wait queue_name={:s}'.format(str(queue_name)))
def signal_done(self, queue_name : str) -> None:
LOGGER.warning('[signal_done] begin queue_name={:s}'.format(str(queue_name)))
LOGGER.warning('[wait_my_turn] [lock] queue_name={:s} mutex_queues={:s}'.format(str(queue_name), str(self.mutex_queues)))
queue : Queue = self.mutex_queues.setdefault(queue_name, Queue())
LOGGER.warning('[wait_my_turn] [lock] queue_name={:s} queue={:s}'.format(str(queue_name), str(queue)))
LOGGER.warning('[wait_my_turn] [lock] before get queue_name={:s}'.format(str(queue_name)))
mutex = queue.get(block=True, timeout=0.1)
LOGGER.warning('[wait_my_turn] [lock] after get queue_name={:s} mutex={:s}'.format(str(queue_name), str(mutex)))
LOGGER.warning('[wait_my_turn] [lock] empty queue_name={:s}'.format(str(queue_name)))
if queue.qsize() == 0:
LOGGER.warning('[wait_my_turn] end queue.qsize==0 queue_name={:s}'.format(str(queue_name)))
return
# otherwise, signal the next task in the queue to start
next_mutex : threading.Event = queue.queue[0]
LOGGER.warning('[wait_my_turn] [lock] before set queue_name={:s} next_mutex={:s}'.format(str(queue_name), str(next_mutex)))
LOGGER.warning('[wait_my_turn] [lock] after set queue_name={:s} next_mutex={:s}'.format(str(queue_name), str(next_mutex)))
LOGGER.warning('[signal_done] end set queue_name={:s}'.format(str(queue_name)))