logo

pleroma

My custom branche(s) on git.pleroma.social/pleroma/pleroma git clone https://hacktivis.me/git/pleroma.git

anti_followbot_policy.ex (2405B)


  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. defmodule Pleroma.Web.ActivityPub.MRF.AntiFollowbotPolicy do
  5. alias Pleroma.User
  6. @moduledoc "Prevent followbots from following with a bit of heuristic"
  7. @behaviour Pleroma.Web.ActivityPub.MRF.Policy
  8. # XXX: this should become User.normalize_by_ap_id() or similar, really.
  9. defp normalize_by_ap_id(%{"id" => id}), do: User.get_cached_by_ap_id(id)
  10. defp normalize_by_ap_id(uri) when is_binary(uri), do: User.get_cached_by_ap_id(uri)
  11. defp normalize_by_ap_id(_), do: nil
  12. defp score_nickname("followbot@" <> _), do: 1.0
  13. defp score_nickname("federationbot@" <> _), do: 1.0
  14. defp score_nickname("federation_bot@" <> _), do: 1.0
  15. defp score_nickname(_), do: 0.0
  16. defp score_displayname("federation bot"), do: 1.0
  17. defp score_displayname("federationbot"), do: 1.0
  18. defp score_displayname("fedibot"), do: 1.0
  19. defp score_displayname(_), do: 0.0
  20. defp determine_if_followbot(%User{nickname: nickname, name: displayname, actor_type: actor_type}) do
  21. # nickname will be a binary string except when following a relay
  22. nick_score =
  23. if is_binary(nickname) do
  24. nickname
  25. |> String.downcase()
  26. |> score_nickname()
  27. else
  28. 0.0
  29. end
  30. # displayname will either be a binary string or nil, if a displayname isn't set.
  31. name_score =
  32. if is_binary(displayname) do
  33. displayname
  34. |> String.downcase()
  35. |> score_displayname()
  36. else
  37. 0.0
  38. end
  39. # actor_type "Service" is a Bot account
  40. actor_type_score =
  41. if actor_type == "Service" do
  42. 1.0
  43. else
  44. 0.0
  45. end
  46. nick_score + name_score + actor_type_score
  47. end
  48. defp bot_allowed?(%{"object" => target}, bot_actor) do
  49. %User{} = user = normalize_by_ap_id(target)
  50. User.following?(user, bot_actor)
  51. end
  52. @impl true
  53. def filter(%{"type" => "Follow", "actor" => actor_id} = message) do
  54. %User{} = actor = normalize_by_ap_id(actor_id)
  55. score = determine_if_followbot(actor)
  56. if score < 0.8 || bot_allowed?(message, actor) do
  57. {:ok, message}
  58. else
  59. {:reject, "[AntiFollowbotPolicy] Scored #{actor_id} as #{score}"}
  60. end
  61. end
  62. @impl true
  63. def filter(message), do: {:ok, message}
  64. @impl true
  65. def describe, do: {:ok, %{}}
  66. end