Skip to content
Snippets Groups Projects
Commit 71447b94 authored by Paul McCarthy's avatar Paul McCarthy :mountain_bicyclist:
Browse files

Fix bug in MutexFactory - it was possible for multiple locks to be created for

one instance, if two or more threads accessed a method at the same time.
parent 20883ff7
No related branches found
No related tags found
No related merge requests found
...@@ -862,6 +862,17 @@ class MutexFactory(object): ...@@ -862,6 +862,17 @@ class MutexFactory(object):
""" """
createLock = threading.Lock()
"""This lock is used by all ``MutexFactory`` instances when a decorated
instance method is accessed for the first time.
The first time that a mutexed method is accessed on an instance, a new
``threading.Lock`` is created, to be shared by all mutexed methods of that
instance. The ``createLock`` is used to ensure that this can only occur
once for each instance.
"""
def __init__(self, function): def __init__(self, function):
"""Create a ``MutexFactory``. """Create a ``MutexFactory``.
""" """
...@@ -883,25 +894,25 @@ class MutexFactory(object): ...@@ -883,25 +894,25 @@ class MutexFactory(object):
if instance is None: if instance is None:
return self.__func return self.__func
# Get the lock object, creating if it necessary # Get the lock object, creating if it necessary.
lock = getattr(instance, '_async_mutex_lock', None) # We use the createLock in case multiple threads
if lock is None: # access a method at the same time, in which case
lock = threading.Lock() # only one of them will be able to create the
instance._async_mutex_lock = lock # instance lock.
with MutexFactory.createLock:
# The true decorator function:
# - Acquire the lock (blocking until it has been released) lock = getattr(instance, '_idle_mutex_lock', None)
# - Run the decorated method if lock is None:
# - Release the lock lock = threading.Lock()
def decorator(*args, **kwargs): instance._idle_mutex_lock = lock
lock.acquire()
try: # The true decorator function
return self.__func(instance, *args, **kwargs) def decorator(*args, **kwargs):
finally: with instance._idle_mutex_lock:
lock.release() return self.__func(instance, *args, **kwargs)
# Replace this MutexFactory with # Replace this MutexFactory with
# the decorator on the instance # the decorator on the instance
decorator = functools.update_wrapper(decorator, self.__func) decorator = functools.update_wrapper(decorator, self.__func)
setattr(instance, self.__func.__name__, decorator) setattr(instance, self.__func.__name__, decorator)
return decorator return decorator
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment