You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
471 lines
13 KiB
Python
471 lines
13 KiB
Python
5 years ago
|
import unittest
|
||
|
|
||
|
import gevent.testing as greentest
|
||
|
from gevent.testing import TestCase
|
||
|
import gevent
|
||
|
from gevent.hub import get_hub, LoopExit
|
||
|
from gevent import util
|
||
|
from gevent import queue
|
||
|
from gevent.queue import Empty, Full
|
||
|
from gevent.event import AsyncResult
|
||
|
from gevent.testing.timing import AbstractGenericGetTestCase
|
||
|
|
||
|
# pylint:disable=too-many-ancestors
|
||
|
|
||
|
class TestQueue(TestCase):
|
||
|
|
||
|
def test_send_first(self):
|
||
|
self.switch_expected = False
|
||
|
q = queue.Queue()
|
||
|
q.put('hi')
|
||
|
self.assertEqual(q.peek(), 'hi')
|
||
|
self.assertEqual(q.get(), 'hi')
|
||
|
|
||
|
def test_peek_empty(self):
|
||
|
q = queue.Queue()
|
||
|
# No putters waiting, in the main loop: LoopExit
|
||
|
with self.assertRaises(LoopExit):
|
||
|
q.peek()
|
||
|
|
||
|
def waiter(q):
|
||
|
self.assertRaises(Empty, q.peek, timeout=0.01)
|
||
|
g = gevent.spawn(waiter, q)
|
||
|
gevent.sleep(0.1)
|
||
|
g.join()
|
||
|
|
||
|
def test_peek_multi_greenlet(self):
|
||
|
q = queue.Queue()
|
||
|
g = gevent.spawn(q.peek)
|
||
|
g.start()
|
||
|
gevent.sleep(0)
|
||
|
q.put(1)
|
||
|
g.join()
|
||
|
self.assertTrue(g.exception is None)
|
||
|
self.assertEqual(q.peek(), 1)
|
||
|
|
||
|
def test_send_last(self):
|
||
|
q = queue.Queue()
|
||
|
|
||
|
def waiter(q):
|
||
|
with gevent.Timeout(0.1 if not greentest.RUNNING_ON_APPVEYOR else 0.5):
|
||
|
self.assertEqual(q.get(), 'hi2')
|
||
|
return "OK"
|
||
|
|
||
|
p = gevent.spawn(waiter, q)
|
||
|
gevent.sleep(0.01)
|
||
|
q.put('hi2')
|
||
|
gevent.sleep(0.01)
|
||
|
assert p.get(timeout=0) == "OK"
|
||
|
|
||
|
def test_max_size(self):
|
||
|
q = queue.Queue(2)
|
||
|
results = []
|
||
|
|
||
|
def putter(q):
|
||
|
q.put('a')
|
||
|
results.append('a')
|
||
|
q.put('b')
|
||
|
results.append('b')
|
||
|
q.put('c')
|
||
|
results.append('c')
|
||
|
return "OK"
|
||
|
|
||
|
p = gevent.spawn(putter, q)
|
||
|
gevent.sleep(0)
|
||
|
self.assertEqual(results, ['a', 'b'])
|
||
|
self.assertEqual(q.get(), 'a')
|
||
|
gevent.sleep(0)
|
||
|
self.assertEqual(results, ['a', 'b', 'c'])
|
||
|
self.assertEqual(q.get(), 'b')
|
||
|
self.assertEqual(q.get(), 'c')
|
||
|
assert p.get(timeout=0) == "OK"
|
||
|
|
||
|
def test_zero_max_size(self):
|
||
|
q = queue.Channel()
|
||
|
|
||
|
def sender(evt, q):
|
||
|
q.put('hi')
|
||
|
evt.set('done')
|
||
|
|
||
|
def receiver(evt, q):
|
||
|
x = q.get()
|
||
|
evt.set(x)
|
||
|
|
||
|
e1 = AsyncResult()
|
||
|
e2 = AsyncResult()
|
||
|
|
||
|
p1 = gevent.spawn(sender, e1, q)
|
||
|
gevent.sleep(0.001)
|
||
|
self.assertTrue(not e1.ready())
|
||
|
p2 = gevent.spawn(receiver, e2, q)
|
||
|
self.assertEqual(e2.get(), 'hi')
|
||
|
self.assertEqual(e1.get(), 'done')
|
||
|
with gevent.Timeout(0):
|
||
|
gevent.joinall([p1, p2])
|
||
|
|
||
|
def test_multiple_waiters(self):
|
||
|
# tests that multiple waiters get their results back
|
||
|
q = queue.Queue()
|
||
|
|
||
|
def waiter(q, evt):
|
||
|
evt.set(q.get())
|
||
|
|
||
|
sendings = ['1', '2', '3', '4']
|
||
|
evts = [AsyncResult() for x in sendings]
|
||
|
for i, _ in enumerate(sendings):
|
||
|
gevent.spawn(waiter, q, evts[i]) # XXX use waitall for them
|
||
|
|
||
|
gevent.sleep(0.01) # get 'em all waiting
|
||
|
|
||
|
results = set()
|
||
|
|
||
|
def collect_pending_results():
|
||
|
for e in evts:
|
||
|
with gevent.Timeout(0.001, False):
|
||
|
x = e.get()
|
||
|
results.add(x)
|
||
|
return len(results)
|
||
|
|
||
|
q.put(sendings[0])
|
||
|
self.assertEqual(collect_pending_results(), 1)
|
||
|
q.put(sendings[1])
|
||
|
self.assertEqual(collect_pending_results(), 2)
|
||
|
q.put(sendings[2])
|
||
|
q.put(sendings[3])
|
||
|
self.assertEqual(collect_pending_results(), 4)
|
||
|
|
||
|
def test_waiters_that_cancel(self):
|
||
|
q = queue.Queue()
|
||
|
|
||
|
def do_receive(q, evt):
|
||
|
with gevent.Timeout(0, RuntimeError()):
|
||
|
try:
|
||
|
result = q.get()
|
||
|
evt.set(result) # pragma: no cover (should have raised)
|
||
|
except RuntimeError:
|
||
|
evt.set('timed out')
|
||
|
|
||
|
evt = AsyncResult()
|
||
|
gevent.spawn(do_receive, q, evt)
|
||
|
self.assertEqual(evt.get(), 'timed out')
|
||
|
|
||
|
q.put('hi')
|
||
|
self.assertEqual(q.get(), 'hi')
|
||
|
|
||
|
def test_senders_that_die(self):
|
||
|
q = queue.Queue()
|
||
|
|
||
|
def do_send(q):
|
||
|
q.put('sent')
|
||
|
|
||
|
gevent.spawn(do_send, q)
|
||
|
self.assertEqual(q.get(), 'sent')
|
||
|
|
||
|
def test_two_waiters_one_dies(self):
|
||
|
|
||
|
def waiter(q, evt):
|
||
|
evt.set(q.get())
|
||
|
|
||
|
def do_receive(q, evt):
|
||
|
with gevent.Timeout(0, RuntimeError()):
|
||
|
try:
|
||
|
result = q.get()
|
||
|
evt.set(result) # pragma: no cover (should have raised)
|
||
|
except RuntimeError:
|
||
|
evt.set('timed out')
|
||
|
|
||
|
q = queue.Queue()
|
||
|
dying_evt = AsyncResult()
|
||
|
waiting_evt = AsyncResult()
|
||
|
gevent.spawn(do_receive, q, dying_evt)
|
||
|
gevent.spawn(waiter, q, waiting_evt)
|
||
|
gevent.sleep(0.1)
|
||
|
q.put('hi')
|
||
|
self.assertEqual(dying_evt.get(), 'timed out')
|
||
|
self.assertEqual(waiting_evt.get(), 'hi')
|
||
|
|
||
|
def test_two_bogus_waiters(self):
|
||
|
def do_receive(q, evt):
|
||
|
with gevent.Timeout(0, RuntimeError()):
|
||
|
try:
|
||
|
result = q.get()
|
||
|
evt.set(result) # pragma: no cover (should have raised)
|
||
|
except RuntimeError:
|
||
|
evt.set('timed out')
|
||
|
|
||
|
q = queue.Queue()
|
||
|
e1 = AsyncResult()
|
||
|
e2 = AsyncResult()
|
||
|
gevent.spawn(do_receive, q, e1)
|
||
|
gevent.spawn(do_receive, q, e2)
|
||
|
gevent.sleep(0.1)
|
||
|
q.put('sent')
|
||
|
self.assertEqual(e1.get(), 'timed out')
|
||
|
self.assertEqual(e2.get(), 'timed out')
|
||
|
self.assertEqual(q.get(), 'sent')
|
||
|
|
||
|
|
||
|
class TestChannel(TestCase):
|
||
|
|
||
|
def test_send(self):
|
||
|
channel = queue.Channel()
|
||
|
|
||
|
events = []
|
||
|
|
||
|
def another_greenlet():
|
||
|
events.append(channel.get())
|
||
|
events.append(channel.get())
|
||
|
|
||
|
g = gevent.spawn(another_greenlet)
|
||
|
|
||
|
events.append('sending')
|
||
|
channel.put('hello')
|
||
|
events.append('sent hello')
|
||
|
channel.put('world')
|
||
|
events.append('sent world')
|
||
|
|
||
|
self.assertEqual(['sending', 'hello', 'sent hello', 'world', 'sent world'], events)
|
||
|
g.get()
|
||
|
|
||
|
def test_wait(self):
|
||
|
channel = queue.Channel()
|
||
|
events = []
|
||
|
|
||
|
def another_greenlet():
|
||
|
events.append('sending hello')
|
||
|
channel.put('hello')
|
||
|
events.append('sending world')
|
||
|
channel.put('world')
|
||
|
events.append('sent world')
|
||
|
|
||
|
g = gevent.spawn(another_greenlet)
|
||
|
|
||
|
events.append('waiting')
|
||
|
events.append(channel.get())
|
||
|
events.append(channel.get())
|
||
|
|
||
|
self.assertEqual(['waiting', 'sending hello', 'hello', 'sending world', 'world'], events)
|
||
|
gevent.sleep(0)
|
||
|
self.assertEqual(['waiting', 'sending hello', 'hello', 'sending world', 'world', 'sent world'], events)
|
||
|
g.get()
|
||
|
|
||
|
def test_iterable(self):
|
||
|
channel = queue.Channel()
|
||
|
gevent.spawn(channel.put, StopIteration)
|
||
|
r = list(channel)
|
||
|
self.assertEqual(r, [])
|
||
|
|
||
|
class TestJoinableQueue(TestCase):
|
||
|
|
||
|
def test_task_done(self):
|
||
|
channel = queue.JoinableQueue()
|
||
|
X = object()
|
||
|
gevent.spawn(channel.put, X)
|
||
|
result = channel.get()
|
||
|
self.assertIs(result, X)
|
||
|
self.assertEqual(1, channel.unfinished_tasks)
|
||
|
channel.task_done()
|
||
|
self.assertEqual(0, channel.unfinished_tasks)
|
||
|
|
||
|
|
||
|
class TestNoWait(TestCase):
|
||
|
|
||
|
def test_put_nowait_simple(self):
|
||
|
result = []
|
||
|
q = queue.Queue(1)
|
||
|
|
||
|
def store_result(func, *args):
|
||
|
result.append(func(*args))
|
||
|
|
||
|
run_callback = get_hub().loop.run_callback
|
||
|
|
||
|
run_callback(store_result, util.wrap_errors(Full, q.put_nowait), 2)
|
||
|
run_callback(store_result, util.wrap_errors(Full, q.put_nowait), 3)
|
||
|
gevent.sleep(0)
|
||
|
assert len(result) == 2, result
|
||
|
assert result[0] is None, result
|
||
|
assert isinstance(result[1], queue.Full), result
|
||
|
|
||
|
def test_get_nowait_simple(self):
|
||
|
result = []
|
||
|
q = queue.Queue(1)
|
||
|
q.put(4)
|
||
|
|
||
|
def store_result(func, *args):
|
||
|
result.append(func(*args))
|
||
|
|
||
|
run_callback = get_hub().loop.run_callback
|
||
|
|
||
|
run_callback(store_result, util.wrap_errors(Empty, q.get_nowait))
|
||
|
run_callback(store_result, util.wrap_errors(Empty, q.get_nowait))
|
||
|
gevent.sleep(0)
|
||
|
assert len(result) == 2, result
|
||
|
assert result[0] == 4, result
|
||
|
assert isinstance(result[1], queue.Empty), result
|
||
|
|
||
|
# get_nowait must work from the mainloop
|
||
|
def test_get_nowait_unlock(self):
|
||
|
result = []
|
||
|
q = queue.Queue(1)
|
||
|
p = gevent.spawn(q.put, 5)
|
||
|
|
||
|
def store_result(func, *args):
|
||
|
result.append(func(*args))
|
||
|
|
||
|
assert q.empty(), q
|
||
|
gevent.sleep(0)
|
||
|
assert q.full(), q
|
||
|
get_hub().loop.run_callback(store_result, q.get_nowait)
|
||
|
gevent.sleep(0)
|
||
|
assert q.empty(), q
|
||
|
assert result == [5], result
|
||
|
assert p.ready(), p
|
||
|
assert p.dead, p
|
||
|
assert q.empty(), q
|
||
|
|
||
|
def test_get_nowait_unlock_channel(self):
|
||
|
# get_nowait runs fine in the hub, and
|
||
|
# it switches to a waiting putter if needed.
|
||
|
result = []
|
||
|
q = queue.Channel()
|
||
|
p = gevent.spawn(q.put, 5)
|
||
|
|
||
|
def store_result(func, *args):
|
||
|
result.append(func(*args))
|
||
|
|
||
|
self.assertTrue(q.empty())
|
||
|
self.assertTrue(q.full())
|
||
|
|
||
|
gevent.sleep(0.001)
|
||
|
self.assertTrue(q.empty())
|
||
|
self.assertTrue(q.full())
|
||
|
|
||
|
get_hub().loop.run_callback(store_result, q.get_nowait)
|
||
|
gevent.sleep(0.001)
|
||
|
self.assertTrue(q.empty())
|
||
|
self.assertTrue(q.full())
|
||
|
self.assertEqual(result, [5])
|
||
|
self.assertTrue(p.ready())
|
||
|
self.assertTrue(p.dead)
|
||
|
self.assertTrue(q.empty())
|
||
|
|
||
|
# put_nowait must work from the mainloop
|
||
|
def test_put_nowait_unlock(self):
|
||
|
result = []
|
||
|
q = queue.Queue()
|
||
|
p = gevent.spawn(q.get)
|
||
|
|
||
|
def store_result(func, *args):
|
||
|
result.append(func(*args))
|
||
|
|
||
|
self.assertTrue(q.empty(), q)
|
||
|
self.assertFalse(q.full(), q)
|
||
|
gevent.sleep(0.001)
|
||
|
|
||
|
self.assertTrue(q.empty(), q)
|
||
|
self.assertFalse(q.full(), q)
|
||
|
|
||
|
get_hub().loop.run_callback(store_result, q.put_nowait, 10)
|
||
|
|
||
|
self.assertFalse(p.ready(), p)
|
||
|
gevent.sleep(0.001)
|
||
|
|
||
|
self.assertEqual(result, [None])
|
||
|
self.assertTrue(p.ready(), p)
|
||
|
self.assertFalse(q.full(), q)
|
||
|
self.assertTrue(q.empty(), q)
|
||
|
|
||
|
|
||
|
class TestJoinEmpty(TestCase):
|
||
|
|
||
|
def test_issue_45(self):
|
||
|
"""Test that join() exits immediately if not jobs were put into the queue"""
|
||
|
self.switch_expected = False
|
||
|
q = queue.JoinableQueue()
|
||
|
q.join()
|
||
|
|
||
|
class AbstractTestWeakRefMixin(object):
|
||
|
|
||
|
def test_weak_reference(self):
|
||
|
import weakref
|
||
|
one = self._makeOne()
|
||
|
ref = weakref.ref(one)
|
||
|
self.assertIs(one, ref())
|
||
|
|
||
|
|
||
|
class TestGetInterrupt(AbstractTestWeakRefMixin, AbstractGenericGetTestCase):
|
||
|
|
||
|
Timeout = Empty
|
||
|
|
||
|
kind = queue.Queue
|
||
|
|
||
|
def wait(self, timeout):
|
||
|
return self._makeOne().get(timeout=timeout)
|
||
|
|
||
|
def _makeOne(self):
|
||
|
return self.kind()
|
||
|
|
||
|
class TestGetInterruptJoinableQueue(TestGetInterrupt):
|
||
|
kind = queue.JoinableQueue
|
||
|
|
||
|
class TestGetInterruptLifoQueue(TestGetInterrupt):
|
||
|
kind = queue.LifoQueue
|
||
|
|
||
|
class TestGetInterruptPriorityQueue(TestGetInterrupt):
|
||
|
kind = queue.PriorityQueue
|
||
|
|
||
|
class TestGetInterruptChannel(TestGetInterrupt):
|
||
|
kind = queue.Channel
|
||
|
|
||
|
|
||
|
class TestPutInterrupt(AbstractGenericGetTestCase):
|
||
|
kind = queue.Queue
|
||
|
Timeout = Full
|
||
|
|
||
|
def setUp(self):
|
||
|
super(TestPutInterrupt, self).setUp()
|
||
|
self.queue = self._makeOne()
|
||
|
|
||
|
def wait(self, timeout):
|
||
|
while not self.queue.full():
|
||
|
self.queue.put(1)
|
||
|
return self.queue.put(2, timeout=timeout)
|
||
|
|
||
|
def _makeOne(self):
|
||
|
return self.kind(1)
|
||
|
|
||
|
|
||
|
class TestPutInterruptJoinableQueue(TestPutInterrupt):
|
||
|
kind = queue.JoinableQueue
|
||
|
|
||
|
class TestPutInterruptLifoQueue(TestPutInterrupt):
|
||
|
kind = queue.LifoQueue
|
||
|
|
||
|
class TestPutInterruptPriorityQueue(TestPutInterrupt):
|
||
|
kind = queue.PriorityQueue
|
||
|
|
||
|
class TestPutInterruptChannel(TestPutInterrupt):
|
||
|
kind = queue.Channel
|
||
|
|
||
|
def _makeOne(self):
|
||
|
return self.kind()
|
||
|
|
||
|
|
||
|
if hasattr(queue, 'SimpleQueue'):
|
||
|
|
||
|
class TestGetInterruptSimpleQueue(TestGetInterrupt):
|
||
|
kind = queue.SimpleQueue
|
||
|
|
||
|
def test_raises_timeout_Timeout(self):
|
||
|
raise unittest.SkipTest("Not supported")
|
||
|
|
||
|
test_raises_timeout_Timeout_exc_customized = test_raises_timeout_Timeout
|
||
|
test_outer_timeout_is_not_lost = test_raises_timeout_Timeout
|
||
|
|
||
|
|
||
|
del AbstractGenericGetTestCase
|
||
|
|
||
|
|
||
|
if __name__ == '__main__':
|
||
|
greentest.main()
|