Как я могу использовать SearchRank в поле модели Django?
У меня есть модель Post:
class Post(models.Model):
post_title = models.CharField(max_length=120)
post_subtitle = models.TextField()
post_text = models.TextField()
vector_column = SearchVectorField(null=True)
class Meta:
indexes = (GinIndex(fields=['vector_column']),)
и я создал триггер в базе данных для обновления значения vector_column
:
create function core_post_trigger() returns trigger as $$
begin
new.vector_column :=
setweight(to_tsvector('pg_catalog.english', coalesce(new.post_title, '')), 'A') ||
setweight(to_tsvector('pg_catalog.english', coalesce(new.post_subtitle, '')), 'B') ||
setweight(to_tsvector('pg_catalog.english', coalesce(new.post_text, '')), 'C');
return new;
end
$$ language plpgsql;
create trigger vector_column_trigger
before insert or update on core_post
for each row execute procedure
core_post_trigger();
И я ищу в этой модели следующим образом:
Post.objects.filter(vector_column=SearchQuery(query, config='english', search_type='websearch')
Хотя я применил веса в этом поиске, я не применил ранжирование (to_tsrank
). Я знаю, что в Django можно применить ранжирование следующим образом:
vector = SearchVector('post_title', weight='A', config='english) + SearchVector('post_subtitle', weight='B', config='english') +
SearchVector('post_text', weight='C', config='english')
query = SearchQuery(query, search_type="websearch")
Post.objects.annotate(rank=SearchRank(vector, query)).order_by('-rank')
Вопрос в том, что я не знаю, как использовать SearchRank
с моим vector_column
столбцом, так как этот столбец имеет GinIndex. Если я использую эту переменную vector
, созданную выше, я не буду использовать индекс, и тогда поиск будет медленным. Я пытался написать этот запрос разными способами, но не знаю, как сделать. Я пробовал, например, так:
Post.objects.annotate(SearchRank(vector_column, SearchQuery('Angel', config='english'))).order_by('-rank')
Но выдает ошибку vector_column is not defined
.
Как я могу этого достичь?
Я нашел его. Мне нужно было использовать выражение F:
Post.objects.annotate(
rank=SearchRank(F('vector_column'),
SearchQuery(query, config='english'))
).order_by('-rank'), 15