beat.py 20 KB

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