Loading...
Skip to Content

How to create periodic tasks with Celery 4 and Celery 5

Let's analyze the usage of periodic tasks as a useful example. Our task is to get information about a free ebook from Packt Publishing (it could be any other site). We'd like to have an email every day at 7.00 AM with the title of this book.

Let's look at how to do it with Celery (both 4 and 5 versions) and Django. In both cases 'sd' is the name of my Django's project. I've separated my settings into a few parts and 'settings.production' part is information that I've got a 'production.py' file in the 'settings' directory.

For Celery 4:

    in the directory of the Django's project, we should create a standard celery.py file:

import os
from celery import Celery

os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'settings.production')

app = Celery('sd')
app.config_from_object('django.conf:settings', namespace='CELERY')
app.autodiscover_tasks()


    in the '__init__.py' of the project we should add this one line:

__all__ = ['celery_app']


    inside one of the applications we should create 'tasks.py' file:

import requests

from bs4 import BeautifulSoup

from django.core.mail import EmailMessage
from celery.schedules import crontab
from celery.task import periodic_task


@periodic_task(run_every=(crontab(minute=0, hour=7)),
               name='task_send_email_about_ebook',
               ignore_result=True
               )
def task_send_email_about_ebook():
    page = requests.get('https://www.packtpub.com/free-learning')
    soup = BeautifulSoup(page.content, 'html.parser')
    book_title = soup.find('h3', attrs={'class': 'product-info__title'}).text.split('-', 1)[1].strip()
    subject = "Your free e-book from PacktPub is available!"
    message = f"Your new e-book is '{book_title}'. \n"
    message += f"Book is available at <a href='https://www.packtpub.com/packt/offers/free-learning'>Free Learning</a>."
    email = EmailMessage(subject,
                         message,
                         'from@domaine.com',
                         ['to@somedomain.com'])
    email.send(fail_silently=False)


For Celery 5:

    in the directory of the Django's project, we should create a standard 'celery.py' file:

import os
from celery import Celery

# set the default Django settings module for the 'celery' program.
os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'settings.production')

app = Celery('sd')

app.config_from_object('django.conf:settings', namespace='CELERY')
app.autodiscover_tasks()


    in the '__init__.py' of the project we should add this one line:

from .celery import app as celery_app


    inside one of the applications we should create 'tasks.py' file:

import requests

from bs4 import BeautifulSoup
from celery.schedules import crontab
from django.core.mail import EmailMessage

from sd.celery import app


@app.task
def task_send_email_about_ebook():
    page = requests.get('https://www.packtpub.com/free-learning')
    soup = BeautifulSoup(page.content, 'html.parser')
    book_title = soup.find('h3', attrs={'class': 'product-info__title'}).text.split('-', 1)[1].strip()
    subject = "Your free e-book from PacktPub is available!"
    message = f"Your new e-book is '{book_title}'. \n"
    message += f"Book is available at <a href='https://www.packtpub.com/packt/offers/free-learning'>Free Learning</a>."
    email = EmailMessage(subject,
                         message,
                         'from@domaine.com',
                         ['to@somedomain.com'])
    email.send(fail_silently=False)


app.conf.beat_schedule = {
    "task_send_email_about_ebook": {
        "task": "frontend.tasks.task_send_email_about_ebook",
        "schedule": crontab(hour=7, minute=0)
        }
    }


The difference between the two versions of Celery in defining periodic tasks is not so spectacular, but it is worth knowing it.