From 5b8726ad2d8bce4d7c947af2bb6c5bd8f060452b Mon Sep 17 00:00:00 2001 From: Marcus Martins Date: Tue, 26 May 2015 14:39:17 -0300 Subject: [PATCH 1/3] Fixes #502 Fixes some broken tests and misbehaviour with ttls. There was a temporal coupling between saving the job and setting its expires parameter. --- rq/job.py | 5 ++++- tests/test_job.py | 28 ++++++++++++++++++++++------ 2 files changed, 26 insertions(+), 7 deletions(-) diff --git a/rq/job.py b/rq/job.py index cbdf7f4..c1022cc 100644 --- a/rq/job.py +++ b/rq/job.py @@ -417,6 +417,7 @@ class Job(object): self.result_ttl = int(obj.get('result_ttl')) if obj.get('result_ttl') else None # noqa self._status = as_text(obj.get('status') if obj.get('status') else None) self._dependency_id = as_text(obj.get('dependency_id', None)) + self.ttl = int(obj.get('ttl', -1)) self.meta = unpickle(obj.get('meta')) if obj.get('meta') else {} def to_dict(self): @@ -447,6 +448,8 @@ class Job(object): obj['dependency_id'] = self._dependency_id if self.meta: obj['meta'] = dumps(self.meta) + if self.ttl: + obj['ttl'] = self.ttl return obj @@ -456,7 +459,7 @@ class Job(object): connection = pipeline if pipeline is not None else self.connection connection.hmset(key, self.to_dict()) - self.cleanup(self.ttl) + self.cleanup(self.ttl, pipeline=connection) def cancel(self): """Cancels the given job, which will prevent the job from ever being diff --git a/tests/test_job.py b/tests/test_job.py index 36570cf..08d6385 100644 --- a/tests/test_job.py +++ b/tests/test_job.py @@ -2,6 +2,7 @@ from __future__ import (absolute_import, division, print_function, unicode_literals) +import time from datetime import datetime from tests import RQTestCase @@ -15,7 +16,7 @@ from rq.registry import DeferredJobRegistry from rq.utils import utcformat from rq.worker import Worker -from . import fixtures +import fixtures try: from cPickle import loads, dumps @@ -103,7 +104,8 @@ class TestJob(RQTestCase): job = Job.create(func='tests.fixtures.say_hello', args=('World',)) # Job data is set - self.assertEquals(job.func, fixtures.say_hello) + self.assertTrue(job.func.func_code.co_filename in fixtures.say_hello.func_code.co_filename) + self.assertEquals(job.func.func_code.co_firstlineno, fixtures.say_hello.func_code.co_firstlineno) self.assertIsNone(job.instance) self.assertEquals(job.args, ('World',)) @@ -147,7 +149,7 @@ class TestJob(RQTestCase): # Saving writes pickled job data unpickled_data = loads(self.testconn.hget(job.key, 'data')) - self.assertEquals(unpickled_data[0], 'tests.fixtures.some_calculation') + self.assertEquals(unpickled_data[0], 'fixtures.some_calculation') def test_fetch(self): """Fetching jobs.""" @@ -288,9 +290,9 @@ class TestJob(RQTestCase): job.save() Job.fetch(job.id, connection=self.testconn) if PY2: - self.assertEqual(job.description, "tests.fixtures.say_hello(u'Lionel')") + self.assertEqual(job.description, "fixtures.say_hello(u'Lionel')") else: - self.assertEqual(job.description, "tests.fixtures.say_hello('Lionel')") + self.assertEqual(job.description, "fixtures.say_hello('Lionel')") def test_job_access_outside_job_fails(self): """The current job is accessible only within a job context.""" @@ -406,4 +408,18 @@ class TestJob(RQTestCase): job = queue.enqueue(fixtures.echo, arg_with_unicode=fixtures.UnicodeStringObject()) self.assertIsNotNone(job.get_call_string()) - job.perform() \ No newline at end of file + job.perform() + + def test_create_job_with_ttl_should_have_ttl_after_enqueued(self): + """test creating jobs with ttl and checks if get_jobs returns it properly [issue502]""" + queue = Queue(connection=self.testconn) + queue.enqueue(fixtures.say_hello, job_id="1234", ttl=10) + job = queue.get_jobs()[0] + self.assertEqual(job.ttl, 10) + + def test_create_job_with_ttl_should_expire(self): + """test if a job created with ttl expires [issue502]""" + queue = Queue(connection=self.testconn) + queue.enqueue(fixtures.say_hello, job_id="1234", ttl=1) + time.sleep(1) + self.assertEqual(0, len(queue.get_jobs())) From ab6c129833dcb013142beefede6d44da3c3f16be Mon Sep 17 00:00:00 2001 From: Marcus Martins Date: Tue, 26 May 2015 16:30:35 -0300 Subject: [PATCH 2/3] Fix broken tests --- tests/test_job.py | 13 ++++++------- 1 file changed, 6 insertions(+), 7 deletions(-) diff --git a/tests/test_job.py b/tests/test_job.py index 08d6385..a078ce6 100644 --- a/tests/test_job.py +++ b/tests/test_job.py @@ -2,8 +2,8 @@ from __future__ import (absolute_import, division, print_function, unicode_literals) -import time from datetime import datetime +import time from tests import RQTestCase from tests.helpers import strip_microseconds @@ -16,7 +16,7 @@ from rq.registry import DeferredJobRegistry from rq.utils import utcformat from rq.worker import Worker -import fixtures +from . import fixtures try: from cPickle import loads, dumps @@ -104,8 +104,7 @@ class TestJob(RQTestCase): job = Job.create(func='tests.fixtures.say_hello', args=('World',)) # Job data is set - self.assertTrue(job.func.func_code.co_filename in fixtures.say_hello.func_code.co_filename) - self.assertEquals(job.func.func_code.co_firstlineno, fixtures.say_hello.func_code.co_firstlineno) + self.assertEquals(job.func, fixtures.say_hello) self.assertIsNone(job.instance) self.assertEquals(job.args, ('World',)) @@ -149,7 +148,7 @@ class TestJob(RQTestCase): # Saving writes pickled job data unpickled_data = loads(self.testconn.hget(job.key, 'data')) - self.assertEquals(unpickled_data[0], 'fixtures.some_calculation') + self.assertEquals(unpickled_data[0], 'tests.fixtures.some_calculation') def test_fetch(self): """Fetching jobs.""" @@ -290,9 +289,9 @@ class TestJob(RQTestCase): job.save() Job.fetch(job.id, connection=self.testconn) if PY2: - self.assertEqual(job.description, "fixtures.say_hello(u'Lionel')") + self.assertEqual(job.description, "tests.fixtures.say_hello(u'Lionel')") else: - self.assertEqual(job.description, "fixtures.say_hello('Lionel')") + self.assertEqual(job.description, "tests.fixtures.say_hello('Lionel')") def test_job_access_outside_job_fails(self): """The current job is accessible only within a job context.""" From 513f6310d2d6946907b34141f13d7637de6acb2b Mon Sep 17 00:00:00 2001 From: Marcus Martins Date: Thu, 28 May 2015 23:35:30 -0300 Subject: [PATCH 3/3] Change default TTL to None insted of -1 --- rq/job.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/rq/job.py b/rq/job.py index c1022cc..ebccb13 100644 --- a/rq/job.py +++ b/rq/job.py @@ -417,7 +417,7 @@ class Job(object): self.result_ttl = int(obj.get('result_ttl')) if obj.get('result_ttl') else None # noqa self._status = as_text(obj.get('status') if obj.get('status') else None) self._dependency_id = as_text(obj.get('dependency_id', None)) - self.ttl = int(obj.get('ttl', -1)) + self.ttl = int(obj.get('ttl')) if obj.get('ttl') else None self.meta = unpickle(obj.get('meta')) if obj.get('meta') else {} def to_dict(self):