|
| 1 | +import logging |
| 2 | +import traceback |
| 3 | +from cStringIO import StringIO |
| 4 | + |
| 5 | +from psycopg2 import OperationalError |
| 6 | + |
| 7 | +import openerp |
| 8 | +from openerp import http |
| 9 | +from openerp.service.model import PG_CONCURRENCY_ERRORS_TO_RETRY |
| 10 | + |
| 11 | +from ..session import ConnectorSessionHandler |
| 12 | +from ..queue.job import (OpenERPJobStorage, |
| 13 | + ENQUEUED) |
| 14 | +from ..exception import (NoSuchJobError, |
| 15 | + NotReadableJobError, |
| 16 | + RetryableJobError, |
| 17 | + FailedJobError, |
| 18 | + NothingToDoJob) |
| 19 | + |
| 20 | +_logger = logging.getLogger(__name__) |
| 21 | + |
| 22 | +PG_RETRY = 5 # seconds |
| 23 | + |
| 24 | + |
| 25 | +# TODO: perhaps the notion of ConnectionSession is less important |
| 26 | +# now that we are running jobs inside a normal Odoo worker |
| 27 | + |
| 28 | + |
| 29 | +class RunJobController(http.Controller): |
| 30 | + |
| 31 | + job_storage_class = OpenERPJobStorage |
| 32 | + |
| 33 | + def _load_job(self, session, job_uuid): |
| 34 | + """ Reload a job from the backend """ |
| 35 | + try: |
| 36 | + job = self.job_storage_class(session).load(job_uuid) |
| 37 | + except NoSuchJobError: |
| 38 | + # just skip it |
| 39 | + job = None |
| 40 | + except NotReadableJobError: |
| 41 | + _logger.exception('Could not read job: %s', job_uuid) |
| 42 | + raise |
| 43 | + return job |
| 44 | + |
| 45 | + @http.route('/connector/runjob', type='http', auth='none') |
| 46 | + def runjob(self, db, job_uuid, **kw): |
| 47 | + |
| 48 | + session_hdl = ConnectorSessionHandler(db, |
| 49 | + openerp.SUPERUSER_ID) |
| 50 | + |
| 51 | + def retry_postpone(job, message, seconds=None): |
| 52 | + with session_hdl.session() as session: |
| 53 | + job.postpone(result=message, seconds=seconds) |
| 54 | + job.set_pending(self) |
| 55 | + self.job_storage_class(session).store(job) |
| 56 | + |
| 57 | + with session_hdl.session() as session: |
| 58 | + job = self._load_job(session, job_uuid) |
| 59 | + if job is None: |
| 60 | + return "" |
| 61 | + |
| 62 | + try: |
| 63 | + # if the job has been manually set to DONE or PENDING, |
| 64 | + # or if something tries to run a job that is not enqueued |
| 65 | + # before its execution, stop |
| 66 | + if job.state != ENQUEUED: |
| 67 | + _logger.warning('job %s is in state %s ' |
| 68 | + 'instead of enqueued in /runjob', |
| 69 | + job.state, job_uuid) |
| 70 | + return |
| 71 | + |
| 72 | + with session_hdl.session() as session: |
| 73 | + # TODO: set_started should be done atomically with |
| 74 | + # update queue_job set=state=started |
| 75 | + # where state=enqueid and id= |
| 76 | + job.set_started() |
| 77 | + self.job_storage_class(session).store(job) |
| 78 | + |
| 79 | + _logger.debug('%s started', job) |
| 80 | + with session_hdl.session() as session: |
| 81 | + job.perform(session) |
| 82 | + job.set_done() |
| 83 | + self.job_storage_class(session).store(job) |
| 84 | + _logger.debug('%s done', job) |
| 85 | + |
| 86 | + except NothingToDoJob as err: |
| 87 | + if unicode(err): |
| 88 | + msg = unicode(err) |
| 89 | + else: |
| 90 | + msg = None |
| 91 | + job.cancel(msg) |
| 92 | + with session_hdl.session() as session: |
| 93 | + self.job_storage_class(session).store(job) |
| 94 | + |
| 95 | + except RetryableJobError as err: |
| 96 | + # delay the job later, requeue |
| 97 | + retry_postpone(job, unicode(err)) |
| 98 | + _logger.debug('%s postponed', job) |
| 99 | + |
| 100 | + except OperationalError as err: |
| 101 | + # Automatically retry the typical transaction serialization errors |
| 102 | + if err.pgcode not in PG_CONCURRENCY_ERRORS_TO_RETRY: |
| 103 | + raise |
| 104 | + retry_postpone(job, unicode(err), seconds=PG_RETRY) |
| 105 | + _logger.debug('%s OperationalError, postponed', job) |
| 106 | + |
| 107 | + except (FailedJobError, Exception): |
| 108 | + buff = StringIO() |
| 109 | + traceback.print_exc(file=buff) |
| 110 | + _logger.error(buff.getvalue()) |
| 111 | + |
| 112 | + job.set_failed(exc_info=buff.getvalue()) |
| 113 | + with session_hdl.session() as session: |
| 114 | + self.job_storage_class(session).store(job) |
| 115 | + raise |
| 116 | + |
| 117 | + return "" |
0 commit comments