2020-04-14 16:59:04 +00:00
|
|
|
|
# Pleroma: A lightweight social networking server
|
2021-01-13 06:49:20 +00:00
|
|
|
|
# Copyright © 2017-2021 Pleroma Authors <https://pleroma.social/>
|
2020-04-14 16:59:04 +00:00
|
|
|
|
# SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
|
|
|
|
|
defmodule Pleroma.Web.ActivityPub.MRF.StealEmojiPolicy do
|
|
|
|
|
require Logger
|
|
|
|
|
|
|
|
|
|
alias Pleroma.Config
|
|
|
|
|
|
|
|
|
|
@moduledoc "Detect new emojis by their shortcode and steals them"
|
2021-06-07 19:22:08 +00:00
|
|
|
|
@behaviour Pleroma.Web.ActivityPub.MRF.Policy
|
2020-04-14 16:59:04 +00:00
|
|
|
|
|
|
|
|
|
defp accept_host?(host), do: host in Config.get([:mrf_steal_emoji, :hosts], [])
|
|
|
|
|
|
2022-05-18 19:25:10 +00:00
|
|
|
|
defp shortcode_matches?(shortcode, pattern) when is_binary(pattern) do
|
|
|
|
|
shortcode == pattern
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
defp shortcode_matches?(shortcode, pattern) do
|
|
|
|
|
String.match?(shortcode, pattern)
|
|
|
|
|
end
|
|
|
|
|
|
2024-02-20 20:11:26 +00:00
|
|
|
|
defp reject_emoji?({shortcode, _url}, installed_emoji) do
|
|
|
|
|
valid_shortcode? = String.match?(shortcode, ~r/^[a-zA-Z0-9_-]+$/)
|
|
|
|
|
|
|
|
|
|
rejected_shortcode? =
|
|
|
|
|
[:mrf_steal_emoji, :rejected_shortcodes]
|
|
|
|
|
|> Config.get([])
|
|
|
|
|
|> Enum.any?(fn pattern -> shortcode_matches?(shortcode, pattern) end)
|
|
|
|
|
|
|
|
|
|
emoji_installed? = Enum.member?(installed_emoji, shortcode)
|
|
|
|
|
|
|
|
|
|
!valid_shortcode? or rejected_shortcode? or emoji_installed?
|
|
|
|
|
end
|
|
|
|
|
|
2024-03-07 12:07:02 +00:00
|
|
|
|
defp steal_emoji(%{} = response, {shortcode, url}, emoji_dir_path) do
|
|
|
|
|
extension =
|
|
|
|
|
url
|
|
|
|
|
|> URI.parse()
|
|
|
|
|
|> Map.get(:path)
|
|
|
|
|
|> Path.basename()
|
|
|
|
|
|> Path.extname()
|
|
|
|
|
|
|
|
|
|
shortcode = Path.basename(shortcode)
|
|
|
|
|
file_path = Path.join(emoji_dir_path, shortcode <> (extension || ".png"))
|
|
|
|
|
|
|
|
|
|
case File.write(file_path, response.body) do
|
|
|
|
|
:ok ->
|
|
|
|
|
shortcode
|
|
|
|
|
|
|
|
|
|
e ->
|
|
|
|
|
Logger.warning("MRF.StealEmojiPolicy: Failed to write to #{file_path}: #{inspect(e)}")
|
|
|
|
|
nil
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
defp maybe_steal_emoji({shortcode, url}, emoji_dir_path) do
|
2020-04-14 16:59:04 +00:00
|
|
|
|
url = Pleroma.Web.MediaProxy.url(url)
|
|
|
|
|
|
2020-12-25 08:30:36 +00:00
|
|
|
|
with {:ok, %{status: status} = response} when status in 200..299 <- Pleroma.HTTP.get(url) do
|
|
|
|
|
size_limit = Config.get([:mrf_steal_emoji, :size_limit], 50_000)
|
|
|
|
|
|
|
|
|
|
if byte_size(response.body) <= size_limit do
|
2024-03-07 12:07:02 +00:00
|
|
|
|
steal_emoji(response, {shortcode, url}, emoji_dir_path)
|
2020-12-24 17:27:28 +00:00
|
|
|
|
else
|
2020-12-25 08:30:36 +00:00
|
|
|
|
Logger.debug(
|
2021-10-06 06:08:21 +00:00
|
|
|
|
"MRF.StealEmojiPolicy: :#{shortcode}: at #{url} (#{byte_size(response.body)} B) over size limit (#{size_limit} B)"
|
2020-12-25 08:30:36 +00:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
nil
|
2020-04-14 16:59:04 +00:00
|
|
|
|
end
|
|
|
|
|
else
|
2020-12-25 08:30:36 +00:00
|
|
|
|
e ->
|
2023-08-01 10:43:50 +00:00
|
|
|
|
Logger.warning("MRF.StealEmojiPolicy: Failed to fetch #{url}: #{inspect(e)}")
|
2020-12-25 08:30:36 +00:00
|
|
|
|
nil
|
2020-04-14 16:59:04 +00:00
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
@impl true
|
|
|
|
|
def filter(%{"object" => %{"emoji" => foreign_emojis, "actor" => actor}} = message) do
|
|
|
|
|
host = URI.parse(actor).host
|
|
|
|
|
|
2020-12-25 08:30:36 +00:00
|
|
|
|
if host != Pleroma.Web.Endpoint.host() and accept_host?(host) do
|
2020-04-14 16:59:04 +00:00
|
|
|
|
installed_emoji = Pleroma.Emoji.get_all() |> Enum.map(fn {k, _} -> k end)
|
|
|
|
|
|
2020-12-24 17:27:28 +00:00
|
|
|
|
emoji_dir_path =
|
|
|
|
|
Config.get(
|
|
|
|
|
[:mrf_steal_emoji, :path],
|
|
|
|
|
Path.join(Config.get([:instance, :static_dir]), "emoji/stolen")
|
|
|
|
|
)
|
|
|
|
|
|
2020-12-27 18:58:15 +00:00
|
|
|
|
File.mkdir_p(emoji_dir_path)
|
2020-12-24 17:27:28 +00:00
|
|
|
|
|
2020-04-14 16:59:04 +00:00
|
|
|
|
new_emojis =
|
|
|
|
|
foreign_emojis
|
2024-02-20 20:11:26 +00:00
|
|
|
|
|> Enum.reject(&reject_emoji?(&1, installed_emoji))
|
2024-03-07 12:07:02 +00:00
|
|
|
|
|> Enum.map(&maybe_steal_emoji(&1, emoji_dir_path))
|
2020-04-14 16:59:04 +00:00
|
|
|
|
|> Enum.filter(& &1)
|
|
|
|
|
|
|
|
|
|
if !Enum.empty?(new_emojis) do
|
|
|
|
|
Logger.info("Stole new emojis: #{inspect(new_emojis)}")
|
|
|
|
|
Pleroma.Emoji.reload()
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
{:ok, message}
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
def filter(message), do: {:ok, message}
|
|
|
|
|
|
2021-08-14 16:08:39 +00:00
|
|
|
|
@impl true
|
|
|
|
|
@spec config_description :: %{
|
|
|
|
|
children: [
|
|
|
|
|
%{
|
|
|
|
|
description: <<_::272, _::_*256>>,
|
|
|
|
|
key: :hosts | :rejected_shortcodes | :size_limit,
|
|
|
|
|
suggestions: [any(), ...],
|
|
|
|
|
type: {:list, :string} | {:list, :string} | :integer
|
|
|
|
|
},
|
|
|
|
|
...
|
|
|
|
|
],
|
|
|
|
|
description: <<_::448>>,
|
|
|
|
|
key: :mrf_steal_emoji,
|
|
|
|
|
label: <<_::80>>,
|
|
|
|
|
related_policy: <<_::352>>
|
|
|
|
|
}
|
|
|
|
|
def config_description do
|
|
|
|
|
%{
|
|
|
|
|
key: :mrf_steal_emoji,
|
|
|
|
|
related_policy: "Pleroma.Web.ActivityPub.MRF.StealEmojiPolicy",
|
|
|
|
|
label: "MRF Emojis",
|
|
|
|
|
description: "Steals emojis from selected instances when it sees them.",
|
|
|
|
|
children: [
|
|
|
|
|
%{
|
|
|
|
|
key: :hosts,
|
|
|
|
|
type: {:list, :string},
|
|
|
|
|
description: "List of hosts to steal emojis from",
|
|
|
|
|
suggestions: [""]
|
|
|
|
|
},
|
|
|
|
|
%{
|
|
|
|
|
key: :rejected_shortcodes,
|
|
|
|
|
type: {:list, :string},
|
2022-05-18 19:25:10 +00:00
|
|
|
|
description: """
|
|
|
|
|
A list of patterns or matches to reject shortcodes with.
|
|
|
|
|
|
|
|
|
|
Each pattern can be a string or [Regex](https://hexdocs.pm/elixir/Regex.html) in the format of `~r/PATTERN/`.
|
|
|
|
|
""",
|
|
|
|
|
suggestions: ["foo", ~r/foo/]
|
2021-08-14 16:08:39 +00:00
|
|
|
|
},
|
|
|
|
|
%{
|
|
|
|
|
key: :size_limit,
|
|
|
|
|
type: :integer,
|
|
|
|
|
description: "File size limit (in bytes), checked before an emoji is saved to the disk",
|
|
|
|
|
suggestions: ["100000"]
|
|
|
|
|
}
|
|
|
|
|
]
|
|
|
|
|
}
|
|
|
|
|
end
|
|
|
|
|
|
2020-04-14 16:59:04 +00:00
|
|
|
|
@impl true
|
|
|
|
|
def describe do
|
|
|
|
|
{:ok, %{}}
|
|
|
|
|
end
|
|
|
|
|
end
|