Advanced Django-RQ Example


This is part 4 in a 4 part series looking at how to do background/async tasks in Django. See this post for more details.

This is based on my notes from the following GitHub repository: https://github.com/stuartmaxwell/django-django_rq-advanced-example

Advanced Django-RQ Example

This builds on the previous basic Django-RQ example by adding a long-running task that is able to report its progress. In order to accurately report progress, you need to know how many things you need to do, and how many things you have completed. For example, if processing a CSV file with 1000 rows then you’ll know which row you’re currently on as well as how many rows in total. With this information you can create a percentage which becomes the progress. In this example, I added a simple sleep counter to the contact_form so that the task takes at least 60 seconds.

  • First we edit the contact_form > tasks.py file by adding the sleep function which reports progress to the RQ job by using the meta field:
from django.core.mail import send_mail, BadHeaderError
from djangorq_project.settings import DEFAULT_FROM_EMAIL
from django_rq import job
from rq import get_current_job

import logging
from time import sleep

logger = logging.getLogger(__name__)


@job
def send_email_task(to, subject, message):
    logger.info(f"from={DEFAULT_FROM_EMAIL}, {to=}, {subject=}, {message=}")

    job = get_current_job()

    secs = 30
    for sec in range(secs):
        progress = int(sec / secs * 100)
        job.meta["progress"] = progress
        job.save_meta()
        logger.debug(f"{job.meta['progress']=}")
        sleep(1)

    try:
        logger.info("About to send_mail")
        send_mail(subject, message, DEFAULT_FROM_EMAIL, [DEFAULT_FROM_EMAIL])
        job.meta["progress"] = 100
        job.save_meta()
    except BadHeaderError:
        logger.info("BadHeaderError")
    except Exception as e:
        logger.error(e)
  • To enable the logging, we need to add a basic logging configuration in the project settings.py file:
...
# Logging
LOGGING = {
    "version": 1,
    "disable_existing_loggers": False,
    "handlers": {"console": {"class": "logging.StreamHandler",},},
    "root": {"handlers": ["console"], "level": "DEBUG",},
}
  • Then we create a new view in contact_form > views.py that can be called to return the status and progress of a job as a JSON response. We also add in some logging so we can see what’s happening in the console. The ContactFormView class gets some tweaks so that the send_email function returns the job object just created. This used to get the job ID which is passed to a new template that uses JavaScript to update the page based on the progress:
from django.http import JsonResponse
from django.views import View
from django.views.generic import FormView
from django.shortcuts import render

import django_rq

import logging

from .forms import ContactFormModelForm
from .tasks import send_email_task

logger = logging.getLogger(__name__)


class ContactFormView(FormView):
    form_class = ContactFormModelForm
    template_name = "contact_form/contact_form.html"
    success_url = "/"

    def form_valid(self, form):
        form.save()
        job = self.send_email(form.cleaned_data)
        logger.debug(f"Job id: {job.id}")

        return render(self.request, "contact_form/progress.html", {"job_id": job.id},)

    def send_email(self, valid_data):
        email = valid_data["email"]
        subject = "Contact form sent from website"
        message = (
            f"You have received a contact form.\n"
            f"Email: {valid_data['email']}\n"
            f"Name: {valid_data['name']}\n"
            f"Subject: {valid_data['subject']}\n"
            f"{valid_data['message']}\n"
        )
        return send_email_task.delay(email, subject, message,)


class JobStatusView(View):
    def get(self, request, job_id):

        job = django_rq.get_queue().fetch_job(job_id)

        if job:
            response = {
                "status": job.get_status(),
                "progress": job.meta.get("progress", ""),
            }
            logger.debug(f"{job=}")
            logger.debug(f"{response=}")
        else:
            response = {"status": "invalid", "progress": ""}

        return JsonResponse(response)
  • Create a new template to show the progress in contact_form > templates > contact_form > progress.html. The example below is a basic use of using JavaScript (JQuery) to retrieve the JSON response from the new view we have just created which is updated every second. You can extend this to show a pretty progress bar that changes its CSS width property with the progress reported.
<!doctype html>
<html lang="en">
    <head>
        <meta charset="utf-8">
        <title>Progress</title>
    </head>
    <body>
        <div id="app">
            <h2>Progress</h2>
            <div id="status"></div>
            <div id="progress"></div>
        </div>
        <script src="https://code.jquery.com/jquery-3.5.0.min.js" integrity="sha256-xNzN2a4ltkB44Mc/Jz3pT4iU1cmeR0FkXs4pru/JxaQ=" crossorigin="anonymous"></script>
        <script>
            $(document).ready(function () {
                function update_progress() {
                    status_url = "{% url 'contact_form:job_status' job_id %}";

                    status_div = "#status";
                    progress_div = "#progress";

                    // send GET request to status URL
                    $.getJSON(status_url, function (data) {
                        // update UI
                        status = "Job status: " + data["status"];
                        if (data["progress"] == null) {
                            progress = "Progress: 0%";
                        } else {
                            progress = "Progress: " + data["progress"] + "%";
                        }

                        $(status_div).html(status);
                        $(progress_div).html(progress);

                        // Checks if the script is finished
                        if (data["status"] == "finished") {
                            $(status_div).html(status);
                            $(progress_div).html("Progress: 100%");
                        } else {
                            setTimeout(function () {
                                update_progress();
                            }, 1000);
                        }
                    });
                }
                update_progress();
            });
        </script>
    </body>
</html>
  • The last thing to do is to add a URL pattern in contact_form > urls.py for the new view we have created:
from django.urls import path

from .views import ContactFormView, JobStatusView

app_name = "contact_form"
urlpatterns = [
    path("", ContactFormView.as_view(), name="contact_form"),
    path("taskstatus/<str:job_id>", JobStatusView.as_view(), name="job_status"),
]

One response to “Advanced Django-RQ Example”

  1. Thanks for this tutorial. This isn’t quite working for me and I have a few questions. What function is actually performing the send mail? send_email_task() or send_mail()? I notice each function calls the other so I don’t understand how that works. As I get to the end of the progress and always see this message in my terminal output:

    About to send_mail
    [Errno 60] Operation timed out

    What’s causing that? What you describe in your introduction – processing large CSV files in the background – is exactly what I’m trying to figure out but I am running into constant obstacles and banging my head on the table in frustration. Any chance you have a working repository that demonstrates how to do this? Thank you.

Leave a Reply

Your email address will not be published. Required fields are marked *