Django динамически фильтруется с q объектами

Я пытаюсь запросить базу данных на основе тегов ввода пользователя. Количество тегов может быть от 0 до 5, поэтому мне нужно создать запрос динамически.

Итак, у меня есть список тегов, tag_list, и я хочу запросить базу данных:

design_list = Design.objects.filter(Q(tags__tag__contains = "tag1") and Q(tags__tag__contains = "tag2") and etc. etc. )

Как создать эту функцию?

Ответ 1

Вам нужно выполнить цикл через tag_list и применить фильтр для каждого из них.

tag_list = ['tag1', 'tag2', 'tag3']
base_qs = Design.objects.all()
for t in tag_list:
    base_qs = base_qs.filter(tags__tag__contains=t)

Это даст вам результаты, соответствующие всем тегам, как показано в примере с помощью and. Если на самом деле вам понадобилось or вместо этого, вам, вероятно, понадобятся объекты Q.

Изменить: Я думаю, у меня есть то, что вы ищете сейчас.

tags = ['tag1', 'tag2', 'tag3']
q_objects = Q() # Create an empty Q object to start with
for t in tags:
    q_objects |= Q(tags__tag__contains=t) # 'or' the Q objects together

designs = Design.objects.filter(q_objects)

Я тестировал это и, похоже, работает очень хорошо.

Изменить 2: Кредит на kezabelle в #django на Freenode для первоначальной идеи.

Ответ 2

Сначала подготовьте список тегов, затем выполните запрос следующим образом:

tags = ['tag1', 'tag2',...]
design_list = Design.objects.filter(tags__tag__contains__in = tags)