mirror of
https://github.com/python/cpython.git
synced 2025-08-03 08:34:29 +00:00
bpo-45835: Fix race condition in test_queue (GH-29601)
Some of the tests in test_queue had a race condition in which a
non-sentinel value could be enqueued after the final sentinel value
leading to not all the inputs being processed (and test failures).
This changes feed() to enqueue a sentinel once the inputs are exhausted,
which guarantees that the final queued object is a sentinel. This
requires the number of feeder threads to match the number of consumer
threads, but that's already the case in the relevant tests.
(cherry picked from commit df3e53d86b
)
Co-authored-by: Sam Gross <colesbury@gmail.com>
This commit is contained in:
parent
71d842b9c1
commit
9450c751cc
2 changed files with 12 additions and 11 deletions
|
@ -418,11 +418,12 @@ class BaseSimpleQueueTest:
|
|||
def setUp(self):
|
||||
self.q = self.type2test()
|
||||
|
||||
def feed(self, q, seq, rnd):
|
||||
def feed(self, q, seq, rnd, sentinel):
|
||||
while True:
|
||||
try:
|
||||
val = seq.pop()
|
||||
except IndexError:
|
||||
q.put(sentinel)
|
||||
return
|
||||
q.put(val)
|
||||
if rnd.random() > 0.5:
|
||||
|
@ -461,11 +462,10 @@ class BaseSimpleQueueTest:
|
|||
return
|
||||
results.append(val)
|
||||
|
||||
def run_threads(self, n_feeders, n_consumers, q, inputs,
|
||||
feed_func, consume_func):
|
||||
def run_threads(self, n_threads, q, inputs, feed_func, consume_func):
|
||||
results = []
|
||||
sentinel = None
|
||||
seq = inputs + [sentinel] * n_consumers
|
||||
seq = inputs.copy()
|
||||
seq.reverse()
|
||||
rnd = random.Random(42)
|
||||
|
||||
|
@ -479,11 +479,11 @@ class BaseSimpleQueueTest:
|
|||
return wrapper
|
||||
|
||||
feeders = [threading.Thread(target=log_exceptions(feed_func),
|
||||
args=(q, seq, rnd))
|
||||
for i in range(n_feeders)]
|
||||
args=(q, seq, rnd, sentinel))
|
||||
for i in range(n_threads)]
|
||||
consumers = [threading.Thread(target=log_exceptions(consume_func),
|
||||
args=(q, results, sentinel))
|
||||
for i in range(n_consumers)]
|
||||
for i in range(n_threads)]
|
||||
|
||||
with support.start_threads(feeders + consumers):
|
||||
pass
|
||||
|
@ -541,7 +541,7 @@ class BaseSimpleQueueTest:
|
|||
# Test a pair of concurrent put() and get()
|
||||
q = self.q
|
||||
inputs = list(range(100))
|
||||
results = self.run_threads(1, 1, q, inputs, self.feed, self.consume)
|
||||
results = self.run_threads(1, q, inputs, self.feed, self.consume)
|
||||
|
||||
# One producer, one consumer => results appended in well-defined order
|
||||
self.assertEqual(results, inputs)
|
||||
|
@ -551,7 +551,7 @@ class BaseSimpleQueueTest:
|
|||
N = 50
|
||||
q = self.q
|
||||
inputs = list(range(10000))
|
||||
results = self.run_threads(N, N, q, inputs, self.feed, self.consume)
|
||||
results = self.run_threads(N, q, inputs, self.feed, self.consume)
|
||||
|
||||
# Multiple consumers without synchronization append the
|
||||
# results in random order
|
||||
|
@ -562,7 +562,7 @@ class BaseSimpleQueueTest:
|
|||
N = 50
|
||||
q = self.q
|
||||
inputs = list(range(10000))
|
||||
results = self.run_threads(N, N, q, inputs,
|
||||
results = self.run_threads(N, q, inputs,
|
||||
self.feed, self.consume_nonblock)
|
||||
|
||||
self.assertEqual(sorted(results), inputs)
|
||||
|
@ -572,7 +572,7 @@ class BaseSimpleQueueTest:
|
|||
N = 50
|
||||
q = self.q
|
||||
inputs = list(range(1000))
|
||||
results = self.run_threads(N, N, q, inputs,
|
||||
results = self.run_threads(N, q, inputs,
|
||||
self.feed, self.consume_timeout)
|
||||
|
||||
self.assertEqual(sorted(results), inputs)
|
||||
|
|
|
@ -0,0 +1 @@
|
|||
Fix race condition in test_queue tests with multiple "feeder" threads.
|
Loading…
Add table
Add a link
Reference in a new issue