First
[anni] / lib / pleroma / web / activity_pub / mrf / anti_followbot_policy.ex
1 # Pleroma: A lightweight social networking server
2 # Copyright © 2017-2022 Pleroma Authors <https://pleroma.social/>
3 # SPDX-License-Identifier: AGPL-3.0-only
4
5 defmodule Pleroma.Web.ActivityPub.MRF.AntiFollowbotPolicy do
6   alias Pleroma.User
7
8   @moduledoc "Prevent followbots from following with a bit of heuristic"
9
10   @behaviour Pleroma.Web.ActivityPub.MRF.Policy
11
12   # XXX: this should become User.normalize_by_ap_id() or similar, really.
13   defp normalize_by_ap_id(%{"id" => id}), do: User.get_cached_by_ap_id(id)
14   defp normalize_by_ap_id(uri) when is_binary(uri), do: User.get_cached_by_ap_id(uri)
15   defp normalize_by_ap_id(_), do: nil
16
17   defp score_nickname("followbot@" <> _), do: 1.0
18   defp score_nickname("federationbot@" <> _), do: 1.0
19   defp score_nickname("federation_bot@" <> _), do: 1.0
20   defp score_nickname(_), do: 0.0
21
22   defp score_displayname("federation bot"), do: 1.0
23   defp score_displayname("federationbot"), do: 1.0
24   defp score_displayname("fedibot"), do: 1.0
25   defp score_displayname(_), do: 0.0
26
27   defp determine_if_followbot(%User{nickname: nickname, name: displayname, actor_type: actor_type}) do
28     # nickname will be a binary string except when following a relay
29     nick_score =
30       if is_binary(nickname) do
31         nickname
32         |> String.downcase()
33         |> score_nickname()
34       else
35         0.0
36       end
37
38     # displayname will either be a binary string or nil, if a displayname isn't set.
39     name_score =
40       if is_binary(displayname) do
41         displayname
42         |> String.downcase()
43         |> score_displayname()
44       else
45         0.0
46       end
47
48     # actor_type "Service" is a Bot account
49     actor_type_score =
50       if actor_type == "Service" do
51         1.0
52       else
53         0.0
54       end
55
56     nick_score + name_score + actor_type_score
57   end
58
59   defp determine_if_followbot(_), do: 0.0
60
61   defp bot_allowed?(%{"object" => target}, bot_actor) do
62     %User{} = user = normalize_by_ap_id(target)
63
64     User.following?(user, bot_actor)
65   end
66
67   @impl true
68   def filter(%{"type" => "Follow", "actor" => actor_id} = message) do
69     %User{} = actor = normalize_by_ap_id(actor_id)
70
71     score = determine_if_followbot(actor)
72
73     if score < 0.8 || bot_allowed?(message, actor) do
74       {:ok, message}
75     else
76       {:reject, "[AntiFollowbotPolicy] Scored #{actor_id} as #{score}"}
77     end
78   end
79
80   @impl true
81   def filter(message), do: {:ok, message}
82
83   @impl true
84   def describe, do: {:ok, %{}}
85 end