2011-11-14 11:10:59 +00:00
|
|
|
import unittest
|
2011-11-15 07:13:16 +00:00
|
|
|
from pickle import loads
|
2011-11-14 11:10:59 +00:00
|
|
|
from blinker import signal
|
|
|
|
from redis import Redis
|
2011-11-15 07:13:16 +00:00
|
|
|
from rq import conn, Queue, job
|
|
|
|
|
2011-11-15 08:36:18 +00:00
|
|
|
# Test data
|
2011-11-15 07:13:16 +00:00
|
|
|
@job('my-queue')
|
2011-11-15 07:43:06 +00:00
|
|
|
def testjob(name=None):
|
|
|
|
if name is None:
|
|
|
|
name = 'Stranger'
|
|
|
|
return 'Hi there, %s!' % (name,)
|
2011-11-14 20:25:52 +00:00
|
|
|
|
2011-11-15 08:36:18 +00:00
|
|
|
@job() # no queue spec'ed
|
|
|
|
def queueless_job(name=None):
|
|
|
|
if name is None:
|
|
|
|
name = 'Stranger'
|
|
|
|
return 'Hi there, %s!' % (name,)
|
|
|
|
|
2011-11-14 11:10:59 +00:00
|
|
|
|
|
|
|
class RQTestCase(unittest.TestCase):
|
|
|
|
def setUp(self):
|
|
|
|
super(RQTestCase, self).setUp()
|
2011-11-14 13:18:21 +00:00
|
|
|
|
2011-11-14 20:25:52 +00:00
|
|
|
# Set up connection to Redis
|
|
|
|
testconn = Redis()
|
|
|
|
conn.push(testconn)
|
2011-11-14 13:18:21 +00:00
|
|
|
|
2011-11-14 20:25:52 +00:00
|
|
|
# Flush beforewards (we like our hygiene)
|
|
|
|
conn.flushdb()
|
2011-11-14 11:10:59 +00:00
|
|
|
signal('setup').send(self)
|
|
|
|
|
2011-11-14 20:25:52 +00:00
|
|
|
# Store the connection (for sanity checking)
|
|
|
|
self.testconn = testconn
|
|
|
|
|
2011-11-14 11:10:59 +00:00
|
|
|
def tearDown(self):
|
|
|
|
signal('teardown').send(self)
|
2011-11-14 13:18:21 +00:00
|
|
|
|
2011-11-14 20:25:52 +00:00
|
|
|
# Flush afterwards
|
|
|
|
conn.flushdb()
|
|
|
|
|
|
|
|
# Pop the connection to Redis
|
|
|
|
testconn = conn.pop()
|
|
|
|
assert testconn == self.testconn, 'Wow, something really nasty happened to the Redis connection stack. Check your setup.'
|
2011-11-14 13:18:21 +00:00
|
|
|
|
2011-11-14 11:10:59 +00:00
|
|
|
super(RQTestCase, self).tearDown()
|
|
|
|
|
|
|
|
|
2011-11-15 07:13:16 +00:00
|
|
|
def assertQueueContains(self, queue, that_func):
|
|
|
|
# Do a queue scan (this is O(n), but we're in a test, so hey)
|
|
|
|
for message in queue.messages:
|
|
|
|
f, _, args, kwargs = loads(message)
|
|
|
|
if f == that_func:
|
|
|
|
return
|
|
|
|
self.fail('Queue %s does not contain message for function %s' %
|
|
|
|
(queue.key, that_func))
|
|
|
|
|
|
|
|
|
2011-11-14 13:18:21 +00:00
|
|
|
class TestQueue(RQTestCase):
|
|
|
|
def test_create_queue(self):
|
|
|
|
"""Creating queues."""
|
|
|
|
q = Queue('my-queue')
|
|
|
|
self.assertEquals(q.name, 'my-queue')
|
|
|
|
|
|
|
|
def test_queue_empty(self):
|
|
|
|
"""Detecting empty queues."""
|
|
|
|
q = Queue('my-queue')
|
|
|
|
self.assertEquals(q.empty, True)
|
|
|
|
|
2011-11-14 20:25:52 +00:00
|
|
|
conn.rpush('rq:my-queue', 'some val')
|
2011-11-14 13:18:21 +00:00
|
|
|
self.assertEquals(q.empty, False)
|
2011-11-14 11:10:59 +00:00
|
|
|
|
2011-11-15 08:36:32 +00:00
|
|
|
|
|
|
|
def test_enqueue(self):
|
2011-11-15 07:43:06 +00:00
|
|
|
"""Putting work on queues using delay."""
|
2011-11-15 07:13:16 +00:00
|
|
|
q = Queue('my-queue')
|
|
|
|
self.assertEquals(q.empty, True)
|
|
|
|
|
2011-11-15 07:43:06 +00:00
|
|
|
# testjob spec holds which queue this is sent to
|
2011-11-15 07:13:16 +00:00
|
|
|
testjob.delay()
|
|
|
|
self.assertEquals(q.empty, False)
|
|
|
|
self.assertQueueContains(q, testjob)
|
|
|
|
|
2011-11-15 08:36:32 +00:00
|
|
|
def test_enqueue_to_different_queue(self):
|
2011-11-15 07:43:06 +00:00
|
|
|
"""Putting work on alternative queues using enqueue."""
|
|
|
|
|
|
|
|
# Override testjob spec holds which queue
|
|
|
|
q = Queue('different-queue')
|
|
|
|
self.assertEquals(q.empty, True)
|
|
|
|
testjob.enqueue(q)
|
|
|
|
self.assertEquals(q.empty, False)
|
|
|
|
self.assertQueueContains(q, testjob)
|
|
|
|
|
2011-11-15 08:36:32 +00:00
|
|
|
def test_enqueue_to_different_queue_reverse(self):
|
2011-11-15 07:43:06 +00:00
|
|
|
"""Putting work on specific queues using the Queue object."""
|
|
|
|
|
|
|
|
q = Queue('alt-queue')
|
|
|
|
self.assertEquals(q.empty, True)
|
|
|
|
q.enqueue(testjob)
|
|
|
|
self.assertEquals(q.empty, False)
|
|
|
|
self.assertQueueContains(q, testjob)
|
|
|
|
|
2011-11-15 08:36:32 +00:00
|
|
|
|
2011-11-15 08:36:29 +00:00
|
|
|
def test_dequeue(self):
|
|
|
|
"""Fetching work from specific queue."""
|
|
|
|
q = Queue('foo')
|
|
|
|
testjob.enqueue(q, 'Rick')
|
|
|
|
|
|
|
|
# Pull it off the queue (normally, a worker would do this)
|
|
|
|
f, rv_key, args, kwargs = q.dequeue()
|
|
|
|
self.assertEquals(f, testjob)
|
|
|
|
self.assertEquals(args[0], 'Rick')
|
2011-11-14 11:10:59 +00:00
|
|
|
|
2011-11-15 08:36:32 +00:00
|
|
|
|
2011-11-15 08:36:18 +00:00
|
|
|
class TestJob(RQTestCase):
|
|
|
|
def test_job_methods(self):
|
|
|
|
"""Jobs have methods to enqueue them."""
|
|
|
|
self.assertTrue(hasattr(testjob, 'delay'))
|
|
|
|
self.assertTrue(hasattr(testjob, 'enqueue'))
|
|
|
|
self.assertTrue(hasattr(queueless_job, 'delay'))
|
|
|
|
self.assertTrue(hasattr(queueless_job, 'enqueue'))
|
|
|
|
|
|
|
|
def test_queue_empty(self):
|
|
|
|
"""Detecting empty queues."""
|
|
|
|
q = Queue('my-queue')
|
|
|
|
self.assertEquals(q.empty, True)
|
|
|
|
|
|
|
|
conn.rpush('rq:my-queue', 'some val')
|
|
|
|
self.assertEquals(q.empty, False)
|
|
|
|
|
|
|
|
def test_put_work_on_queue(self):
|
|
|
|
"""Putting work on queues using delay."""
|
|
|
|
q = Queue('my-queue')
|
|
|
|
self.assertEquals(q.empty, True)
|
|
|
|
|
|
|
|
# testjob spec holds which queue this is sent to
|
|
|
|
testjob.delay()
|
|
|
|
self.assertEquals(q.empty, False)
|
|
|
|
self.assertQueueContains(q, testjob)
|
|
|
|
|
|
|
|
def test_put_work_on_queue_fails_for_queueless_jobs(self):
|
|
|
|
"""Putting work on queues using delay fails for queueless jobs."""
|
|
|
|
self.assertRaises(ValueError, queueless_job.delay, 'Rick')
|
|
|
|
|
|
|
|
def test_put_work_on_different_queue(self):
|
|
|
|
"""Putting work on alternative queues using enqueue."""
|
|
|
|
|
|
|
|
# Override testjob spec holds which queue
|
|
|
|
q = Queue('different-queue')
|
|
|
|
self.assertEquals(q.empty, True)
|
|
|
|
testjob.enqueue(q)
|
|
|
|
self.assertEquals(q.empty, False)
|
|
|
|
self.assertQueueContains(q, testjob)
|
|
|
|
|
|
|
|
def test_put_work_on_different_queue_reverse(self):
|
|
|
|
"""Putting work on specific queues using the Queue object."""
|
|
|
|
|
|
|
|
q = Queue('alt-queue')
|
|
|
|
self.assertEquals(q.empty, True)
|
|
|
|
q.enqueue(testjob)
|
|
|
|
self.assertEquals(q.empty, False)
|
|
|
|
self.assertQueueContains(q, testjob)
|
|
|
|
|
|
|
|
|
2011-11-14 11:10:59 +00:00
|
|
|
if __name__ == '__main__':
|
|
|
|
unittest.main()
|