periodic-tasks.rst 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456
  1. .. _guide-beat:
  2. ================
  3. Periodic Tasks
  4. ================
  5. .. contents::
  6. :local:
  7. Introduction
  8. ============
  9. :program:`celery beat` is a scheduler; It kicks off tasks at regular intervals,
  10. that are then executed by available worker nodes in the cluster.
  11. By default the entries are taken from the :setting:`beat_schedule` setting,
  12. but custom stores can also be used, like storing the entries in a SQL database.
  13. You have to ensure only a single scheduler is running for a schedule
  14. at a time, otherwise you'd end up with duplicate tasks. Using
  15. a centralized approach means the schedule doesn't have to be synchronized,
  16. and the service can operate without using locks.
  17. .. _beat-timezones:
  18. Time Zones
  19. ==========
  20. The periodic task schedules uses the UTC time zone by default,
  21. but you can change the time zone used using the :setting:`timezone`
  22. setting.
  23. An example time zone could be `Europe/London`:
  24. .. code-block:: python
  25. timezone = 'Europe/London'
  26. This setting must be added to your app, either by configuration it directly
  27. using (``app.conf.timezone = 'Europe/London'``), or by adding
  28. it to your configuration module if you have set one up using
  29. ``app.config_from_object``. See :ref:`celerytut-configuration` for
  30. more information about configuration options.
  31. The default scheduler (storing the schedule in the :file:`celerybeat-schedule`
  32. file) will automatically detect that the time zone has changed, and so will
  33. reset the schedule itself, but other schedulers may not be so smart (e.g., the
  34. Django database scheduler, see below) and in that case you'll have to reset the
  35. schedule manually.
  36. .. admonition:: Django Users
  37. Celery recommends and is compatible with the new ``USE_TZ`` setting introduced
  38. in Django 1.4.
  39. For Django users the time zone specified in the ``TIME_ZONE`` setting
  40. will be used, or you can specify a custom time zone for Celery alone
  41. by using the :setting:`timezone` setting.
  42. The database scheduler won't reset when timezone related settings
  43. change, so you must do this manually:
  44. .. code-block:: console
  45. $ python manage.py shell
  46. >>> from djcelery.models import PeriodicTask
  47. >>> PeriodicTask.objects.update(last_run_at=None)
  48. .. _beat-entries:
  49. Entries
  50. =======
  51. To call a task periodically you have to add an entry to the
  52. beat schedule list.
  53. .. code-block:: python
  54. from celery import Celery
  55. from celery.schedules import crontab
  56. app = Celery()
  57. @app.on_after_configure.connect
  58. def setup_periodic_tasks(sender, **kwargs):
  59. # Calls test('hello') every 10 seconds.
  60. sender.add_periodic_task(10.0, test.s('hello'), name='add every 10')
  61. # Calls test('world') every 30 seconds
  62. sender.add_periodic_task(30.0, test.s('world'), expires=10)
  63. # Executes every Monday morning at 7:30 a.m.
  64. sender.add_periodic_task(
  65. crontab(hour=7, minute=30, day_of_week=1),
  66. test.s('Happy Mondays!'),
  67. )
  68. @app.task
  69. def test(arg):
  70. print(arg)
  71. Setting these up from within the :data:`~@on_after_configure` handler means
  72. that we'll not evaluate the app at module level when using ``test.s()``.
  73. The :meth:`~@add_periodic_task` function will add the entry to the
  74. :setting:`beat_schedule` setting behind the scenes, and the same setting
  75. can also can be used to set up periodic tasks manually:
  76. Example: Run the `tasks.add` task every 30 seconds.
  77. .. code-block:: python
  78. app.conf.beat_schedule = {
  79. 'add-every-30-seconds': {
  80. 'task': 'tasks.add',
  81. 'schedule': 30.0,
  82. 'args': (16, 16)
  83. },
  84. }
  85. app.conf.timezone = 'UTC'
  86. .. note::
  87. If you're wondering where these settings should go then
  88. please see :ref:`celerytut-configuration`. You can either
  89. set these options on your app directly or you can keep
  90. a separate module for configuration.
  91. If you want to use a single item tuple for `args`, don't forget
  92. that the constructor is a comma, and not a pair of parentheses.
  93. Using a :class:`~datetime.timedelta` for the schedule means the task will
  94. be sent in 30 second intervals (the first task will be sent 30 seconds
  95. after `celery beat` starts, and then every 30 seconds
  96. after the last run).
  97. A Crontab like schedule also exists, see the section on `Crontab schedules`_.
  98. Like with :command:`cron`, the tasks may overlap if the first task doesn't complete
  99. before the next. If that's a concern you should use a locking
  100. strategy to ensure only one instance can run at a time (see for example
  101. :ref:`cookbook-task-serial`).
  102. .. _beat-entry-fields:
  103. Available Fields
  104. ----------------
  105. * `task`
  106. The name of the task to execute.
  107. * `schedule`
  108. The frequency of execution.
  109. This can be the number of seconds as an integer, a
  110. :class:`~datetime.timedelta`, or a :class:`~celery.schedules.crontab`.
  111. You can also define your own custom schedule types, by extending the
  112. interface of :class:`~celery.schedules.schedule`.
  113. * `args`
  114. Positional arguments (:class:`list` or :class:`tuple`).
  115. * `kwargs`
  116. Keyword arguments (:class:`dict`).
  117. * `options`
  118. Execution options (:class:`dict`).
  119. This can be any argument supported by
  120. :meth:`~celery.task.base.Task.apply_async` --
  121. `exchange`, `routing_key`, `expires`, and so on.
  122. * `relative`
  123. If `relative` is true :class:`~datetime.timedelta` schedules are scheduled
  124. "by the clock." This means the frequency is rounded to the nearest
  125. second, minute, hour or day depending on the period of the
  126. :class:`~datetime.timedelta`.
  127. By default `relative` is false, the frequency isn't rounded and will be
  128. relative to the time when :program:`celery beat` was started.
  129. .. _beat-crontab:
  130. Crontab schedules
  131. =================
  132. If you want more control over when the task is executed, for
  133. example, a particular time of day or day of the week, you can use
  134. the :class:`~celery.schedules.crontab` schedule type:
  135. .. code-block:: python
  136. from celery.schedules import crontab
  137. app.conf.beat_schedule = {
  138. # Executes every Monday morning at 7:30 a.m.
  139. 'add-every-monday-morning': {
  140. 'task': 'tasks.add',
  141. 'schedule': crontab(hour=7, minute=30, day_of_week=1),
  142. 'args': (16, 16),
  143. },
  144. }
  145. The syntax of these Crontab expressions are very flexible.
  146. Some examples:
  147. +-----------------------------------------+--------------------------------------------+
  148. | **Example** | **Meaning** |
  149. +-----------------------------------------+--------------------------------------------+
  150. | ``crontab()`` | Execute every minute. |
  151. +-----------------------------------------+--------------------------------------------+
  152. | ``crontab(minute=0, hour=0)`` | Execute daily at midnight. |
  153. +-----------------------------------------+--------------------------------------------+
  154. | ``crontab(minute=0, hour='*/3')`` | Execute every three hours: |
  155. | | midnight, 3am, 6am, 9am, |
  156. | | noon, 3pm, 6pm, 9pm. |
  157. +-----------------------------------------+--------------------------------------------+
  158. | ``crontab(minute=0,`` | Same as previous. |
  159. | ``hour='0,3,6,9,12,15,18,21')`` | |
  160. +-----------------------------------------+--------------------------------------------+
  161. | ``crontab(minute='*/15')`` | Execute every 15 minutes. |
  162. +-----------------------------------------+--------------------------------------------+
  163. | ``crontab(day_of_week='sunday')`` | Execute every minute (!) at Sundays. |
  164. +-----------------------------------------+--------------------------------------------+
  165. | ``crontab(minute='*',`` | Same as previous. |
  166. | ``hour='*',`` | |
  167. | ``day_of_week='sun')`` | |
  168. +-----------------------------------------+--------------------------------------------+
  169. | ``crontab(minute='*/10',`` | Execute every ten minutes, but only |
  170. | ``hour='3,17,22',`` | between 3-4 am, 5-6 pm, and 10-11 pm on |
  171. | ``day_of_week='thu,fri')`` | Thursdays or Fridays. |
  172. +-----------------------------------------+--------------------------------------------+
  173. | ``crontab(minute=0, hour='*/2,*/3')`` | Execute every even hour, and every hour |
  174. | | divisible by three. This means: |
  175. | | at every hour *except*: 1am, |
  176. | | 5am, 7am, 11am, 1pm, 5pm, 7pm, |
  177. | | 11pm |
  178. +-----------------------------------------+--------------------------------------------+
  179. | ``crontab(minute=0, hour='*/5')`` | Execute hour divisible by 5. This means |
  180. | | that it is triggered at 3pm, not 5pm |
  181. | | (since 3pm equals the 24-hour clock |
  182. | | value of "15", which is divisible by 5). |
  183. +-----------------------------------------+--------------------------------------------+
  184. | ``crontab(minute=0, hour='*/3,8-17')`` | Execute every hour divisible by 3, and |
  185. | | every hour during office hours (8am-5pm). |
  186. +-----------------------------------------+--------------------------------------------+
  187. | ``crontab(0, 0, day_of_month='2')`` | Execute on the second day of every month. |
  188. | | |
  189. +-----------------------------------------+--------------------------------------------+
  190. | ``crontab(0, 0,`` | Execute on every even numbered day. |
  191. | ``day_of_month='2-30/3')`` | |
  192. +-----------------------------------------+--------------------------------------------+
  193. | ``crontab(0, 0,`` | Execute on the first and third weeks of |
  194. | ``day_of_month='1-7,15-21')`` | the month. |
  195. +-----------------------------------------+--------------------------------------------+
  196. | ``crontab(0, 0, day_of_month='11',`` | Execute on the eleventh of May every year. |
  197. | ``month_of_year='5')`` | |
  198. +-----------------------------------------+--------------------------------------------+
  199. | ``crontab(0, 0,`` | Execute on the first month of every |
  200. | ``month_of_year='*/3')`` | quarter. |
  201. +-----------------------------------------+--------------------------------------------+
  202. See :class:`celery.schedules.crontab` for more documentation.
  203. .. _beat-solar:
  204. Solar schedules
  205. =================
  206. If you have a task that should be executed according to sunrise,
  207. sunset, dawn or dusk, you can use the
  208. :class:`~celery.schedules.solar` schedule type:
  209. .. code-block:: python
  210. from celery.schedules import solar
  211. app.conf.beat_schedule = {
  212. # Executes at sunset in Melbourne
  213. 'add-at-melbourne-sunset': {
  214. 'task': 'tasks.add',
  215. 'schedule': solar('sunset', -37.81753, 144.96715),
  216. 'args': (16, 16),
  217. },
  218. }
  219. The arguments are simply: ``solar(event, latitude, longitude)``
  220. Be sure to use the correct sign for latitude and longitude:
  221. +---------------+-------------------+----------------------+
  222. | **Sign** | **Argument** | **Meaning** |
  223. +---------------+-------------------+----------------------+
  224. | ``+`` | ``latitude`` | North |
  225. +---------------+-------------------+----------------------+
  226. | ``-`` | ``latitude`` | South |
  227. +---------------+-------------------+----------------------+
  228. | ``+`` | ``longitude`` | East |
  229. +---------------+-------------------+----------------------+
  230. | ``-`` | ``longitude`` | West |
  231. +---------------+-------------------+----------------------+
  232. Possible event types are:
  233. +-----------------------------------------+--------------------------------------------+
  234. | **Event** | **Meaning** |
  235. +-----------------------------------------+--------------------------------------------+
  236. | ``dawn_astronomical`` | Execute at the moment after which the sky |
  237. | | is no longer completely dark. This is when |
  238. | | the sun is 18 degrees below the horizon. |
  239. +-----------------------------------------+--------------------------------------------+
  240. | ``dawn_nautical`` | Execute when there's enough sunlight for |
  241. | | the horizon and some objects to be |
  242. | | distinguishable; formally, when the sun is |
  243. | | 12 degrees below the horizon. |
  244. +-----------------------------------------+--------------------------------------------+
  245. | ``dawn_civil`` | Execute when there's enough light for |
  246. | | objects to be distinguishable so that |
  247. | | outdoor activities can commence; |
  248. | | formally, when the Sun is 6 degrees below |
  249. | | the horizon. |
  250. +-----------------------------------------+--------------------------------------------+
  251. | ``sunrise`` | Execute when the upper edge of the sun |
  252. | | appears over the eastern horizon in the |
  253. | | morning. |
  254. +-----------------------------------------+--------------------------------------------+
  255. | ``solar_noon`` | Execute when the sun is highest above the |
  256. | | horizon on that day. |
  257. +-----------------------------------------+--------------------------------------------+
  258. | ``sunset`` | Execute when the trailing edge of the sun |
  259. | | disappears over the western horizon in the |
  260. | | evening. |
  261. +-----------------------------------------+--------------------------------------------+
  262. | ``dusk_civil`` | Execute at the end of civil twilight, when |
  263. | | objects are still distinguishable and some |
  264. | | stars and planets are visible. Formally, |
  265. | | when the sun is 6 degrees below the |
  266. | | horizon. |
  267. +-----------------------------------------+--------------------------------------------+
  268. | ``dusk_nautical`` | Execute when the sun is 12 degrees below |
  269. | | the horizon. Objects are no longer |
  270. | | distinguishable, and the horizon is no |
  271. | | longer visible to the naked eye. |
  272. +-----------------------------------------+--------------------------------------------+
  273. | ``dusk_astronomical`` | Execute at the moment after which the sky |
  274. | | becomes completely dark; formally, when |
  275. | | the sun is 18 degrees below the horizon. |
  276. +-----------------------------------------+--------------------------------------------+
  277. All solar events are calculated using UTC, and are therefore
  278. unaffected by your timezone setting.
  279. In polar regions, the sun may not rise or set every day. The scheduler
  280. is able to handle these cases (i.e., a ``sunrise`` event won't run on a day
  281. when the sun doesn't rise). The one exception is ``solar_noon``, which is
  282. formally defined as the moment the sun transits the celestial meridian,
  283. and will occur every day even if the sun is below the horizon.
  284. Twilight is defined as the period between dawn and sunrise; and between
  285. sunset and dusk. You can schedule an event according to "twilight"
  286. depending on your definition of twilight (civil, nautical, or astronomical),
  287. and whether you want the event to take place at the beginning or end
  288. of twilight, using the appropriate event from the list above.
  289. See :class:`celery.schedules.solar` for more documentation.
  290. .. _beat-starting:
  291. Starting the Scheduler
  292. ======================
  293. To start the :program:`celery beat` service:
  294. .. code-block:: console
  295. $ celery -A proj beat
  296. You can also embed `beat` inside the worker by enabling the
  297. workers :option:`-B <celery worker -B>` option, this is convenient if you'll
  298. never run more than one worker node, but it's not commonly used and for that
  299. reason isn't recommended for production use:
  300. .. code-block:: console
  301. $ celery -A proj worker -B
  302. Beat needs to store the last run times of the tasks in a local database
  303. file (named `celerybeat-schedule` by default), so it needs access to
  304. write in the current directory, or alternatively you can specify a custom
  305. location for this file:
  306. .. code-block:: console
  307. $ celery -A proj beat -s /home/celery/var/run/celerybeat-schedule
  308. .. note::
  309. To daemonize beat see :ref:`daemonizing`.
  310. .. _beat-custom-schedulers:
  311. Using custom scheduler classes
  312. ------------------------------
  313. Custom scheduler classes can be specified on the command-line (the
  314. :option:`-S <celery beat -S>` argument).
  315. The default scheduler is the :class:`celery.beat.PersistentScheduler`,
  316. that simply keeps track of the last run times in a local :mod:`shelve`
  317. database file.
  318. There's also the :pypi:`django-celery-beat` extension that stores the schedule
  319. in the Django database, and presents a convenient admin interface to manage
  320. periodic tasks at runtime.
  321. To install and use this extension:
  322. #. Use :command:`pip` to install the package:
  323. .. code-block:: console
  324. $ pip install django-celery-beat
  325. #. Add the ``django_celery_beat`` module to ``INSTALLED_APPS`` in your
  326. Django project' :file:`settings.py`::
  327. INSTALLED_APPS = (
  328. ...,
  329. 'django_celery_beat',
  330. )
  331. Note that there is no dash in the module name, only underscores.
  332. #. Apply Django database migrations so that the necessary tables are created:
  333. .. code-block:: console
  334. $ python manage.py migrate
  335. #. Start the :program:`celery beat` service using the ``django`` scheduler:
  336. .. code-block:: console
  337. $ celery -A proj beat -l info -S django
  338. #. Visit the Django-Admin interface to set up some periodic tasks.