Skip to content

Commit 8362863

Browse files
committed
Fix hang caused by steal command with empty test queue
Fixes pytest-dev#884
1 parent 58fd7cc commit 8362863

File tree

3 files changed

+47
-2
lines changed

3 files changed

+47
-2
lines changed

changelog/884.bugfix

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
Fix hang caused by `steal` command with empty test queue (`worksteal` scheduler).

src/xdist/remote.py

Lines changed: 12 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -58,6 +58,7 @@ def worker_title(title):
5858

5959
class WorkerInteractor:
6060
SHUTDOWN_MARK = object()
61+
RETRY_MARK = object()
6162

6263
def __init__(self, config, channel):
6364
self.config = config
@@ -72,6 +73,12 @@ def __init__(self, config, channel):
7273
def _make_queue(self):
7374
return self.channel.gateway.execmodel.queue.Queue()
7475

76+
def _get_next_item_index(self):
77+
result = self.RETRY_MARK
78+
while result is self.RETRY_MARK:
79+
result = self.torun.get()
80+
return result
81+
7582
def sendevent(self, name, **kwargs):
7683
self.log("sending", name, kwargs)
7784
self.channel.send((name, kwargs))
@@ -136,19 +143,22 @@ def old_queue_get_nowait_noraise():
136143
self.torun.put(i)
137144

138145
self.sendevent("unscheduled", indices=stolen)
146+
old_queue.put(self.RETRY_MARK)
139147

140148
@pytest.hookimpl
141149
def pytest_runtestloop(self, session):
142150
self.log("entering main loop")
143151
self.channel.setcallback(self.handle_command, endmarker=self.SHUTDOWN_MARK)
144-
self.nextitem_index = self.torun.get()
152+
self.nextitem_index = self._get_next_item_index()
145153
while self.nextitem_index is not self.SHUTDOWN_MARK:
146154
self.run_one_test()
147155
return True
148156

149157
def run_one_test(self):
158+
self.item_index = self.nextitem_index
159+
self.nextitem_index = self._get_next_item_index()
160+
150161
items = self.session.items
151-
self.item_index, self.nextitem_index = self.nextitem_index, self.torun.get()
152162
item = items[self.item_index]
153163
if self.nextitem_index is self.SHUTDOWN_MARK:
154164
nextitem = None

testing/test_remote.py

Lines changed: 34 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -271,6 +271,40 @@ def test_func4(): pass
271271
ev = worker.popevent("workerfinished")
272272
assert "workeroutput" in ev.kwargs
273273

274+
def test_steal_empty_queue(self, worker: WorkerSetup, unserialize_report) -> None:
275+
worker.pytester.makepyfile(
276+
"""
277+
def test_func(): pass
278+
def test_func2(): pass
279+
"""
280+
)
281+
worker.setup()
282+
ev = worker.popevent("collectionfinish")
283+
ids = ev.kwargs["ids"]
284+
assert len(ids) == 2
285+
worker.sendcommand("runtests_all")
286+
287+
for when in ["setup", "call", "teardown"]:
288+
ev = worker.popevent("testreport")
289+
rep = unserialize_report(ev.kwargs["data"])
290+
assert rep.nodeid.endswith("::test_func")
291+
assert rep.when == when
292+
293+
worker.sendcommand("steal", indices=[0, 1])
294+
ev = worker.popevent("unscheduled")
295+
assert ev.kwargs["indices"] == []
296+
297+
worker.sendcommand("shutdown")
298+
299+
for when in ["setup", "call", "teardown"]:
300+
ev = worker.popevent("testreport")
301+
rep = unserialize_report(ev.kwargs["data"])
302+
assert rep.nodeid.endswith("::test_func2")
303+
assert rep.when == when
304+
305+
ev = worker.popevent("workerfinished")
306+
assert "workeroutput" in ev.kwargs
307+
274308

275309
def test_remote_env_vars(pytester: pytest.Pytester) -> None:
276310
pytester.makepyfile(

0 commit comments

Comments
 (0)