04.07.2018       Выпуск 237 (02.07.2018 - 08.07.2018)       Статьи

Django. Как сделать поиск по нескольким моделям данных Django

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

Читать>>




Экспериментальная функция:

Ниже вы видите текст статьи по ссылке. По нему можно быстро понять ссылка достойна прочтения или нет

Просим обратить внимание, что текст по ссылке и здесь может не совпадать.

В предыдущих статьях мы рассматривали, как сделать поиск на сайте на сайте. А именно:

Но что если у вас больше, чем один тип контента. У вас могут быть статьи, комментарии, форум и сообщения на форуме. Как тогда быть?

Если Вы хотите сделать всё самостоятельно, без применения сторонних библиотек, то тогда нужно будет сделать поиск по всем необходимым моделям и объединить результат. У меня сделано точно также на сайте.

urls.py

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

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

В файле urls.py пропишем маршрут для поисковой выдачи

app_name = 'home'
urlpatterns = [
    path('search/', views.SearchView.as_view(), name='search'),
]

views.py

Допустим у нас есть несколько типов контента:

Нужно в представлении выполнить поиск по все видам контента и объединить их в один QuerySet и подготовить для пагинации и выдачи

from django.shortcuts import render
from django.core.paginator import Paginator, PageNotAnInteger, EmptyPage
from django.views import View

from .models import Article, Comment, Topic, Post


class ESearchView(View):
    template_name = 'search/index.html'

    def get(self, request, *args, **kwargs):
        context = {}

        q = request.GET.get('q')
        if q:
            query_sets = []  # Общий QuerySet
            
            # Ищем по всем моделям
            query_sets.append(Article.objects.search(query=q))
            query_sets.append(Comment.objects.search(query=q))
            query_sets.append(Topic.objects.search(query=q))
            query_sets.append(Post.objects.search(query=q))

            # и объединяем выдачу
            final_set = list(chain(*query_sets))
            final_set.sort(key=lambda x: x.pub_date, reverse=True)  # Выполняем сортировку

            context['last_question'] = '?q=%s' % query_sets

            current_page = Paginator(final_set, 10)

            page = request.GET.get('page')
            try:
                context['object_list'] = current_page.page(page)
            except PageNotAnInteger:
                context['object_list'] = current_page.page(1)
            except EmptyPage:
                context['object_list'] = current_page.page(current_page.num_pages)

        return render(request=request, template_name=self.template_name, context=context)

Нюанс выше приведённого кода в том, что мы объединяем все модели данных, а также выполняем их сортировку по дате. Чтобы это было возможным, мы воспользуемся возможностями языка программирования Python, а именно утиной типизацией. Сортировка по дате стала возможной потому, что все модели данных имеют поле даты публикации с одинаковым названием pub_date.

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

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

ArticleManager

Рассмотрим менеджер модели статей. Наследуем его от базового менеджера моделей и определим метод search с логикой поиска. Аналогичным образом нужно будет прописать данный метод для всех моделей, в которых будет выполняться поиск.

from django.db import models
from django.db.models import Q

class ArticleManager(models.Manager):
    use_for_related_fields = True

    def search(self, query=None):
        qs = self.get_queryset()
        if query:
            or_lookup = (Q(title__icontains=query) | Q(content__icontains=query))
            qs = qs.filter(or_lookup)

        return qs

Установка менеджера в модель

class Article(models.Model):
    objects = ArticleManager()

search/index.html

А для шаблона поиска можно использовать немного модифицированный шаблон из одной из первых статей по организации поиска.

{% load bootstrap34%}
{% block page %}
    <h1>Поиск</h1>
    {% if object_list %}
        {% for object in object_list %}
            <div>
                <a href="{{ object.get_absolute_url }}">
                    <h2>{{ object.itle }}</h2>
                </a>
                {{ object.content|safe }}
                <p><a class="btn btn-default btn-sm" href="{{ object.get_absolute_url }}">Читать далее</a></p>
            </div>
        {% endfor %}
        {% bootstrap_pagination object_list url=last_question %}
    {% else %}
        <p>Не найдено публикаций по вашему запросу<br>Попробуйте повторить запрос с другой формулировкой</p>
    {% endif %}
{% endblock %}

Специально использую в данном случае обезличенный параметр object, который не имеет указания на конкретный тип контента, чтобы было ясно, что там может находиться какой угодно объект контента. Главное, чтобы были реализованы все методы у всех моделей, которые применяются в данном шаблоне.

Для Django рекомендую VDS-хостинг TIMEWEB



Лучшая Python рассылка




Разместим вашу рекламу

Пиши: mail@pythondigest.ru

Нашли опечатку?

Выделите фрагмент и отправьте нажатием Ctrl+Enter.

Система Orphus