beat.py 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528
  1. # -*- coding: utf-8 -*-
  2. """
  3. celery.beat
  4. ~~~~~~~~~~~
  5. The periodic task scheduler.
  6. """
  7. from __future__ import absolute_import
  8. import errno
  9. import os
  10. import time
  11. import shelve
  12. import sys
  13. import traceback
  14. from threading import Event, Thread
  15. from billiard import Process, ensure_multiprocessing
  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 .five import items, reraise, values, monotonic
  23. from .schedules import maybe_schedule, crontab
  24. from .utils.imports import instantiate
  25. from .utils.timeutils import humanize_seconds
  26. from .utils.log import get_logger, iter_open_logger_fds
  27. __all__ = ['SchedulingError', 'ScheduleEntry', 'Scheduler',
  28. 'PersistentScheduler', 'Service', 'EmbeddedService']
  29. logger = get_logger(__name__)
  30. debug, info, error, warning = (logger.debug, logger.info,
  31. logger.error, logger.warning)
  32. DEFAULT_MAX_INTERVAL = 300 # 5 minutes
  33. class SchedulingError(Exception):
  34. """An error occured while scheduling a task."""
  35. class ScheduleEntry(object):
  36. """An entry in the scheduler.
  37. :keyword name: see :attr:`name`.
  38. :keyword schedule: see :attr:`schedule`.
  39. :keyword args: see :attr:`args`.
  40. :keyword kwargs: see :attr:`kwargs`.
  41. :keyword options: see :attr:`options`.
  42. :keyword last_run_at: see :attr:`last_run_at`.
  43. :keyword total_run_count: see :attr:`total_run_count`.
  44. :keyword relative: Is the time relative to when the server starts?
  45. """
  46. #: The task name
  47. name = None
  48. #: The schedule (run_every/crontab)
  49. schedule = None
  50. #: Positional arguments to apply.
  51. args = None
  52. #: Keyword arguments to apply.
  53. kwargs = None
  54. #: Task execution options.
  55. options = None
  56. #: The time and date of when this task was last scheduled.
  57. last_run_at = None
  58. #: Total number of times this task has been scheduled.
  59. total_run_count = 0
  60. def __init__(self, name=None, task=None, last_run_at=None,
  61. total_run_count=None, schedule=None, args=(), kwargs={},
  62. options={}, relative=False, app=None):
  63. self.app = app
  64. self.name = name
  65. self.task = task
  66. self.args = args
  67. self.kwargs = kwargs
  68. self.options = options
  69. self.schedule = maybe_schedule(schedule, relative, app=self.app)
  70. self.last_run_at = last_run_at or self._default_now()
  71. self.total_run_count = total_run_count or 0
  72. def _default_now(self):
  73. return self.schedule.now() if self.schedule else self.app.now()
  74. def _next_instance(self, last_run_at=None):
  75. """Return a new instance of the same class, but with
  76. its date and count fields updated."""
  77. return self.__class__(**dict(
  78. self,
  79. last_run_at=last_run_at or self._default_now(),
  80. total_run_count=self.total_run_count + 1,
  81. ))
  82. __next__ = next = _next_instance # for 2to3
  83. def __reduce__(self):
  84. return self.__class__, (
  85. self.name, self.task, self.last_run_at, self.total_run_count,
  86. self.schedule, self.args, self.kwargs, self.options,
  87. )
  88. def update(self, other):
  89. """Update values from another entry.
  90. Does only update "editable" fields (task, schedule, args, kwargs,
  91. options).
  92. """
  93. self.__dict__.update({'task': other.task, 'schedule': other.schedule,
  94. 'args': other.args, 'kwargs': other.kwargs,
  95. 'options': other.options})
  96. def is_due(self):
  97. """See :meth:`~celery.schedule.schedule.is_due`."""
  98. return self.schedule.is_due(self.last_run_at)
  99. def __iter__(self):
  100. return iter(items(vars(self)))
  101. def __repr__(self):
  102. return '<Entry: {0.name} {call} {0.schedule}'.format(
  103. self,
  104. call=reprcall(self.task, self.args or (), self.kwargs or {}),
  105. )
  106. class Scheduler(object):
  107. """Scheduler for periodic tasks.
  108. The :program:`celery beat` program may instantiate this class
  109. multiple times for introspection purposes, but then with the
  110. ``lazy`` argument set. It is important for subclasses to
  111. be idempotent when this argument is set.
  112. :keyword schedule: see :attr:`schedule`.
  113. :keyword max_interval: see :attr:`max_interval`.
  114. :keyword lazy: Do not set up the schedule.
  115. """
  116. Entry = ScheduleEntry
  117. #: The schedule dict/shelve.
  118. schedule = None
  119. #: Maximum time to sleep between re-checking the schedule.
  120. max_interval = DEFAULT_MAX_INTERVAL
  121. #: How often to sync the schedule (3 minutes by default)
  122. sync_every = 3 * 60
  123. _last_sync = None
  124. logger = logger # compat
  125. def __init__(self, app, schedule=None, max_interval=None,
  126. Publisher=None, lazy=False, **kwargs):
  127. self.app = app
  128. self.data = maybe_evaluate({} if schedule is None else schedule)
  129. self.max_interval = (max_interval
  130. or app.conf.CELERYBEAT_MAX_LOOP_INTERVAL
  131. or self.max_interval)
  132. self.Publisher = Publisher or app.amqp.TaskProducer
  133. if not lazy:
  134. self.setup_schedule()
  135. def install_default_entries(self, data):
  136. entries = {}
  137. if self.app.conf.CELERY_TASK_RESULT_EXPIRES and \
  138. not self.app.backend.supports_autoexpire:
  139. if 'celery.backend_cleanup' not in data:
  140. entries['celery.backend_cleanup'] = {
  141. 'task': 'celery.backend_cleanup',
  142. 'schedule': crontab('0', '4', '*'),
  143. 'options': {'expires': 12 * 3600}}
  144. self.update_from_dict(entries)
  145. def maybe_due(self, entry, publisher=None):
  146. is_due, next_time_to_run = entry.is_due()
  147. if is_due:
  148. info('Scheduler: Sending due task %s (%s)', entry.name, entry.task)
  149. try:
  150. result = self.apply_async(entry, publisher=publisher)
  151. except Exception as exc:
  152. error('Message Error: %s\n%s',
  153. exc, traceback.format_stack(), exc_info=True)
  154. else:
  155. debug('%s sent. id->%s', entry.task, result.id)
  156. return next_time_to_run
  157. def tick(self):
  158. """Run a tick, that is one iteration of the scheduler.
  159. Executes all due tasks.
  160. """
  161. remaining_times = []
  162. try:
  163. for entry in values(self.schedule):
  164. next_time_to_run = self.maybe_due(entry, self.publisher)
  165. if next_time_to_run:
  166. remaining_times.append(next_time_to_run)
  167. except RuntimeError:
  168. pass
  169. return min(remaining_times + [self.max_interval])
  170. def should_sync(self):
  171. return (not self._last_sync or
  172. (monotonic() - self._last_sync) > self.sync_every)
  173. def reserve(self, entry):
  174. new_entry = self.schedule[entry.name] = next(entry)
  175. return new_entry
  176. def apply_async(self, entry, publisher=None, **kwargs):
  177. # Update timestamps and run counts before we actually execute,
  178. # so we have that done if an exception is raised (doesn't schedule
  179. # forever.)
  180. entry = self.reserve(entry)
  181. task = self.app.tasks.get(entry.task)
  182. try:
  183. if task:
  184. result = task.apply_async(entry.args, entry.kwargs,
  185. publisher=publisher,
  186. **entry.options)
  187. else:
  188. result = self.send_task(entry.task, entry.args, entry.kwargs,
  189. publisher=publisher,
  190. **entry.options)
  191. except Exception as exc:
  192. reraise(SchedulingError, SchedulingError(
  193. "Couldn't apply scheduled task {0.name}: {exc}".format(
  194. entry, exc=exc)), sys.exc_info()[2])
  195. finally:
  196. if self.should_sync():
  197. self._do_sync()
  198. return result
  199. def send_task(self, *args, **kwargs):
  200. return self.app.send_task(*args, **kwargs)
  201. def setup_schedule(self):
  202. self.install_default_entries(self.data)
  203. def _do_sync(self):
  204. try:
  205. debug('beat: Synchronizing schedule...')
  206. self.sync()
  207. finally:
  208. self._last_sync = monotonic()
  209. def sync(self):
  210. pass
  211. def close(self):
  212. self.sync()
  213. def add(self, **kwargs):
  214. entry = self.Entry(app=self.app, **kwargs)
  215. self.schedule[entry.name] = entry
  216. return entry
  217. def _maybe_entry(self, name, entry):
  218. if isinstance(entry, self.Entry):
  219. entry.app = self.app
  220. return entry
  221. return self.Entry(**dict(entry, name=name, app=self.app))
  222. def update_from_dict(self, dict_):
  223. self.schedule.update(dict(
  224. (name, self._maybe_entry(name, entry))
  225. for name, entry in items(dict_)))
  226. def merge_inplace(self, b):
  227. schedule = self.schedule
  228. A, B = set(schedule), set(b)
  229. # Remove items from disk not in the schedule anymore.
  230. for key in A ^ B:
  231. schedule.pop(key, None)
  232. # Update and add new items in the schedule
  233. for key in B:
  234. entry = self.Entry(**dict(b[key], name=key, app=self.app))
  235. if schedule.get(key):
  236. schedule[key].update(entry)
  237. else:
  238. schedule[key] = entry
  239. def _ensure_connected(self):
  240. # callback called for each retry while the connection
  241. # can't be established.
  242. def _error_handler(exc, interval):
  243. error('beat: Connection error: %s. '
  244. 'Trying again in %s seconds...', exc, interval)
  245. return self.connection.ensure_connection(
  246. _error_handler, self.app.conf.BROKER_CONNECTION_MAX_RETRIES
  247. )
  248. def get_schedule(self):
  249. return self.data
  250. def set_schedule(self, schedule):
  251. self.data = schedule
  252. schedule = property(get_schedule, set_schedule)
  253. @cached_property
  254. def connection(self):
  255. return self.app.connection()
  256. @cached_property
  257. def publisher(self):
  258. return self.Publisher(self._ensure_connected())
  259. @property
  260. def info(self):
  261. return ''
  262. class PersistentScheduler(Scheduler):
  263. persistence = shelve
  264. known_suffixes = ('', '.db', '.dat', '.bak', '.dir')
  265. _store = None
  266. def __init__(self, *args, **kwargs):
  267. self.schedule_filename = kwargs.get('schedule_filename')
  268. Scheduler.__init__(self, *args, **kwargs)
  269. def _remove_db(self):
  270. for suffix in self.known_suffixes:
  271. with platforms.ignore_errno(errno.ENOENT):
  272. os.remove(self.schedule_filename + suffix)
  273. def setup_schedule(self):
  274. try:
  275. self._store = self.persistence.open(self.schedule_filename,
  276. writeback=True)
  277. entries = self._store.setdefault('entries', {})
  278. except Exception as exc:
  279. error('Removing corrupted schedule file %r: %r',
  280. self.schedule_filename, exc, exc_info=True)
  281. self._remove_db()
  282. self._store = self.persistence.open(self.schedule_filename,
  283. writeback=True)
  284. else:
  285. if '__version__' not in self._store:
  286. warning('Reset: Account for new __version__ field')
  287. self._store.clear() # remove schedule at 2.2.2 upgrade.
  288. if 'tz' not in self._store:
  289. warning('Reset: Account for new tz field')
  290. self._store.clear() # remove schedule at 3.0.8 upgrade
  291. if 'utc_enabled' not in self._store:
  292. warning('Reset: Account for new utc_enabled field')
  293. self._store.clear() # remove schedule at 3.0.9 upgrade
  294. tz = self.app.conf.CELERY_TIMEZONE
  295. stored_tz = self._store.get('tz')
  296. if stored_tz is not None and stored_tz != tz:
  297. warning('Reset: Timezone changed from %r to %r', stored_tz, tz)
  298. self._store.clear() # Timezone changed, reset db!
  299. utc = self.app.conf.CELERY_ENABLE_UTC
  300. stored_utc = self._store.get('utc_enabled')
  301. if stored_utc is not None and stored_utc != utc:
  302. choices = {True: 'enabled', False: 'disabled'}
  303. warning('Reset: UTC changed from %s to %s',
  304. choices[stored_utc], choices[utc])
  305. self._store.clear() # UTC setting changed, reset db!
  306. entries = self._store.setdefault('entries', {})
  307. self.merge_inplace(self.app.conf.CELERYBEAT_SCHEDULE)
  308. self.install_default_entries(self.schedule)
  309. self._store.update(__version__=__version__, tz=tz, utc_enabled=utc)
  310. self.sync()
  311. debug('Current schedule:\n' + '\n'.join(
  312. repr(entry) for entry in values(entries)))
  313. def get_schedule(self):
  314. return self._store['entries']
  315. def set_schedule(self, schedule):
  316. self._store['entries'] = schedule
  317. schedule = property(get_schedule, set_schedule)
  318. def sync(self):
  319. if self._store is not None:
  320. self._store.sync()
  321. def close(self):
  322. self.sync()
  323. self._store.close()
  324. @property
  325. def info(self):
  326. return ' . db -> {self.schedule_filename}'.format(self=self)
  327. class Service(object):
  328. scheduler_cls = PersistentScheduler
  329. def __init__(self, app, max_interval=None, schedule_filename=None,
  330. scheduler_cls=None):
  331. self.app = app
  332. self.max_interval = (max_interval
  333. or app.conf.CELERYBEAT_MAX_LOOP_INTERVAL)
  334. self.scheduler_cls = scheduler_cls or self.scheduler_cls
  335. self.schedule_filename = (
  336. schedule_filename or app.conf.CELERYBEAT_SCHEDULE_FILENAME)
  337. self._is_shutdown = Event()
  338. self._is_stopped = Event()
  339. def __reduce__(self):
  340. return self.__class__, (self.max_interval, self.schedule_filename,
  341. self.scheduler_cls, self.app)
  342. def start(self, embedded_process=False):
  343. info('beat: Starting...')
  344. debug('beat: Ticking with max interval->%s',
  345. humanize_seconds(self.scheduler.max_interval))
  346. signals.beat_init.send(sender=self)
  347. if embedded_process:
  348. signals.beat_embedded_init.send(sender=self)
  349. platforms.set_process_title('celery beat')
  350. try:
  351. while not self._is_shutdown.is_set():
  352. interval = self.scheduler.tick()
  353. debug('beat: Waking up %s.',
  354. humanize_seconds(interval, prefix='in '))
  355. time.sleep(interval)
  356. except (KeyboardInterrupt, SystemExit):
  357. self._is_shutdown.set()
  358. finally:
  359. self.sync()
  360. def sync(self):
  361. self.scheduler.close()
  362. self._is_stopped.set()
  363. def stop(self, wait=False):
  364. info('beat: Shutting down...')
  365. self._is_shutdown.set()
  366. wait and self._is_stopped.wait() # block until shutdown done.
  367. def get_scheduler(self, lazy=False):
  368. filename = self.schedule_filename
  369. scheduler = instantiate(self.scheduler_cls,
  370. app=self.app,
  371. schedule_filename=filename,
  372. max_interval=self.max_interval,
  373. lazy=lazy)
  374. return scheduler
  375. @cached_property
  376. def scheduler(self):
  377. return self.get_scheduler()
  378. class _Threaded(Thread):
  379. """Embedded task scheduler using threading."""
  380. def __init__(self, *args, **kwargs):
  381. super(_Threaded, self).__init__()
  382. self.service = Service(*args, **kwargs)
  383. self.daemon = True
  384. self.name = 'Beat'
  385. def run(self):
  386. self.service.start()
  387. def stop(self):
  388. self.service.stop(wait=True)
  389. try:
  390. ensure_multiprocessing()
  391. except NotImplementedError: # pragma: no cover
  392. _Process = None
  393. else:
  394. class _Process(Process): # noqa
  395. def __init__(self, *args, **kwargs):
  396. super(_Process, self).__init__()
  397. self.service = Service(*args, **kwargs)
  398. self.name = 'Beat'
  399. def run(self):
  400. reset_signals(full=False)
  401. platforms.close_open_fds([
  402. sys.__stdin__, sys.__stdout__, sys.__stderr__,
  403. ] + list(iter_open_logger_fds()))
  404. self.service.start(embedded_process=True)
  405. def stop(self):
  406. self.service.stop()
  407. self.terminate()
  408. def EmbeddedService(*args, **kwargs):
  409. """Return embedded clock service.
  410. :keyword thread: Run threaded instead of as a separate process.
  411. Uses :mod:`multiprocessing` by default, if available.
  412. """
  413. if kwargs.pop('thread', False) or _Process is None:
  414. # Need short max interval to be able to stop thread
  415. # in reasonable time.
  416. kwargs.setdefault('max_interval', 1)
  417. return _Threaded(*args, **kwargs)
  418. return _Process(*args, **kwargs)