Improve searching for group actors
Signed-off-by: Thomas Citharel <tcit@tcit.fr>
This commit is contained in:
parent
4fc044c595
commit
4de39d5850
@ -40,13 +40,15 @@ defmodule Mobilizon.GraphQL.API.Search do
|
|||||||
|
|
||||||
true ->
|
true ->
|
||||||
page =
|
page =
|
||||||
Actors.build_actors_by_username_or_name_page(
|
Actors.search_actors(
|
||||||
term,
|
term,
|
||||||
[
|
[
|
||||||
actor_type: [result_type],
|
actor_type: result_type,
|
||||||
radius: Map.get(args, :radius),
|
radius: Map.get(args, :radius),
|
||||||
location: Map.get(args, :location),
|
location: Map.get(args, :location),
|
||||||
minimum_visibility: Map.get(args, :minimum_visibility, :public)
|
minimum_visibility: Map.get(args, :minimum_visibility, :public),
|
||||||
|
current_actor_id: Map.get(args, :current_actor_id),
|
||||||
|
exclude_my_groups: Map.get(args, :exclude_my_groups, false)
|
||||||
],
|
],
|
||||||
page,
|
page,
|
||||||
limit
|
limit
|
||||||
|
@ -13,7 +13,7 @@ defmodule Mobilizon.GraphQL.Resolvers.Followers do
|
|||||||
@spec find_followers_for_group(Actor.t(), map(), map()) :: {:ok, Page.t()}
|
@spec find_followers_for_group(Actor.t(), map(), map()) :: {:ok, Page.t()}
|
||||||
def find_followers_for_group(
|
def find_followers_for_group(
|
||||||
%Actor{id: group_id} = group,
|
%Actor{id: group_id} = group,
|
||||||
%{page: page, limit: limit} = args,
|
args,
|
||||||
%{
|
%{
|
||||||
context: %{
|
context: %{
|
||||||
current_user: %User{role: user_role},
|
current_user: %User{role: user_role},
|
||||||
@ -21,15 +21,23 @@ defmodule Mobilizon.GraphQL.Resolvers.Followers do
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
) do
|
) do
|
||||||
|
followers = group_followers(group, args)
|
||||||
|
|
||||||
if Actors.is_moderator?(actor_id, group_id) or is_moderator(user_role) do
|
if Actors.is_moderator?(actor_id, group_id) or is_moderator(user_role) do
|
||||||
{:ok,
|
{:ok, followers}
|
||||||
Actors.list_paginated_followers_for_actor(group, Map.get(args, :approved), page, limit)}
|
|
||||||
else
|
else
|
||||||
{:error, :unauthorized}
|
{:ok, %Page{followers | elements: []}}
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def find_followers_for_group(_, _, _), do: {:error, :unauthenticated}
|
def find_followers_for_group(%Actor{} = group, args, _) do
|
||||||
|
followers = group_followers(group, args)
|
||||||
|
{:ok, %Page{followers | elements: []}}
|
||||||
|
end
|
||||||
|
|
||||||
|
defp group_followers(group, %{page: page, limit: limit} = args) do
|
||||||
|
Actors.list_paginated_followers_for_actor(group, Map.get(args, :approved), page, limit)
|
||||||
|
end
|
||||||
|
|
||||||
@spec update_follower(any(), map(), map()) :: {:ok, Follower.t()} | {:error, any()}
|
@spec update_follower(any(), map(), map()) :: {:ok, Follower.t()} | {:error, any()}
|
||||||
def update_follower(_, %{id: follower_id, approved: approved}, %{
|
def update_follower(_, %{id: follower_id, approved: approved}, %{
|
||||||
|
@ -21,7 +21,14 @@ defmodule Mobilizon.GraphQL.Resolvers.Search do
|
|||||||
"""
|
"""
|
||||||
@spec search_groups(any(), map(), Absinthe.Resolution.t()) ::
|
@spec search_groups(any(), map(), Absinthe.Resolution.t()) ::
|
||||||
{:ok, Page.t(Actor.t())} | {:error, String.t()}
|
{:ok, Page.t(Actor.t())} | {:error, String.t()}
|
||||||
def search_groups(_parent, %{page: page, limit: limit} = args, _resolution) do
|
def search_groups(
|
||||||
|
_parent,
|
||||||
|
%{page: page, limit: limit} = args,
|
||||||
|
%{context: context} = _resolution
|
||||||
|
) do
|
||||||
|
current_actor = Map.get(context, :current_actor, nil)
|
||||||
|
current_actor_id = if current_actor, do: current_actor.id, else: nil
|
||||||
|
args = Map.put(args, :current_actor_id, current_actor_id)
|
||||||
Search.search_actors(args, page, limit, :Group)
|
Search.search_actors(args, page, limit, :Group)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -59,6 +59,14 @@ defmodule Mobilizon.GraphQL.Schema.SearchType do
|
|||||||
arg(:term, :string, default_value: "", description: "Search term")
|
arg(:term, :string, default_value: "", description: "Search term")
|
||||||
arg(:location, :string, description: "A geohash for coordinates")
|
arg(:location, :string, description: "A geohash for coordinates")
|
||||||
|
|
||||||
|
arg(:exclude_my_groups, :boolean,
|
||||||
|
description: "Whether to include the groups the current actor is member or follower"
|
||||||
|
)
|
||||||
|
|
||||||
|
arg(:minimum_visibility, :group_visibility,
|
||||||
|
description: "The minimum visibility the group must have"
|
||||||
|
)
|
||||||
|
|
||||||
arg(:radius, :float,
|
arg(:radius, :float,
|
||||||
default_value: 50,
|
default_value: 50,
|
||||||
description: "Radius around the location to search in"
|
description: "Radius around the location to search in"
|
||||||
|
@ -450,29 +450,55 @@ defmodule Mobilizon.Actors do
|
|||||||
@doc """
|
@doc """
|
||||||
Builds a page struct for actors by their name or displayed name.
|
Builds a page struct for actors by their name or displayed name.
|
||||||
"""
|
"""
|
||||||
@spec build_actors_by_username_or_name_page(
|
@spec search_actors(
|
||||||
String.t(),
|
String.t(),
|
||||||
Keyword.t(),
|
Keyword.t(),
|
||||||
integer | nil,
|
integer | nil,
|
||||||
integer | nil
|
integer | nil
|
||||||
) :: Page.t()
|
) :: Page.t()
|
||||||
def build_actors_by_username_or_name_page(
|
def search_actors(
|
||||||
term,
|
term,
|
||||||
options \\ [],
|
options \\ [],
|
||||||
page \\ nil,
|
page \\ nil,
|
||||||
limit \\ nil
|
limit \\ nil
|
||||||
) do
|
) do
|
||||||
|
term
|
||||||
|
|> build_actors_by_username_or_name_page_query(options)
|
||||||
|
|> maybe_exclude_my_groups(
|
||||||
|
Keyword.get(options, :exclude_my_groups, false),
|
||||||
|
Keyword.get(options, :current_actor_id)
|
||||||
|
)
|
||||||
|
|> Page.build_page(page, limit)
|
||||||
|
end
|
||||||
|
|
||||||
|
defp maybe_exclude_my_groups(query, true, current_actor_id) when current_actor_id != nil do
|
||||||
|
query
|
||||||
|
|> join(:left, [a], m in Member, on: a.id == m.parent_id)
|
||||||
|
|> join(:left, [a], f in Follower, on: a.id == f.target_actor_id)
|
||||||
|
|> where([_a, ..., m, f], m.actor_id != ^current_actor_id and f.actor_id != ^current_actor_id)
|
||||||
|
end
|
||||||
|
|
||||||
|
defp maybe_exclude_my_groups(query, _, _), do: query
|
||||||
|
|
||||||
|
@spec build_actors_by_username_or_name_page_query(
|
||||||
|
String.t(),
|
||||||
|
Keyword.t()
|
||||||
|
) :: Ecto.Query.t()
|
||||||
|
defp build_actors_by_username_or_name_page_query(
|
||||||
|
term,
|
||||||
|
options
|
||||||
|
) do
|
||||||
anonymous_actor_id = Mobilizon.Config.anonymous_actor_id()
|
anonymous_actor_id = Mobilizon.Config.anonymous_actor_id()
|
||||||
query = from(a in Actor)
|
query = from(a in Actor)
|
||||||
|
|
||||||
query
|
query
|
||||||
|
|> distinct([q], q.id)
|
||||||
|> actor_by_username_or_name_query(term)
|
|> actor_by_username_or_name_query(term)
|
||||||
|> actors_for_location(Keyword.get(options, :location), Keyword.get(options, :radius))
|
|> actors_for_location(Keyword.get(options, :location), Keyword.get(options, :radius))
|
||||||
|> filter_by_types(Keyword.get(options, :actor_type, :Group))
|
|> filter_by_type(Keyword.get(options, :actor_type, :Group))
|
||||||
|> filter_by_minimum_visibility(Keyword.get(options, :minimum_visibility, :public))
|
|> filter_by_minimum_visibility(Keyword.get(options, :minimum_visibility, :public))
|
||||||
|> filter_suspended(false)
|
|> filter_suspended(false)
|
||||||
|> filter_out_anonymous_actor_id(anonymous_actor_id)
|
|> filter_out_anonymous_actor_id(anonymous_actor_id)
|
||||||
|> Page.build_page(page, limit)
|
|
||||||
end
|
end
|
||||||
|
|
||||||
@doc """
|
@doc """
|
||||||
@ -1313,17 +1339,15 @@ defmodule Mobilizon.Actors do
|
|||||||
@spec actors_for_location(Ecto.Queryable.t(), String.t(), integer()) :: Ecto.Query.t()
|
@spec actors_for_location(Ecto.Queryable.t(), String.t(), integer()) :: Ecto.Query.t()
|
||||||
defp actors_for_location(query, location, radius)
|
defp actors_for_location(query, location, radius)
|
||||||
when is_valid_string(location) and not is_nil(radius) do
|
when is_valid_string(location) and not is_nil(radius) do
|
||||||
with {lon, lat} <- Geohax.decode(location),
|
{lon, lat} = Geohax.decode(location)
|
||||||
point <- Geo.WKT.decode!("SRID=4326;POINT(#{lon} #{lat})") do
|
point = Geo.WKT.decode!("SRID=4326;POINT(#{lon} #{lat})")
|
||||||
|
|
||||||
query
|
query
|
||||||
|> join(:inner, [q], a in Address, on: a.id == q.physical_address_id, as: :address)
|
|> join(:inner, [q], a in Address, on: a.id == q.physical_address_id, as: :address)
|
||||||
|> where(
|
|> where(
|
||||||
[q],
|
[q],
|
||||||
st_dwithin_in_meters(^point, as(:address).geom, ^(radius * 1000))
|
st_dwithin_in_meters(^point, as(:address).geom, ^(radius * 1000))
|
||||||
)
|
)
|
||||||
else
|
|
||||||
_ -> query
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
|
|
||||||
defp actors_for_location(query, _location, _radius), do: query
|
defp actors_for_location(query, _location, _radius), do: query
|
||||||
@ -1564,11 +1588,6 @@ defmodule Mobilizon.Actors do
|
|||||||
|
|
||||||
defp filter_by_type(query, _type), do: query
|
defp filter_by_type(query, _type), do: query
|
||||||
|
|
||||||
@spec filter_by_types(Ecto.Queryable.t(), [ActorType.t()]) :: Ecto.Query.t()
|
|
||||||
defp filter_by_types(query, types) do
|
|
||||||
from(a in query, where: a.type in ^types)
|
|
||||||
end
|
|
||||||
|
|
||||||
@spec filter_by_minimum_visibility(Ecto.Queryable.t(), atom()) :: Ecto.Query.t()
|
@spec filter_by_minimum_visibility(Ecto.Queryable.t(), atom()) :: Ecto.Query.t()
|
||||||
defp filter_by_minimum_visibility(query, :private), do: query
|
defp filter_by_minimum_visibility(query, :private), do: query
|
||||||
|
|
||||||
|
@ -38,16 +38,23 @@ defmodule Mobilizon.GraphQL.API.SearchTest do
|
|||||||
|
|
||||||
test "search actors" do
|
test "search actors" do
|
||||||
with_mock Actors,
|
with_mock Actors,
|
||||||
build_actors_by_username_or_name_page: fn "toto", _options, 1, 10 ->
|
search_actors: fn "toto", _options, 1, 10 ->
|
||||||
%Page{total: 1, elements: [%Actor{id: 42}]}
|
%Page{total: 1, elements: [%Actor{id: 42}]}
|
||||||
end do
|
end do
|
||||||
assert {:ok, %{total: 1, elements: [%Actor{id: 42}]}} =
|
assert {:ok, %{total: 1, elements: [%Actor{id: 42}]}} =
|
||||||
Search.search_actors(%{term: "toto"}, 1, 10, :Person)
|
Search.search_actors(%{term: "toto"}, 1, 10, :Person)
|
||||||
|
|
||||||
assert_called(
|
assert_called(
|
||||||
Actors.build_actors_by_username_or_name_page(
|
Actors.search_actors(
|
||||||
"toto",
|
"toto",
|
||||||
[actor_type: [:Person], radius: nil, location: nil, minimum_visibility: :public],
|
[
|
||||||
|
actor_type: :Person,
|
||||||
|
radius: nil,
|
||||||
|
location: nil,
|
||||||
|
minimum_visibility: :public,
|
||||||
|
current_actor_id: nil,
|
||||||
|
exclude_my_groups: false
|
||||||
|
],
|
||||||
1,
|
1,
|
||||||
10
|
10
|
||||||
)
|
)
|
||||||
|
@ -70,7 +70,9 @@ defmodule Mobilizon.Web.Resolvers.FollowerTest do
|
|||||||
variables: %{name: preferred_username}
|
variables: %{name: preferred_username}
|
||||||
)
|
)
|
||||||
|
|
||||||
assert hd(res["errors"])["message"] == "unauthenticated"
|
assert res["errors"] == nil
|
||||||
|
assert res["data"]["group"]["followers"]["total"] == 1
|
||||||
|
assert res["data"]["group"]["followers"]["elements"] == []
|
||||||
end
|
end
|
||||||
|
|
||||||
test "without being a member", %{
|
test "without being a member", %{
|
||||||
@ -88,7 +90,9 @@ defmodule Mobilizon.Web.Resolvers.FollowerTest do
|
|||||||
variables: %{name: preferred_username}
|
variables: %{name: preferred_username}
|
||||||
)
|
)
|
||||||
|
|
||||||
assert hd(res["errors"])["message"] == "unauthorized"
|
assert res["errors"] == nil
|
||||||
|
assert res["data"]["group"]["followers"]["total"] == 1
|
||||||
|
assert res["data"]["group"]["followers"]["elements"] == []
|
||||||
end
|
end
|
||||||
|
|
||||||
test "without being a moderator", %{
|
test "without being a moderator", %{
|
||||||
@ -107,7 +111,9 @@ defmodule Mobilizon.Web.Resolvers.FollowerTest do
|
|||||||
variables: %{name: preferred_username}
|
variables: %{name: preferred_username}
|
||||||
)
|
)
|
||||||
|
|
||||||
assert hd(res["errors"])["message"] == "unauthorized"
|
assert res["errors"] == nil
|
||||||
|
assert res["data"]["group"]["followers"]["total"] == 1
|
||||||
|
assert res["data"]["group"]["followers"]["elements"] == []
|
||||||
end
|
end
|
||||||
|
|
||||||
test "while being a moderator", %{
|
test "while being a moderator", %{
|
||||||
|
@ -183,14 +183,14 @@ defmodule Mobilizon.ActorsTest do
|
|||||||
assert MapSet.new([actor_found_id, actor2_found_id]) == MapSet.new([actor.id, actor2.id])
|
assert MapSet.new([actor_found_id, actor2_found_id]) == MapSet.new([actor.id, actor2.id])
|
||||||
end
|
end
|
||||||
|
|
||||||
test "test build_actors_by_username_or_name_page/4 returns actors with similar usernames",
|
test "test search_actors/4 returns actors with similar usernames",
|
||||||
%{actor: %Actor{id: actor_id}} do
|
%{actor: %Actor{id: actor_id}} do
|
||||||
use_cassette "actors/remote_actor_mastodon_tcit" do
|
use_cassette "actors/remote_actor_mastodon_tcit" do
|
||||||
with {:ok, %Actor{id: actor2_id}} <-
|
with {:ok, %Actor{id: actor2_id}} <-
|
||||||
ActivityPubActor.get_or_fetch_actor_by_url(@remote_account_url) do
|
ActivityPubActor.get_or_fetch_actor_by_url(@remote_account_url) do
|
||||||
%Page{total: 2, elements: actors} =
|
%Page{total: 2, elements: actors} =
|
||||||
Actors.build_actors_by_username_or_name_page("tcit",
|
Actors.search_actors("tcit",
|
||||||
actor_type: [:Person],
|
actor_type: :Person,
|
||||||
minimum_visibility: :private
|
minimum_visibility: :private
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -201,9 +201,8 @@ defmodule Mobilizon.ActorsTest do
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
test "test build_actors_by_username_or_name_page/4 returns actors with similar names" do
|
test "test search_actors/4 returns actors with similar names" do
|
||||||
%{total: 0, elements: actors} =
|
%{total: 0, elements: actors} = Actors.search_actors("ohno", actor_type: :Person)
|
||||||
Actors.build_actors_by_username_or_name_page("ohno", actor_type: [:Person])
|
|
||||||
|
|
||||||
assert actors == []
|
assert actors == []
|
||||||
end
|
end
|
||||||
|
Loading…
Reference in New Issue
Block a user