mirror of https://github.com/peter4431/rq.git
Safer, and shorter, version of the death penalty.
This case protects against JobTimeoutExceptions being raised immediately after the job body has been (successfully) executed. Still, JobTimeoutExceptions pass through naturally, like any other exception, to be handled by the default exception handler that writes failed jobs to the failed queue. Timeouts therefore are reported like any other exception.main
parent
8a856e79ea
commit
b8305a818f
@ -0,0 +1,51 @@
|
|||||||
|
import signal
|
||||||
|
|
||||||
|
|
||||||
|
class JobTimeoutException(Exception):
|
||||||
|
"""Raised when a job takes longer to complete than the allowed maximum
|
||||||
|
timeout value.
|
||||||
|
"""
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
class death_pentalty_after(object):
|
||||||
|
def __init__(self, timeout):
|
||||||
|
self._timeout = timeout
|
||||||
|
|
||||||
|
def __enter__(self):
|
||||||
|
self.setup_death_penalty()
|
||||||
|
|
||||||
|
def __exit__(self, type, value, traceback):
|
||||||
|
# Always cancel immediately, since we're done
|
||||||
|
try:
|
||||||
|
self.cancel_death_penalty()
|
||||||
|
except JobTimeoutException:
|
||||||
|
# Weird case: we're done with the with body, but now the alarm is
|
||||||
|
# fired. We may safely ignore this situation and consider the
|
||||||
|
# body done.
|
||||||
|
pass
|
||||||
|
|
||||||
|
# __exit__ may return True to supress further exception handling. We
|
||||||
|
# don't want to suppress any exceptions here, since all errors should
|
||||||
|
# just pass through, JobTimeoutException being handled as just one of
|
||||||
|
# them.
|
||||||
|
return False
|
||||||
|
|
||||||
|
def handle_death_penalty(self, signum, frame):
|
||||||
|
raise JobTimeoutException('Job exceeded maximum timeout '
|
||||||
|
'value (%d seconds).' % self._timeout)
|
||||||
|
|
||||||
|
def setup_death_penalty(self):
|
||||||
|
"""Sets up an alarm signal and a signal handler that raises
|
||||||
|
a JobTimeoutException after the timeout amount (expressed in
|
||||||
|
seconds).
|
||||||
|
"""
|
||||||
|
signal.signal(signal.SIGALRM, self.handle_death_penalty)
|
||||||
|
signal.alarm(self._timeout)
|
||||||
|
|
||||||
|
def cancel_death_penalty(self):
|
||||||
|
"""Removes the death penalty alarm and puts back the system into
|
||||||
|
default signal handling.
|
||||||
|
"""
|
||||||
|
signal.alarm(0)
|
||||||
|
signal.signal(signal.SIGALRM, signal.SIG_DFL)
|
Loading…
Reference in New Issue