beat.py 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682
  1. # -*- coding: utf-8 -*-
  2. """The periodic task scheduler."""
  3. from __future__ import absolute_import, unicode_literals
  4. import copy
  5. import errno
  6. import heapq
  7. import os
  8. import shelve
  9. import sys
  10. import time
  11. import traceback
  12. from collections import namedtuple
  13. from functools import total_ordering
  14. from threading import Event, Thread
  15. from billiard import ensure_multiprocessing
  16. from billiard.common import reset_signals
  17. from billiard.context import Process
  18. from kombu.utils.functional import maybe_evaluate, reprcall
  19. from kombu.utils.objects import cached_property
  20. from . import __version__, platforms, signals
  21. from .five import (items, monotonic, python_2_unicode_compatible, reraise,
  22. values)
  23. from .schedules import crontab, maybe_schedule
  24. from .utils.imports import load_extension_class_names, symbol_by_name
  25. from .utils.log import get_logger, iter_open_logger_fds
  26. from .utils.time import humanize_seconds
  27. __all__ = (
  28. 'SchedulingError', 'ScheduleEntry', 'Scheduler',
  29. 'PersistentScheduler', 'Service', 'EmbeddedService',
  30. )
  31. event_t = namedtuple('event_t', ('time', 'priority', 'entry'))
  32. logger = get_logger(__name__)
  33. debug, info, error, warning = (logger.debug, logger.info,
  34. logger.error, logger.warning)
  35. DEFAULT_MAX_INTERVAL = 300 # 5 minutes
  36. class SchedulingError(Exception):
  37. """An error occurred while scheduling a task."""
  38. @total_ordering
  39. @python_2_unicode_compatible
  40. class ScheduleEntry(object):
  41. """An entry in the scheduler.
  42. Arguments:
  43. name (str): see :attr:`name`.
  44. schedule (~celery.schedules.schedule): see :attr:`schedule`.
  45. args (Tuple): see :attr:`args`.
  46. kwargs (Dict): see :attr:`kwargs`.
  47. options (Dict): see :attr:`options`.
  48. last_run_at (~datetime.datetime): see :attr:`last_run_at`.
  49. total_run_count (int): see :attr:`total_run_count`.
  50. relative (bool): Is the time relative to when the server starts?
  51. """
  52. #: The task name
  53. name = None
  54. #: The schedule (:class:`~celery.schedules.schedule`)
  55. schedule = None
  56. #: Positional arguments to apply.
  57. args = None
  58. #: Keyword arguments to apply.
  59. kwargs = None
  60. #: Task execution options.
  61. options = None
  62. #: The time and date of when this task was last scheduled.
  63. last_run_at = None
  64. #: Total number of times this task has been scheduled.
  65. total_run_count = 0
  66. def __init__(self, name=None, task=None, last_run_at=None,
  67. total_run_count=None, schedule=None, args=(), kwargs={},
  68. options={}, relative=False, app=None):
  69. self.app = app
  70. self.name = name
  71. self.task = task
  72. self.args = args
  73. self.kwargs = kwargs
  74. self.options = options
  75. self.schedule = maybe_schedule(schedule, relative, app=self.app)
  76. self.last_run_at = last_run_at or self.default_now()
  77. self.total_run_count = total_run_count or 0
  78. def default_now(self):
  79. return self.schedule.now() if self.schedule else self.app.now()
  80. _default_now = default_now # compat
  81. def _next_instance(self, last_run_at=None):
  82. """Return new instance, with date and count fields updated."""
  83. return self.__class__(**dict(
  84. self,
  85. last_run_at=last_run_at or self.default_now(),
  86. total_run_count=self.total_run_count + 1,
  87. ))
  88. __next__ = next = _next_instance # for 2to3
  89. def __reduce__(self):
  90. return self.__class__, (
  91. self.name, self.task, self.last_run_at, self.total_run_count,
  92. self.schedule, self.args, self.kwargs, self.options,
  93. )
  94. def update(self, other):
  95. """Update values from another entry.
  96. Will only update "editable" fields:
  97. ``task``, ``schedule``, ``args``, ``kwargs``, ``options``.
  98. """
  99. self.__dict__.update({
  100. 'task': other.task, 'schedule': other.schedule,
  101. 'args': other.args, 'kwargs': other.kwargs,
  102. 'options': other.options,
  103. })
  104. def is_due(self):
  105. """See :meth:`~celery.schedule.schedule.is_due`."""
  106. return self.schedule.is_due(self.last_run_at)
  107. def __iter__(self):
  108. return iter(items(vars(self)))
  109. def __repr__(self):
  110. return '<{name}: {0.name} {call} {0.schedule}'.format(
  111. self,
  112. call=reprcall(self.task, self.args or (), self.kwargs or {}),
  113. name=type(self).__name__,
  114. )
  115. def __lt__(self, other):
  116. if isinstance(other, ScheduleEntry):
  117. # How the object is ordered doesn't really matter, as
  118. # in the scheduler heap, the order is decided by the
  119. # preceding members of the tuple ``(time, priority, entry)``.
  120. #
  121. # If all that's left to order on is the entry then it can
  122. # just as well be random.
  123. return id(self) < id(other)
  124. return NotImplemented
  125. def editable_fields_equal(self, other):
  126. for attr in ('task', 'args', 'kwargs', 'options', 'schedule'):
  127. if getattr(self, attr) != getattr(other, attr):
  128. return False
  129. return True
  130. def __eq__(self, other):
  131. """Test schedule entries equality.
  132. Will only compare "editable" fields:
  133. ``task``, ``schedule``, ``args``, ``kwargs``, ``options``.
  134. """
  135. return self.editable_fields_equal(other)
  136. def __ne__(self, other):
  137. """Test schedule entries inequality.
  138. Will only compare "editable" fields:
  139. ``task``, ``schedule``, ``args``, ``kwargs``, ``options``.
  140. """
  141. return not self == other
  142. class Scheduler(object):
  143. """Scheduler for periodic tasks.
  144. The :program:`celery beat` program may instantiate this class
  145. multiple times for introspection purposes, but then with the
  146. ``lazy`` argument set. It's important for subclasses to
  147. be idempotent when this argument is set.
  148. Arguments:
  149. schedule (~celery.schedules.schedule): see :attr:`schedule`.
  150. max_interval (int): see :attr:`max_interval`.
  151. lazy (bool): Don't set up the schedule.
  152. """
  153. Entry = ScheduleEntry
  154. #: The schedule dict/shelve.
  155. schedule = None
  156. #: Maximum time to sleep between re-checking the schedule.
  157. max_interval = DEFAULT_MAX_INTERVAL
  158. #: How often to sync the schedule (3 minutes by default)
  159. sync_every = 3 * 60
  160. #: How many tasks can be called before a sync is forced.
  161. sync_every_tasks = None
  162. _last_sync = None
  163. _tasks_since_sync = 0
  164. logger = logger # compat
  165. def __init__(self, app, schedule=None, max_interval=None,
  166. Producer=None, lazy=False, sync_every_tasks=None, **kwargs):
  167. self.app = app
  168. self.data = maybe_evaluate({} if schedule is None else schedule)
  169. self.max_interval = (max_interval or
  170. app.conf.beat_max_loop_interval or
  171. self.max_interval)
  172. self.Producer = Producer or app.amqp.Producer
  173. self._heap = None
  174. self.old_schedulers = None
  175. self.sync_every_tasks = (
  176. app.conf.beat_sync_every if sync_every_tasks is None
  177. else sync_every_tasks)
  178. if not lazy:
  179. self.setup_schedule()
  180. def install_default_entries(self, data):
  181. entries = {}
  182. if self.app.conf.result_expires and \
  183. not self.app.backend.supports_autoexpire:
  184. if 'celery.backend_cleanup' not in data:
  185. entries['celery.backend_cleanup'] = {
  186. 'task': 'celery.backend_cleanup',
  187. 'schedule': crontab('0', '4', '*'),
  188. 'options': {'expires': 12 * 3600}}
  189. self.update_from_dict(entries)
  190. def apply_entry(self, entry, producer=None):
  191. info('Scheduler: Sending due task %s (%s)', entry.name, entry.task)
  192. try:
  193. result = self.apply_async(entry, producer=producer, advance=False)
  194. except Exception as exc: # pylint: disable=broad-except
  195. error('Message Error: %s\n%s',
  196. exc, traceback.format_stack(), exc_info=True)
  197. else:
  198. debug('%s sent. id->%s', entry.task, result.id)
  199. def adjust(self, n, drift=-0.010):
  200. if n and n > 0:
  201. return n + drift
  202. return n
  203. def is_due(self, entry):
  204. return entry.is_due()
  205. def _when(self, entry, next_time_to_run, mktime=time.mktime):
  206. adjust = self.adjust
  207. return (mktime(entry.default_now().timetuple()) +
  208. (adjust(next_time_to_run) or 0))
  209. def populate_heap(self, event_t=event_t, heapify=heapq.heapify):
  210. """Populate the heap with the data contained in the schedule."""
  211. priority = 5
  212. self._heap = []
  213. for entry in values(self.schedule):
  214. is_due, next_call_delay = entry.is_due()
  215. self._heap.append(event_t(
  216. self._when(
  217. entry,
  218. 0 if is_due else next_call_delay
  219. ) or 0,
  220. priority, entry
  221. ))
  222. heapify(self._heap)
  223. # pylint disable=redefined-outer-name
  224. def tick(self, event_t=event_t, min=min, heappop=heapq.heappop,
  225. heappush=heapq.heappush):
  226. """Run a tick - one iteration of the scheduler.
  227. Executes one due task per call.
  228. Returns:
  229. float: preferred delay in seconds for next call.
  230. """
  231. adjust = self.adjust
  232. max_interval = self.max_interval
  233. if (self._heap is None or
  234. not self.schedules_equal(self.old_schedulers, self.schedule)):
  235. self.old_schedulers = copy.copy(self.schedule)
  236. self.populate_heap()
  237. H = self._heap
  238. if not H:
  239. return max_interval
  240. event = H[0]
  241. entry = event[2]
  242. is_due, next_time_to_run = self.is_due(entry)
  243. if is_due:
  244. verify = heappop(H)
  245. if verify is event:
  246. next_entry = self.reserve(entry)
  247. self.apply_entry(entry, producer=self.producer)
  248. heappush(H, event_t(self._when(next_entry, next_time_to_run),
  249. event[1], next_entry))
  250. return 0
  251. else:
  252. heappush(H, verify)
  253. return min(verify[0], max_interval)
  254. return min(adjust(next_time_to_run) or max_interval, max_interval)
  255. def schedules_equal(self, old_schedules, new_schedules):
  256. if set(old_schedules.keys()) != set(new_schedules.keys()):
  257. return False
  258. for name, old_entry in old_schedules.items():
  259. new_entry = new_schedules.get(name)
  260. if not new_entry:
  261. return False
  262. if new_entry != old_entry:
  263. return False
  264. return True
  265. def should_sync(self):
  266. return (
  267. (not self._last_sync or
  268. (monotonic() - self._last_sync) > self.sync_every) or
  269. (self.sync_every_tasks and
  270. self._tasks_since_sync >= self.sync_every_tasks)
  271. )
  272. def reserve(self, entry):
  273. new_entry = self.schedule[entry.name] = next(entry)
  274. return new_entry
  275. def apply_async(self, entry, producer=None, advance=True, **kwargs):
  276. # Update time-stamps and run counts before we actually execute,
  277. # so we have that done if an exception is raised (doesn't schedule
  278. # forever.)
  279. entry = self.reserve(entry) if advance else entry
  280. task = self.app.tasks.get(entry.task)
  281. try:
  282. if task:
  283. return task.apply_async(entry.args, entry.kwargs,
  284. producer=producer,
  285. **entry.options)
  286. else:
  287. return self.send_task(entry.task, entry.args, entry.kwargs,
  288. producer=producer,
  289. **entry.options)
  290. except Exception as exc: # pylint: disable=broad-except
  291. reraise(SchedulingError, SchedulingError(
  292. "Couldn't apply scheduled task {0.name}: {exc}".format(
  293. entry, exc=exc)), sys.exc_info()[2])
  294. finally:
  295. self._tasks_since_sync += 1
  296. if self.should_sync():
  297. self._do_sync()
  298. def send_task(self, *args, **kwargs):
  299. return self.app.send_task(*args, **kwargs)
  300. def setup_schedule(self):
  301. self.install_default_entries(self.data)
  302. def _do_sync(self):
  303. try:
  304. debug('beat: Synchronizing schedule...')
  305. self.sync()
  306. finally:
  307. self._last_sync = monotonic()
  308. self._tasks_since_sync = 0
  309. def sync(self):
  310. pass
  311. def close(self):
  312. self.sync()
  313. def add(self, **kwargs):
  314. entry = self.Entry(app=self.app, **kwargs)
  315. self.schedule[entry.name] = entry
  316. return entry
  317. def _maybe_entry(self, name, entry):
  318. if isinstance(entry, self.Entry):
  319. entry.app = self.app
  320. return entry
  321. return self.Entry(**dict(entry, name=name, app=self.app))
  322. def update_from_dict(self, dict_):
  323. self.schedule.update({
  324. name: self._maybe_entry(name, entry)
  325. for name, entry in items(dict_)
  326. })
  327. def merge_inplace(self, b):
  328. schedule = self.schedule
  329. A, B = set(schedule), set(b)
  330. # Remove items from disk not in the schedule anymore.
  331. for key in A ^ B:
  332. schedule.pop(key, None)
  333. # Update and add new items in the schedule
  334. for key in B:
  335. entry = self.Entry(**dict(b[key], name=key, app=self.app))
  336. if schedule.get(key):
  337. schedule[key].update(entry)
  338. else:
  339. schedule[key] = entry
  340. def _ensure_connected(self):
  341. # callback called for each retry while the connection
  342. # can't be established.
  343. def _error_handler(exc, interval):
  344. error('beat: Connection error: %s. '
  345. 'Trying again in %s seconds...', exc, interval)
  346. return self.connection.ensure_connection(
  347. _error_handler, self.app.conf.broker_connection_max_retries
  348. )
  349. def get_schedule(self):
  350. return self.data
  351. def set_schedule(self, schedule):
  352. self.data = schedule
  353. schedule = property(get_schedule, set_schedule)
  354. @cached_property
  355. def connection(self):
  356. return self.app.connection_for_write()
  357. @cached_property
  358. def producer(self):
  359. return self.Producer(self._ensure_connected(), auto_declare=False)
  360. @property
  361. def info(self):
  362. return ''
  363. class PersistentScheduler(Scheduler):
  364. """Scheduler backed by :mod:`shelve` database."""
  365. persistence = shelve
  366. known_suffixes = ('', '.db', '.dat', '.bak', '.dir')
  367. _store = None
  368. def __init__(self, *args, **kwargs):
  369. self.schedule_filename = kwargs.get('schedule_filename')
  370. Scheduler.__init__(self, *args, **kwargs)
  371. def _remove_db(self):
  372. for suffix in self.known_suffixes:
  373. with platforms.ignore_errno(errno.ENOENT):
  374. os.remove(self.schedule_filename + suffix)
  375. def _open_schedule(self):
  376. return self.persistence.open(self.schedule_filename, writeback=True)
  377. def _destroy_open_corrupted_schedule(self, exc):
  378. error('Removing corrupted schedule file %r: %r',
  379. self.schedule_filename, exc, exc_info=True)
  380. self._remove_db()
  381. return self._open_schedule()
  382. def setup_schedule(self):
  383. try:
  384. self._store = self._open_schedule()
  385. # In some cases there may be different errors from a storage
  386. # backend for corrupted files. Example - DBPageNotFoundError
  387. # exception from bsddb. In such case the file will be
  388. # successfully opened but the error will be raised on first key
  389. # retrieving.
  390. self._store.keys()
  391. except Exception as exc: # pylint: disable=broad-except
  392. self._store = self._destroy_open_corrupted_schedule(exc)
  393. self._create_schedule()
  394. tz = self.app.conf.timezone
  395. stored_tz = self._store.get(str('tz'))
  396. if stored_tz is not None and stored_tz != tz:
  397. warning('Reset: Timezone changed from %r to %r', stored_tz, tz)
  398. self._store.clear() # Timezone changed, reset db!
  399. utc = self.app.conf.enable_utc
  400. stored_utc = self._store.get(str('utc_enabled'))
  401. if stored_utc is not None and stored_utc != utc:
  402. choices = {True: 'enabled', False: 'disabled'}
  403. warning('Reset: UTC changed from %s to %s',
  404. choices[stored_utc], choices[utc])
  405. self._store.clear() # UTC setting changed, reset db!
  406. entries = self._store.setdefault(str('entries'), {})
  407. self.merge_inplace(self.app.conf.beat_schedule)
  408. self.install_default_entries(self.schedule)
  409. self._store.update({
  410. str('__version__'): __version__,
  411. str('tz'): tz,
  412. str('utc_enabled'): utc,
  413. })
  414. self.sync()
  415. debug('Current schedule:\n' + '\n'.join(
  416. repr(entry) for entry in values(entries)))
  417. def _create_schedule(self):
  418. for _ in (1, 2):
  419. try:
  420. self._store[str('entries')]
  421. except KeyError:
  422. # new schedule db
  423. try:
  424. self._store[str('entries')] = {}
  425. except KeyError as exc:
  426. self._store = self._destroy_open_corrupted_schedule(exc)
  427. continue
  428. else:
  429. if str('__version__') not in self._store:
  430. warning('DB Reset: Account for new __version__ field')
  431. self._store.clear() # remove schedule at 2.2.2 upgrade.
  432. elif str('tz') not in self._store:
  433. warning('DB Reset: Account for new tz field')
  434. self._store.clear() # remove schedule at 3.0.8 upgrade
  435. elif str('utc_enabled') not in self._store:
  436. warning('DB Reset: Account for new utc_enabled field')
  437. self._store.clear() # remove schedule at 3.0.9 upgrade
  438. break
  439. def get_schedule(self):
  440. return self._store[str('entries')]
  441. def set_schedule(self, schedule):
  442. self._store[str('entries')] = schedule
  443. schedule = property(get_schedule, set_schedule)
  444. def sync(self):
  445. if self._store is not None:
  446. self._store.sync()
  447. def close(self):
  448. self.sync()
  449. self._store.close()
  450. @property
  451. def info(self):
  452. return ' . db -> {self.schedule_filename}'.format(self=self)
  453. class Service(object):
  454. """Celery periodic task service."""
  455. scheduler_cls = PersistentScheduler
  456. def __init__(self, app, max_interval=None, schedule_filename=None,
  457. scheduler_cls=None):
  458. self.app = app
  459. self.max_interval = (max_interval or
  460. app.conf.beat_max_loop_interval)
  461. self.scheduler_cls = scheduler_cls or self.scheduler_cls
  462. self.schedule_filename = (
  463. schedule_filename or app.conf.beat_schedule_filename)
  464. self._is_shutdown = Event()
  465. self._is_stopped = Event()
  466. def __reduce__(self):
  467. return self.__class__, (self.max_interval, self.schedule_filename,
  468. self.scheduler_cls, self.app)
  469. def start(self, embedded_process=False):
  470. info('beat: Starting...')
  471. debug('beat: Ticking with max interval->%s',
  472. humanize_seconds(self.scheduler.max_interval))
  473. signals.beat_init.send(sender=self)
  474. if embedded_process:
  475. signals.beat_embedded_init.send(sender=self)
  476. platforms.set_process_title('celery beat')
  477. try:
  478. while not self._is_shutdown.is_set():
  479. interval = self.scheduler.tick()
  480. if interval and interval > 0.0:
  481. debug('beat: Waking up %s.',
  482. humanize_seconds(interval, prefix='in '))
  483. time.sleep(interval)
  484. if self.scheduler.should_sync():
  485. self.scheduler._do_sync()
  486. except (KeyboardInterrupt, SystemExit):
  487. self._is_shutdown.set()
  488. finally:
  489. self.sync()
  490. def sync(self):
  491. self.scheduler.close()
  492. self._is_stopped.set()
  493. def stop(self, wait=False):
  494. info('beat: Shutting down...')
  495. self._is_shutdown.set()
  496. wait and self._is_stopped.wait() # block until shutdown done.
  497. def get_scheduler(self, lazy=False,
  498. extension_namespace='celery.beat_schedulers'):
  499. filename = self.schedule_filename
  500. aliases = dict(
  501. load_extension_class_names(extension_namespace) or {})
  502. return symbol_by_name(self.scheduler_cls, aliases=aliases)(
  503. app=self.app,
  504. schedule_filename=filename,
  505. max_interval=self.max_interval,
  506. lazy=lazy,
  507. )
  508. @cached_property
  509. def scheduler(self):
  510. return self.get_scheduler()
  511. class _Threaded(Thread):
  512. """Embedded task scheduler using threading."""
  513. def __init__(self, app, **kwargs):
  514. super(_Threaded, self).__init__()
  515. self.app = app
  516. self.service = Service(app, **kwargs)
  517. self.daemon = True
  518. self.name = 'Beat'
  519. def run(self):
  520. self.app.set_current()
  521. self.service.start()
  522. def stop(self):
  523. self.service.stop(wait=True)
  524. try:
  525. ensure_multiprocessing()
  526. except NotImplementedError: # pragma: no cover
  527. _Process = None
  528. else:
  529. class _Process(Process): # noqa
  530. def __init__(self, app, **kwargs):
  531. super(_Process, self).__init__()
  532. self.app = app
  533. self.service = Service(app, **kwargs)
  534. self.name = 'Beat'
  535. def run(self):
  536. reset_signals(full=False)
  537. platforms.close_open_fds([
  538. sys.__stdin__, sys.__stdout__, sys.__stderr__,
  539. ] + list(iter_open_logger_fds()))
  540. self.app.set_default()
  541. self.app.set_current()
  542. self.service.start(embedded_process=True)
  543. def stop(self):
  544. self.service.stop()
  545. self.terminate()
  546. def EmbeddedService(app, max_interval=None, **kwargs):
  547. """Return embedded clock service.
  548. Arguments:
  549. thread (bool): Run threaded instead of as a separate process.
  550. Uses :mod:`multiprocessing` by default, if available.
  551. """
  552. if kwargs.pop('thread', False) or _Process is None:
  553. # Need short max interval to be able to stop thread
  554. # in reasonable time.
  555. return _Threaded(app, max_interval=1, **kwargs)
  556. return _Process(app, max_interval=max_interval, **kwargs)