diff options
author | Tony Asleson <tasleson@redhat.com> | 2016-07-27 18:27:58 -0500 |
---|---|---|
committer | Tony Asleson <tasleson@redhat.com> | 2016-07-28 11:13:29 -0500 |
commit | 20b21f4fd85532bd63a699aac4fc34f5786cf96d (patch) | |
tree | 412d0520afe3a388b797d45a7afe41677264e25f | |
parent | 8d959b6c75a6cc98691413ec4eab5b01e992d6c1 (diff) | |
download | lvm2-20b21f4fd85532bd63a699aac4fc34f5786cf96d.tar.gz |
lvmdbusd: Fix hang for Job.Wait(n)
When a client is doing a wait on a job, any other clients will hang
when trying to do anything with the service. This is caused by
the wait code which was placing the thread that handles
incoming dbus requests to sleep until either the timeout expired or
the job operation completed.
This change creates a thread for the wait request, so that the thread
processing incoming requests can continue to run.
-rw-r--r-- | daemons/lvmdbusd/background.py | 30 | ||||
-rw-r--r-- | daemons/lvmdbusd/job.py | 8 |
2 files changed, 34 insertions, 4 deletions
diff --git a/daemons/lvmdbusd/background.py b/daemons/lvmdbusd/background.py index 2c719fca9..8078b6629 100644 --- a/daemons/lvmdbusd/background.py +++ b/daemons/lvmdbusd/background.py @@ -13,7 +13,7 @@ from . import cfg import time from .cmdhandler import options_to_cli_args import dbus -from .utils import pv_range_append, pv_dest_ranges, log_error +from .utils import pv_range_append, pv_dest_ranges, log_error, log_debug import traceback _rlock = threading.RLock() @@ -118,6 +118,7 @@ def background_reaper(): for i in range(num_threads, -1, -1): _thread_list[i].join(0) if not _thread_list[i].is_alive(): + log_debug("Removing thread: %s" % _thread_list[i].name) _thread_list.pop(i) time.sleep(3) @@ -171,3 +172,30 @@ def add(command, reporting_job): with _rlock: _thread_list.append(t) + + +def wait_thread(job, timeout, cb, cbe): + # We need to put the wait on it's own thread, so that we don't block the + # entire dbus queue processing thread + try: + cb(job.state.Wait(timeout)) + except Exception as e: + cbe("Wait exception: %s" % str(e)) + return 0 + + +def add_wait(job, timeout, cb, cbe): + + if timeout == 0: + # Users are basically polling, do not create thread + cb(job.Complete) + else: + t = threading.Thread( + target=wait_thread, + name="thread job.Wait: %s" % job.dbus_object_path(), + args=(job, timeout, cb, cbe) + ) + + t.start() + with _rlock: + _thread_list.append(t) diff --git a/daemons/lvmdbusd/job.py b/daemons/lvmdbusd/job.py index d7f8187e5..520f9a87e 100644 --- a/daemons/lvmdbusd/job.py +++ b/daemons/lvmdbusd/job.py @@ -13,6 +13,7 @@ from . import cfg from .cfg import JOB_INTERFACE import dbus import threading +from . import background # noinspection PyPep8Naming @@ -152,9 +153,10 @@ class Job(AutomatedProperties): @dbus.service.method(dbus_interface=JOB_INTERFACE, in_signature='i', - out_signature='b') - def Wait(self, timeout): - return self.state.Wait(timeout) + out_signature='b', + async_callbacks=('cb', 'cbe')) + def Wait(self, timeout, cb, cbe): + background.add_wait(self, timeout, cb, cbe) @property def Result(self): |