2024-05-18 09:43:47 +00:00
|
|
|
# Pleroma: A lightweight social networking server
|
|
|
|
# Copyright © 2017-2021 Pleroma Authors <https://pleroma.social/>
|
|
|
|
# SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
|
|
|
defmodule Pleroma.Search.QdrantSearchTest do
|
|
|
|
use Pleroma.DataCase, async: true
|
|
|
|
use Oban.Testing, repo: Pleroma.Repo
|
|
|
|
|
|
|
|
import Pleroma.Factory
|
|
|
|
import Mox
|
|
|
|
|
|
|
|
alias Pleroma.Search.QdrantSearch
|
2024-05-18 11:02:22 +00:00
|
|
|
alias Pleroma.UnstubbedConfigMock, as: Config
|
|
|
|
alias Pleroma.Web.CommonAPI
|
2024-05-18 09:43:47 +00:00
|
|
|
alias Pleroma.Workers.SearchIndexingWorker
|
|
|
|
|
|
|
|
describe "Qdrant search" do
|
2024-05-23 10:38:30 +00:00
|
|
|
test "searches for a term by encoding it and sending it to qdrant" do
|
|
|
|
user = insert(:user)
|
|
|
|
|
|
|
|
{:ok, activity} =
|
|
|
|
CommonAPI.post(user, %{
|
|
|
|
status: "guys i just don't wanna leave the swamp",
|
|
|
|
visibility: "public"
|
|
|
|
})
|
|
|
|
|
|
|
|
Config
|
|
|
|
|> expect(:get, 3, fn
|
|
|
|
[Pleroma.Search, :module], nil ->
|
|
|
|
QdrantSearch
|
|
|
|
|
|
|
|
[Pleroma.Search.QdrantSearch, key], nil ->
|
|
|
|
%{
|
|
|
|
openai_model: "a_model",
|
|
|
|
openai_url: "https://openai.url",
|
|
|
|
qdrant_url: "https://qdrant.url"
|
|
|
|
}[key]
|
|
|
|
end)
|
|
|
|
|
|
|
|
Tesla.Mock.mock(fn
|
|
|
|
%{url: "https://openai.url/v1/embeddings", method: :post} ->
|
|
|
|
Tesla.Mock.json(%{
|
|
|
|
data: [%{embedding: [1, 2, 3]}]
|
|
|
|
})
|
|
|
|
|
|
|
|
%{url: "https://qdrant.url/collections/posts/points/search", method: :post, body: body} ->
|
|
|
|
data = Jason.decode!(body)
|
|
|
|
refute data["filter"]
|
|
|
|
|
|
|
|
Tesla.Mock.json(%{
|
|
|
|
result: [%{"id" => activity.id |> FlakeId.from_string() |> Ecto.UUID.cast!()}]
|
|
|
|
})
|
|
|
|
end)
|
|
|
|
|
|
|
|
results = QdrantSearch.search(nil, "guys i just don't wanna leave the swamp", %{})
|
|
|
|
|
|
|
|
assert results == [activity]
|
|
|
|
end
|
|
|
|
|
|
|
|
test "for a given actor, ask for only relevant matches" do
|
|
|
|
user = insert(:user)
|
|
|
|
|
|
|
|
{:ok, activity} =
|
|
|
|
CommonAPI.post(user, %{
|
|
|
|
status: "guys i just don't wanna leave the swamp",
|
|
|
|
visibility: "public"
|
|
|
|
})
|
|
|
|
|
|
|
|
Config
|
|
|
|
|> expect(:get, 3, fn
|
|
|
|
[Pleroma.Search, :module], nil ->
|
|
|
|
QdrantSearch
|
|
|
|
|
|
|
|
[Pleroma.Search.QdrantSearch, key], nil ->
|
|
|
|
%{
|
|
|
|
openai_model: "a_model",
|
|
|
|
openai_url: "https://openai.url",
|
|
|
|
qdrant_url: "https://qdrant.url"
|
|
|
|
}[key]
|
|
|
|
end)
|
|
|
|
|
|
|
|
Tesla.Mock.mock(fn
|
|
|
|
%{url: "https://openai.url/v1/embeddings", method: :post} ->
|
|
|
|
Tesla.Mock.json(%{
|
|
|
|
data: [%{embedding: [1, 2, 3]}]
|
|
|
|
})
|
|
|
|
|
|
|
|
%{url: "https://qdrant.url/collections/posts/points/search", method: :post, body: body} ->
|
|
|
|
data = Jason.decode!(body)
|
|
|
|
|
|
|
|
assert data["filter"] == %{
|
|
|
|
"must" => [%{"key" => "actor", "match" => %{"value" => user.ap_id}}]
|
|
|
|
}
|
|
|
|
|
|
|
|
Tesla.Mock.json(%{
|
|
|
|
result: [%{"id" => activity.id |> FlakeId.from_string() |> Ecto.UUID.cast!()}]
|
|
|
|
})
|
|
|
|
end)
|
|
|
|
|
|
|
|
results =
|
2024-05-23 14:55:16 +00:00
|
|
|
QdrantSearch.search(nil, "guys i just don't wanna leave the swamp", %{author: user})
|
2024-05-23 10:38:30 +00:00
|
|
|
|
|
|
|
assert results == [activity]
|
|
|
|
end
|
|
|
|
|
2024-05-18 10:04:32 +00:00
|
|
|
test "indexes a public post on creation, deletes from the index on deletion" do
|
2024-05-18 09:43:47 +00:00
|
|
|
user = insert(:user)
|
|
|
|
|
|
|
|
Tesla.Mock.mock(fn
|
2024-05-19 08:17:46 +00:00
|
|
|
%{method: :post, url: "https://openai.url/v1/embeddings"} ->
|
|
|
|
send(self(), "posted_to_openai")
|
|
|
|
|
|
|
|
Tesla.Mock.json(%{
|
|
|
|
data: [%{embedding: [1, 2, 3]}]
|
|
|
|
})
|
2024-05-18 09:43:47 +00:00
|
|
|
|
|
|
|
%{method: :put, url: "https://qdrant.url/collections/posts/points", body: body} ->
|
|
|
|
send(self(), "posted_to_qdrant")
|
|
|
|
|
2024-05-23 10:38:30 +00:00
|
|
|
data = Jason.decode!(body)
|
|
|
|
%{"points" => [%{"vector" => vector, "payload" => payload}]} = data
|
|
|
|
|
|
|
|
assert vector == [1, 2, 3]
|
|
|
|
assert payload["actor"]
|
|
|
|
assert payload["published_at"]
|
2024-05-18 09:43:47 +00:00
|
|
|
|
|
|
|
Tesla.Mock.json("ok")
|
2024-05-18 10:04:32 +00:00
|
|
|
|
|
|
|
%{method: :post, url: "https://qdrant.url/collections/posts/points/delete"} ->
|
|
|
|
send(self(), "deleted_from_qdrant")
|
|
|
|
Tesla.Mock.json("ok")
|
2024-05-18 09:43:47 +00:00
|
|
|
end)
|
|
|
|
|
|
|
|
Config
|
2024-05-18 10:04:32 +00:00
|
|
|
|> expect(:get, 6, fn
|
2024-05-18 09:43:47 +00:00
|
|
|
[Pleroma.Search, :module], nil ->
|
|
|
|
QdrantSearch
|
|
|
|
|
|
|
|
[Pleroma.Search.QdrantSearch, key], nil ->
|
|
|
|
%{
|
2024-05-19 08:17:46 +00:00
|
|
|
openai_model: "a_model",
|
|
|
|
openai_url: "https://openai.url",
|
2024-05-18 09:43:47 +00:00
|
|
|
qdrant_url: "https://qdrant.url"
|
|
|
|
}[key]
|
|
|
|
end)
|
|
|
|
|
|
|
|
{:ok, activity} =
|
|
|
|
CommonAPI.post(user, %{
|
|
|
|
status: "guys i just don't wanna leave the swamp",
|
|
|
|
visibility: "public"
|
|
|
|
})
|
|
|
|
|
|
|
|
args = %{"op" => "add_to_index", "activity" => activity.id}
|
|
|
|
|
|
|
|
assert_enqueued(
|
|
|
|
worker: SearchIndexingWorker,
|
|
|
|
args: args
|
|
|
|
)
|
|
|
|
|
|
|
|
assert :ok = perform_job(SearchIndexingWorker, args)
|
2024-05-19 08:17:46 +00:00
|
|
|
assert_received("posted_to_openai")
|
2024-05-18 09:43:47 +00:00
|
|
|
assert_received("posted_to_qdrant")
|
2024-05-18 10:04:32 +00:00
|
|
|
|
|
|
|
{:ok, _} = CommonAPI.delete(activity.id, user)
|
|
|
|
|
|
|
|
delete_args = %{"op" => "remove_from_index", "object" => activity.object.id}
|
|
|
|
assert_enqueued(worker: SearchIndexingWorker, args: delete_args)
|
|
|
|
assert :ok = perform_job(SearchIndexingWorker, delete_args)
|
|
|
|
|
|
|
|
assert_received("deleted_from_qdrant")
|
2024-05-18 09:43:47 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|