beat.py 19 KB

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