713b2187b9
- Remove trigram as it tends to rank garbage results highly, resulting in it prioritized above fts, which gives actually decent results. ACKed by kaniini and lain on irc. - Remove a test for handling misspelled requests, since we no longer have trigram - Remove a test for searching users with `nil` display names, because it is unrealistic, we don't accept usernames that are not >1 char strings - Make rank boosting for followers/followees sane again, previous values resulted in garbage matches getting on top just because the users are followers/followees
172 lines
4.7 KiB
Elixir
172 lines
4.7 KiB
Elixir
# Pleroma: A lightweight social networking server
|
|
# Copyright © 2017-2019 Pleroma Authors <https://pleroma.social/>
|
|
# SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
defmodule Pleroma.User.Search do
|
|
alias Pleroma.Pagination
|
|
alias Pleroma.User
|
|
import Ecto.Query
|
|
|
|
@similarity_threshold 0.25
|
|
@limit 20
|
|
|
|
def search(query_string, opts \\ []) do
|
|
resolve = Keyword.get(opts, :resolve, false)
|
|
following = Keyword.get(opts, :following, false)
|
|
result_limit = Keyword.get(opts, :limit, @limit)
|
|
offset = Keyword.get(opts, :offset, 0)
|
|
|
|
for_user = Keyword.get(opts, :for_user)
|
|
|
|
query_string = format_query(query_string)
|
|
|
|
maybe_resolve(resolve, for_user, query_string)
|
|
|
|
results =
|
|
query_string
|
|
|> search_query(for_user, following)
|
|
|> Pagination.fetch_paginated(%{"offset" => offset, "limit" => result_limit}, :offset)
|
|
|
|
results
|
|
end
|
|
|
|
defp format_query(query_string) do
|
|
# Strip the beginning @ off if there is a query
|
|
query_string = String.trim_leading(query_string, "@")
|
|
|
|
with [name, domain] <- String.split(query_string, "@"),
|
|
formatted_domain <- String.replace(domain, ~r/[!-\-|@|[-`|{-~|\/|:|\s]+/, "") do
|
|
name <> "@" <> to_string(:idna.encode(formatted_domain))
|
|
else
|
|
_ -> query_string
|
|
end
|
|
end
|
|
|
|
defp search_query(query_string, for_user, following) do
|
|
for_user
|
|
|> base_query(following)
|
|
|> filter_blocked_user(for_user)
|
|
|> filter_blocked_domains(for_user)
|
|
|> fts_subquery(query_string)
|
|
|> subquery()
|
|
|> where([u], u.search_rank > @similarity_threshold)
|
|
|> boost_search_rank(for_user)
|
|
|> order_by(desc: :search_rank)
|
|
|> maybe_restrict_local(for_user)
|
|
end
|
|
|
|
@nickname_regex ~r/^[a-zA-Z0-9.!#$%&'*+\/=?^_`{|}~\-@]+$/
|
|
defp fts_subquery(query, query_string) do
|
|
{nickname_weight, name_weight} =
|
|
if String.match?(query_string, @nickname_regex) do
|
|
{"A", "B"}
|
|
else
|
|
{"B", "A"}
|
|
end
|
|
|
|
query_string = to_tsquery(query_string)
|
|
|
|
from(
|
|
u in query,
|
|
select_merge: %{
|
|
search_rank:
|
|
fragment(
|
|
"""
|
|
ts_rank_cd((setweight(to_tsvector('simple', ?), ?) || setweight(to_tsvector('simple', ?), ?)), to_tsquery('simple', ?))
|
|
""",
|
|
u.name,
|
|
^name_weight,
|
|
u.nickname,
|
|
^nickname_weight,
|
|
^query_string
|
|
)
|
|
}
|
|
)
|
|
end
|
|
|
|
defp to_tsquery(query_string) do
|
|
String.trim_trailing(query_string, "@" <> local_domain())
|
|
|> String.replace(~r/[!-\/|@|[-`|{-~|:-?]+/, " ")
|
|
|> String.trim()
|
|
|> String.split()
|
|
|> Enum.map(&(&1 <> ":*"))
|
|
|> Enum.join(" | ")
|
|
end
|
|
|
|
defp base_query(_user, false), do: User
|
|
defp base_query(user, true), do: User.get_followers_query(user)
|
|
|
|
defp filter_blocked_user(query, %User{info: %{blocks: blocks}})
|
|
when length(blocks) > 0 do
|
|
from(q in query, where: not (q.ap_id in ^blocks))
|
|
end
|
|
|
|
defp filter_blocked_user(query, _), do: query
|
|
|
|
defp filter_blocked_domains(query, %User{info: %{domain_blocks: domain_blocks}})
|
|
when length(domain_blocks) > 0 do
|
|
domains = Enum.join(domain_blocks, ",")
|
|
|
|
from(
|
|
q in query,
|
|
where: fragment("substring(ap_id from '.*://([^/]*)') NOT IN (?)", ^domains)
|
|
)
|
|
end
|
|
|
|
defp filter_blocked_domains(query, _), do: query
|
|
|
|
defp maybe_resolve(true, user, query) do
|
|
case {limit(), user} do
|
|
{:all, _} -> :noop
|
|
{:unauthenticated, %User{}} -> User.get_or_fetch(query)
|
|
{:unauthenticated, _} -> :noop
|
|
{false, _} -> User.get_or_fetch(query)
|
|
end
|
|
end
|
|
|
|
defp maybe_resolve(_, _, _), do: :noop
|
|
|
|
defp maybe_restrict_local(q, user) do
|
|
case {limit(), user} do
|
|
{:all, _} -> restrict_local(q)
|
|
{:unauthenticated, %User{}} -> q
|
|
{:unauthenticated, _} -> restrict_local(q)
|
|
{false, _} -> q
|
|
end
|
|
end
|
|
|
|
defp limit, do: Pleroma.Config.get([:instance, :limit_to_local_content], :unauthenticated)
|
|
|
|
defp restrict_local(q), do: where(q, [u], u.local == true)
|
|
|
|
defp local_domain, do: Pleroma.Config.get([Pleroma.Web.Endpoint, :url, :host])
|
|
|
|
defp boost_search_rank(query, %User{} = for_user) do
|
|
friends_ids = User.get_friends_ids(for_user)
|
|
followers_ids = User.get_followers_ids(for_user)
|
|
|
|
from(u in subquery(query),
|
|
select_merge: %{
|
|
search_rank:
|
|
fragment(
|
|
"""
|
|
CASE WHEN (?) THEN (?) * 1.5
|
|
WHEN (?) THEN (?) * 1.3
|
|
WHEN (?) THEN (?) * 1.1
|
|
ELSE (?) END
|
|
""",
|
|
u.id in ^friends_ids and u.id in ^followers_ids,
|
|
u.search_rank,
|
|
u.id in ^friends_ids,
|
|
u.search_rank,
|
|
u.id in ^followers_ids,
|
|
u.search_rank,
|
|
u.search_rank
|
|
)
|
|
}
|
|
)
|
|
end
|
|
|
|
defp boost_search_rank(query, _for_user), do: query
|
|
end
|