Evgenii Legotckoi
10 мая 2017 г. 22:38

Django - Урок 024. Частые опросы с AJAX

Частые опросы с AJAX позволяют организовать постоянное соединение браузера с сервером для того, чтобы обновлять какие-нибудь данные, например, имеются ли новые уведомления на сайте для пользователя. Например, я организовал небольшую систему уведомлений для зарегистрированных пользователей, которая позволяет им при входе на сайт узнать, были ли ответы в статьях и вопросах форуме, на которые они подписаны, а также, появлялись ли новые статьи и вопросы на форуме, на разделы которых пользователи также были подписаны.

Уведомления выглядят следующим образом:


Принцип работы частых опросов

Принцип заключается в том, что из браузера пользователя с определённой периодичностью посылаются запросы, которые проверяют, имеются ли какие-нибудь изменения на сервере или нет. В случае, если изменения имеются, то сервер посылает ответ с этими изменениями, в противном случае посылает отрицательный результат.

Для организации периодичности опроса можно использовать функцию setInterval() , которая задаёт функцию и период вызова этой функции.

  1. setInterval(function () {
  2. $.ajax({
  3. url: "/get_notifications/",
  4. type: 'POST',
  5. data: {'check': true},
  6.  
  7. success: function (json) {
  8. if (json.result) {
  9. $('#notify_icon').addClass("notification");
  10. var doc = $.parseHTML(json.notifications_list);
  11. $('#notifications-list').html(doc);
  12. }
  13. }
  14. });
  15. }, 60000);

В данном случае с периодичностью один раз в минуту вызывается проверка уведомлений, и если результат проверки положительный то в панели уведомлений заменяются уведомления теми, которые были присланы с сервера. А также подсвечивается красным цветом колокольчик уведомлений.

Поскольку сайт работает на VDS сервере и на данный момент не имеет такой нагрузки, которая была бы для сервера ощутимой, то я делаю рендеринг уведомлений сразу на сервере и высылаю уже готовый html код, который требуется добавить на страницу в необходимом месте. При возрастании нагрузки, конечно, будет рассматриваться вариант пересылки чисто информации, которая требуется для вставки на сайт, а шаблонизация будет производиться уже javascript кодом. Но это будет реализовано позже

Что касается выше приведённого кода, то:

  1. // Подсвечиваем колокольчик
  2. $('#notify_icon').addClass("notification");
  3. // Создаём HTML код из JSON переменной
  4. var doc = $.parseHTML(json.notifications_list);
  5. // Заменяем html код внутри панели уведомлений
  6. $('#notifications-list').html(doc);

Со стороны Django необходимо написать view, которое будет отвечать за проверку наличия уведомлений у пользователя:

  1. # -*- coding: utf-8 -*-
  2.  
  3. import json
  4.  
  5. from django.http import HttpResponse
  6. from django.views import View
  7. from django.template.loader import render_to_string
  8.  
  9. class CheckNoticeView(View):
  10. def post(self, request):
  11.  
  12. result = request.user.notice_set.has_unreaded()
  13.  
  14. if result:
  15. return HttpResponse(
  16. json.dumps({
  17. "result": result,
  18. "notifications_list": render_to_string('notifications_list.html', {'user': request.user}),
  19. }),
  20. content_type="application/json"
  21. )
  22. else:
  23. return HttpResponse(
  24. json.dumps({
  25. "result": result,
  26. }),
  27. content_type="application/json"
  28. )

В данном случае запросы выполняются только для авторизованных пользователей, что контролируется декоратором login_required() в файле urls.py. Поэтому из запроса забираем пользователя и проверяем наличие непрочитанных уведомлений.

  1. request.user.notice_set.has_unreaded()
  • notice_set - это query_set уведомлений, которые относятся к пользователю
  • has_unreaded() - это специальный метод в кастомном ModelManager, которые возвращает true, если есть непрочитанные уведомления и false в противном случае.

Дальше, если имеются уведомления, то последние 5 штук рендерятся в HTML код сортируясь по дате и статусу (Прочитано/Не прочитано).

В самом простом шаблоне это может выглядеть так:

  1. {% for notice in user.notice_set.all|dictsortreversed:"date"|dictsort:"is_readed"|slice:":5" %}
  2. <div>
  3. {{ notice.content }}
  4. </div>
  5. {% endfor %}

В файле urls.py подключение view будет выглядеть следующим образом:

  1. url(r'^notice/(?P<pk>\d+)/read/$',
  2. login_required(views.ReadNoticeView.as_view()),
  3. name='notice_read'),

Выводы

Таким образом можно организовать опросы сервера, которые будут забирать новые уведомления с сайта для авторизованных пользователей. Это самый простой механизм периодического обновления информации на странице без перезагрузки, но и самый затратный по ресурсам.

Если у вас сайт находится на VDS сервере и при этом нагрузка на текущий момент небольшая, а функционал очень хочется внедрить, то это будет самый простой и эффективный способ внедрения некоторого реалтайма на сайте, но нужно отслеживать нагрузку и при значительном возрастании посещаемости переходить на более эффективные методики обновления информации, такие как Long Polling и Web Sockets.

Для Django рекомендую VDS-сервера хостера Timeweb .

Вам это нравится? Поделитесь в социальных сетях!

d
  • 26 февраля 2018 г. 18:50

спасибо, продолжай в том же духе

Комментарии

Только авторизованные пользователи могут публиковать комментарии.
Пожалуйста, авторизуйтесь или зарегистрируйтесь