mirror of
git://sourceware.org/git/lvm2.git
synced 2025-01-04 09:18:36 +03:00
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.
This commit is contained in:
parent
8d959b6c75
commit
20b21f4fd8
@ -13,7 +13,7 @@ from . import cfg
|
|||||||
import time
|
import time
|
||||||
from .cmdhandler import options_to_cli_args
|
from .cmdhandler import options_to_cli_args
|
||||||
import dbus
|
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
|
import traceback
|
||||||
|
|
||||||
_rlock = threading.RLock()
|
_rlock = threading.RLock()
|
||||||
@ -118,6 +118,7 @@ def background_reaper():
|
|||||||
for i in range(num_threads, -1, -1):
|
for i in range(num_threads, -1, -1):
|
||||||
_thread_list[i].join(0)
|
_thread_list[i].join(0)
|
||||||
if not _thread_list[i].is_alive():
|
if not _thread_list[i].is_alive():
|
||||||
|
log_debug("Removing thread: %s" % _thread_list[i].name)
|
||||||
_thread_list.pop(i)
|
_thread_list.pop(i)
|
||||||
|
|
||||||
time.sleep(3)
|
time.sleep(3)
|
||||||
@ -171,3 +172,30 @@ def add(command, reporting_job):
|
|||||||
|
|
||||||
with _rlock:
|
with _rlock:
|
||||||
_thread_list.append(t)
|
_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)
|
||||||
|
@ -13,6 +13,7 @@ from . import cfg
|
|||||||
from .cfg import JOB_INTERFACE
|
from .cfg import JOB_INTERFACE
|
||||||
import dbus
|
import dbus
|
||||||
import threading
|
import threading
|
||||||
|
from . import background
|
||||||
|
|
||||||
|
|
||||||
# noinspection PyPep8Naming
|
# noinspection PyPep8Naming
|
||||||
@ -152,9 +153,10 @@ class Job(AutomatedProperties):
|
|||||||
|
|
||||||
@dbus.service.method(dbus_interface=JOB_INTERFACE,
|
@dbus.service.method(dbus_interface=JOB_INTERFACE,
|
||||||
in_signature='i',
|
in_signature='i',
|
||||||
out_signature='b')
|
out_signature='b',
|
||||||
def Wait(self, timeout):
|
async_callbacks=('cb', 'cbe'))
|
||||||
return self.state.Wait(timeout)
|
def Wait(self, timeout, cb, cbe):
|
||||||
|
background.add_wait(self, timeout, cb, cbe)
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def Result(self):
|
def Result(self):
|
||||||
|
Loading…
Reference in New Issue
Block a user