2019-01-16 17:15:13 +03:00
|
|
|
# Pleroma: A lightweight social networking server
|
2021-01-13 07:49:20 +01:00
|
|
|
# Copyright © 2017-2021 Pleroma Authors <https://pleroma.social/>
|
2019-01-16 17:15:13 +03:00
|
|
|
# SPDX-License-Identifier: AGPL-3.0-only
|
2019-01-18 09:32:52 +03:00
|
|
|
|
2019-01-15 19:43:52 +03:00
|
|
|
defmodule Pleroma.Web.Metadata.Providers.OpenGraph do
|
2019-02-09 16:16:26 +01:00
|
|
|
alias Pleroma.User
|
2021-06-09 09:58:29 -05:00
|
|
|
alias Pleroma.Web.MediaProxy
|
2019-02-09 16:16:26 +01:00
|
|
|
alias Pleroma.Web.Metadata
|
2019-02-06 20:20:02 +01:00
|
|
|
alias Pleroma.Web.Metadata.Providers.Provider
|
2019-02-11 23:59:04 +00:00
|
|
|
alias Pleroma.Web.Metadata.Utils
|
2019-01-15 19:43:52 +03:00
|
|
|
|
|
|
|
@behaviour Provider
|
2019-07-12 16:42:54 +00:00
|
|
|
@media_types ["image", "audio", "video"]
|
2019-01-15 19:43:52 +03:00
|
|
|
|
2024-04-12 05:16:47 +01:00
|
|
|
defp user_avatar_tags(user) do
|
|
|
|
if Utils.visible?(user) do
|
|
|
|
[
|
|
|
|
{:meta, [property: "og:image", content: MediaProxy.preview_url(User.avatar_url(user))],
|
|
|
|
[]},
|
|
|
|
{:meta, [property: "og:image:width", content: 150], []},
|
|
|
|
{:meta, [property: "og:image:height", content: 150], []}
|
|
|
|
]
|
|
|
|
else
|
|
|
|
[]
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-01-15 19:43:52 +03:00
|
|
|
@impl Provider
|
2019-01-17 20:18:28 +03:00
|
|
|
def build_tags(%{
|
2019-01-18 09:32:52 +03:00
|
|
|
object: object,
|
2019-01-19 10:58:27 +03:00
|
|
|
url: url,
|
2019-01-17 20:18:28 +03:00
|
|
|
user: user
|
|
|
|
}) do
|
2024-04-12 05:16:47 +01:00
|
|
|
attachments =
|
|
|
|
if Utils.visible?(object) do
|
|
|
|
build_attachments(object)
|
|
|
|
else
|
|
|
|
[]
|
|
|
|
end
|
|
|
|
|
|
|
|
scrubbed_content =
|
|
|
|
if Utils.visible?(object) do
|
|
|
|
Utils.scrub_html_and_truncate(object)
|
|
|
|
else
|
|
|
|
"Content cannot be displayed."
|
|
|
|
end
|
2019-01-15 21:20:27 +03:00
|
|
|
|
2019-01-16 18:02:46 +03:00
|
|
|
[
|
|
|
|
{:meta,
|
|
|
|
[
|
|
|
|
property: "og:title",
|
2021-06-04 04:15:54 +00:00
|
|
|
content: Utils.user_name_string(user)
|
2019-01-16 18:02:46 +03:00
|
|
|
], []},
|
2019-01-19 10:58:27 +03:00
|
|
|
{:meta, [property: "og:url", content: url], []},
|
2019-01-17 09:18:46 +03:00
|
|
|
{:meta,
|
|
|
|
[
|
|
|
|
property: "og:description",
|
2021-06-04 04:15:54 +00:00
|
|
|
content: scrubbed_content
|
2019-01-17 09:18:46 +03:00
|
|
|
], []},
|
2021-06-08 17:14:30 -05:00
|
|
|
{:meta, [property: "og:type", content: "article"], []}
|
2019-01-16 18:02:46 +03:00
|
|
|
] ++
|
2019-01-18 09:32:52 +03:00
|
|
|
if attachments == [] or Metadata.activity_nsfw?(object) do
|
2024-04-12 05:16:47 +01:00
|
|
|
user_avatar_tags(user)
|
2019-01-16 18:02:46 +03:00
|
|
|
else
|
|
|
|
attachments
|
|
|
|
end
|
2019-01-15 19:43:52 +03:00
|
|
|
end
|
|
|
|
|
|
|
|
@impl Provider
|
|
|
|
def build_tags(%{user: user}) do
|
2024-04-12 05:16:47 +01:00
|
|
|
if Utils.visible?(user) do
|
|
|
|
truncated_bio = Utils.scrub_html_and_truncate(user.bio)
|
|
|
|
|
2019-01-15 19:43:52 +03:00
|
|
|
[
|
|
|
|
{:meta,
|
|
|
|
[
|
|
|
|
property: "og:title",
|
2019-02-11 23:59:04 +00:00
|
|
|
content: Utils.user_name_string(user)
|
2019-01-15 19:43:52 +03:00
|
|
|
], []},
|
2020-03-30 02:01:09 +02:00
|
|
|
{:meta, [property: "og:url", content: user.uri || user.ap_id], []},
|
2019-01-15 19:43:52 +03:00
|
|
|
{:meta, [property: "og:description", content: truncated_bio], []},
|
2024-04-12 05:16:47 +01:00
|
|
|
{:meta, [property: "og:type", content: "article"], []}
|
|
|
|
] ++ user_avatar_tags(user)
|
|
|
|
else
|
|
|
|
[]
|
2019-01-15 19:43:52 +03:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-01-18 09:32:52 +03:00
|
|
|
defp build_attachments(%{data: %{"attachment" => attachments}}) do
|
2019-01-16 16:52:01 +03:00
|
|
|
Enum.reduce(attachments, [], fn attachment, acc ->
|
2019-01-15 21:17:56 +03:00
|
|
|
rendered_tags =
|
2019-01-16 16:52:01 +03:00
|
|
|
Enum.reduce(attachment["url"], [], fn url, acc ->
|
2021-06-08 16:58:33 -05:00
|
|
|
# TODO: Whatsapp only wants JPEG or PNGs. It seems that if we add a second og:image
|
2019-03-05 05:37:33 +01:00
|
|
|
# object when a Video or GIF is attached it will display that in Whatsapp Rich Preview.
|
2019-07-12 16:42:54 +00:00
|
|
|
case Utils.fetch_media_type(@media_types, url["mediaType"]) do
|
2019-01-17 09:18:46 +03:00
|
|
|
"audio" ->
|
|
|
|
[
|
2021-06-09 09:58:29 -05:00
|
|
|
{:meta, [property: "og:audio", content: MediaProxy.url(url["href"])], []}
|
2019-01-17 09:18:46 +03:00
|
|
|
| acc
|
|
|
|
]
|
|
|
|
|
2021-06-09 11:09:14 -05:00
|
|
|
# Not using preview_url for this. It saves bandwidth, but the image dimensions will
|
|
|
|
# be wrong. We generate it on the fly and have no way to capture or analyze the
|
2021-06-09 11:58:51 -05:00
|
|
|
# image to get the dimensions. This can be an issue for apps/FEs rendering images
|
|
|
|
# in timelines too, but you can get clever with the aspect ratio metadata as a
|
|
|
|
# workaround.
|
2019-01-17 09:18:46 +03:00
|
|
|
"image" ->
|
|
|
|
[
|
2021-06-09 09:58:29 -05:00
|
|
|
{:meta, [property: "og:image", content: MediaProxy.url(url["href"])], []},
|
2021-06-04 04:15:54 +00:00
|
|
|
{:meta, [property: "og:image:alt", content: attachment["name"]], []}
|
2019-01-17 09:18:46 +03:00
|
|
|
| acc
|
|
|
|
]
|
2021-06-08 16:58:33 -05:00
|
|
|
|> maybe_add_dimensions(url)
|
2019-01-17 09:18:46 +03:00
|
|
|
|
|
|
|
"video" ->
|
|
|
|
[
|
2021-06-09 09:58:29 -05:00
|
|
|
{:meta, [property: "og:video", content: MediaProxy.url(url["href"])], []}
|
2019-01-17 09:18:46 +03:00
|
|
|
| acc
|
|
|
|
]
|
2021-06-08 16:58:33 -05:00
|
|
|
|> maybe_add_dimensions(url)
|
2021-06-09 10:02:41 -05:00
|
|
|
|> maybe_add_video_thumbnail(url)
|
2019-01-17 09:18:46 +03:00
|
|
|
|
|
|
|
_ ->
|
|
|
|
acc
|
2019-01-15 21:17:56 +03:00
|
|
|
end
|
|
|
|
end)
|
|
|
|
|
|
|
|
acc ++ rendered_tags
|
|
|
|
end)
|
|
|
|
end
|
2019-07-08 12:16:32 +03:00
|
|
|
|
|
|
|
defp build_attachments(_), do: []
|
2021-06-08 16:58:33 -05:00
|
|
|
|
|
|
|
# We can use url["mediaType"] to dynamically fill the metadata
|
|
|
|
defp maybe_add_dimensions(metadata, url) do
|
|
|
|
type = url["mediaType"] |> String.split("/") |> List.first()
|
|
|
|
|
|
|
|
cond do
|
|
|
|
!is_nil(url["height"]) && !is_nil(url["width"]) ->
|
|
|
|
metadata ++
|
|
|
|
[
|
|
|
|
{:meta, [property: "og:#{type}:width", content: "#{url["width"]}"], []},
|
|
|
|
{:meta, [property: "og:#{type}:height", content: "#{url["height"]}"], []}
|
|
|
|
]
|
|
|
|
|
|
|
|
true ->
|
|
|
|
metadata
|
|
|
|
end
|
|
|
|
end
|
2021-06-09 10:02:41 -05:00
|
|
|
|
2021-06-10 09:56:43 -05:00
|
|
|
# Media Preview Proxy makes thumbnails of videos without resizing, so we can trust the
|
|
|
|
# width and height of the source video.
|
2021-06-09 11:05:24 -05:00
|
|
|
defp maybe_add_video_thumbnail(metadata, url) do
|
2021-06-09 10:02:41 -05:00
|
|
|
cond do
|
|
|
|
Pleroma.Config.get([:media_preview_proxy, :enabled], false) ->
|
2021-06-09 11:05:24 -05:00
|
|
|
metadata ++
|
2021-06-09 11:06:53 -05:00
|
|
|
[
|
|
|
|
{:meta, [property: "og:image:width", content: "#{url["width"]}"], []},
|
|
|
|
{:meta, [property: "og:image:height", content: "#{url["height"]}"], []},
|
|
|
|
{:meta, [property: "og:image", content: MediaProxy.preview_url(url["href"])], []}
|
|
|
|
]
|
2021-06-09 10:02:41 -05:00
|
|
|
|
|
|
|
true ->
|
|
|
|
metadata
|
|
|
|
end
|
|
|
|
end
|
2019-01-15 19:43:52 +03:00
|
|
|
end
|