5

我需要 Haystack 搜索来排除一些具有 value 的对象,published=False到目前为止我管理它的方式是添加exclude(published=True)如下:

class MymodelIndex(indexes.RealTimeSearchIndex, indexes.Indexable):
    def get_queryset(self):
        return Mymodel.objects.all().exclude(published=False)

它按预期工作,问题是./manage.py rebuild_index每次将新对象添加到数据库时我都需要这样做,这很糟糕。

我怎样才能在不需要运行其他任何东西的情况下做到这一点?

笔记:

Haystack 上的索引适用于许多模型,因此如下所示:

search = (
    SearchQuerySet().filter(content=term)
)

返回多种对象,而不仅仅是一个模型。

谢谢

4

4 回答 4

3

我最近不得不做类似的事情,这很痛苦。我找不到任何其他方法来做到这一点。

首先解决 Haystack 在许多模型上工作的问题,因此 filter 返回所有匹配项:

Haystack 使用它索引的名为 django_ct 的属性在后台处理模型过滤,该属性等于应用程序名称和模型名称。在我的特殊情况下,它看起来像django_ct='books.Title'.

您可以尝试通过执行过滤

SearchQuerySet.filter(content=term, django_ct='app.Model')

但我不知道它是否会那样工作。在我的特殊情况下,无论如何我都必须进行原始搜索,因此我可以直接将过滤添加到其中:

sqs = SearchQuerySet()
sqs = sqs.raw_search(u'(title:(%s)^500 OR author:"%s"^400 OR "%s"~2 OR (%s)) AND (django_ct:books.Title)' % term)

不管你如何得到它,在你得到SearchQuerySet你想要在不更新索引的情况下进行额外过滤之后,你必须用你自己的代码来做。

# each item in a queryset has a pk property to the model instance it references
pks = [item.pk for item in list(sqs)] # have to wrap sqs in a list otherwise it causes problems

# use those pks to create a standard django queryset object
results = Model.objects.filter(pk__in=pks)

# Now you can do any additional filtering like normal
results = results.exclude(published=False)

当然,您可以结合最后两个查询,我只是将它们拆分为明确的。

代码不多,但由于各种原因,我花了很长时间才让它工作。希望它可以帮助你。

于 2013-03-12T23:24:18.350 回答
2

从 haystack 2.4.0 开始,您可以提高haystack.exceptions.SkipDocument跳过使用容易排除的单个记录index_queryset

https://github.com/django-haystack/django-haystack/releases/tag/v2.4.0

于 2016-07-20T17:44:25.707 回答
1

在 Haystack 中有一种按对象过滤的方法。models我遇到了与上述类似的问题,并在 Haystack SearchQuerySetAPI 文档中遇到了该方法。

从干草堆(http://django-haystack.readthedocs.org/en/latest/searchqueryset_api.html#models

SearchQuerySet.models(self, *models)

接受任意数量的模型类以包含在搜索中。这会将搜索结果缩小到仅包含来自指定模型的结果。

例子:

SearchQuerySet().filter(content='foo').models(BlogEntry, Comment)

因此,您可以从这里根据需要按对象内的内容进行过滤。

于 2014-10-11T13:19:33.727 回答
0

感谢@SI Eric,我能够找到答案,它也有点老套,但它有效。

search = (
    SearchQuerySet().filter(content=term)
)

search_list = search[:]
unpublished_Mymodels = Mymodel.objects.filter(published=False)

for match in search_list:
    try:
        if match.model_name == 'Mymodel':
            if match._get_object() in unpublished_Mymodels:
                search_list.remove(match)
    except:
        pass

search = search_list
于 2013-03-13T00:45:50.333 回答