2021-12-12 10:23:44 -07:00
|
|
|
defmodule Pleroma.Search.Elasticsearch do
|
|
|
|
@behaviour Pleroma.Search
|
|
|
|
|
|
|
|
alias Pleroma.Web.MastodonAPI.StatusView
|
2021-12-12 11:25:20 -07:00
|
|
|
alias Pleroma.Web.ActivityPub.Visibility
|
2021-12-12 10:23:44 -07:00
|
|
|
|
|
|
|
defp to_es(term) when is_binary(term) do
|
|
|
|
%{
|
|
|
|
match: %{
|
|
|
|
content: %{
|
|
|
|
query: term,
|
|
|
|
operator: "AND"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
defp to_es({:quoted, term}), do: to_es(term)
|
|
|
|
|
|
|
|
defp to_es({:filter, ["hashtag", query]}) do
|
|
|
|
%{
|
|
|
|
term: %{
|
|
|
|
hashtags: %{
|
|
|
|
value: query
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
defp to_es({:filter, [field, query]}) do
|
|
|
|
%{
|
|
|
|
term: %{
|
|
|
|
field => %{
|
|
|
|
value: query
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
defp parse(query) do
|
|
|
|
query
|
|
|
|
|> SearchParser.parse!()
|
|
|
|
|> Enum.map(&to_es/1)
|
|
|
|
end
|
|
|
|
|
|
|
|
@impl Pleroma.Search
|
|
|
|
def search(%{assigns: %{user: user}} = _conn, %{q: query} = _params, _options) do
|
|
|
|
q = %{
|
2021-12-14 06:58:16 -07:00
|
|
|
size: 500,
|
|
|
|
terminate_after: 500,
|
|
|
|
timeout: "10s",
|
|
|
|
sort: [
|
|
|
|
%{"_timestamp" => "desc"}
|
|
|
|
],
|
2021-12-12 10:23:44 -07:00
|
|
|
query: %{
|
|
|
|
bool: %{
|
2021-12-12 12:31:47 -07:00
|
|
|
must: parse(String.trim(query))
|
2021-12-12 10:23:44 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
out = Pleroma.Elasticsearch.search_activities(q)
|
|
|
|
|
|
|
|
with {:ok, raw_results} <- out do
|
|
|
|
results =
|
|
|
|
raw_results
|
|
|
|
|> Map.get(:body, %{})
|
|
|
|
|> Map.get("hits", %{})
|
|
|
|
|> Map.get("hits", [])
|
|
|
|
|> Enum.map(fn result -> result["_id"] end)
|
|
|
|
|> Pleroma.Activity.all_by_ids_with_object()
|
2021-12-14 06:58:16 -07:00
|
|
|
|> Enum.filter(fn x -> Visibility.visible_for_user?(x, user) end)
|
|
|
|
|> Enum.reverse()
|
2021-12-12 10:23:44 -07:00
|
|
|
|
|
|
|
%{
|
|
|
|
"accounts" => [],
|
|
|
|
"hashtags" => [],
|
|
|
|
"statuses" =>
|
|
|
|
StatusView.render("index.json",
|
|
|
|
activities: results,
|
|
|
|
for: user,
|
|
|
|
as: :activity
|
|
|
|
)
|
|
|
|
}
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|