2019-01-16 07:15:13 -07:00
|
|
|
# Pleroma: A lightweight social networking server
|
2021-01-12 23:49:20 -07:00
|
|
|
# Copyright © 2017-2021 Pleroma Authors <https://pleroma.social/>
|
2019-01-16 07:15:13 -07:00
|
|
|
# SPDX-License-Identifier: AGPL-3.0-only
|
2019-01-17 23:32:52 -07:00
|
|
|
|
2019-01-15 09:43:52 -07:00
|
|
|
defmodule Pleroma.Web.Metadata.Providers.OpenGraph do
|
2019-02-09 08:16:26 -07:00
|
|
|
alias Pleroma.User
|
2021-06-09 08:58:29 -06:00
|
|
|
alias Pleroma.Web.MediaProxy
|
2019-02-09 08:16:26 -07:00
|
|
|
alias Pleroma.Web.Metadata
|
2019-02-06 12:20:02 -07:00
|
|
|
alias Pleroma.Web.Metadata.Providers.Provider
|
2019-02-11 16:59:04 -07:00
|
|
|
alias Pleroma.Web.Metadata.Utils
|
2019-01-15 09:43:52 -07:00
|
|
|
|
|
|
|
@behaviour Provider
|
2019-07-12 10:42:54 -06:00
|
|
|
@media_types ["image", "audio", "video"]
|
2019-01-15 09:43:52 -07:00
|
|
|
|
2024-04-11 22:16:47 -06: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 09:43:52 -07:00
|
|
|
@impl Provider
|
2019-01-17 10:18:28 -07:00
|
|
|
def build_tags(%{
|
2019-01-17 23:32:52 -07:00
|
|
|
object: object,
|
2019-01-19 00:58:27 -07:00
|
|
|
url: url,
|
2019-01-17 10:18:28 -07:00
|
|
|
user: user
|
|
|
|
}) do
|
2024-04-11 22:16:47 -06: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 11:20:27 -07:00
|
|
|
|
2019-01-16 08:02:46 -07:00
|
|
|
[
|
|
|
|
{:meta,
|
|
|
|
[
|
|
|
|
property: "og:title",
|
2021-06-03 22:15:54 -06:00
|
|
|
content: Utils.user_name_string(user)
|
2019-01-16 08:02:46 -07:00
|
|
|
], []},
|
2019-01-19 00:58:27 -07:00
|
|
|
{:meta, [property: "og:url", content: url], []},
|
2019-01-16 23:18:46 -07:00
|
|
|
{:meta,
|
|
|
|
[
|
|
|
|
property: "og:description",
|
2021-06-03 22:15:54 -06:00
|
|
|
content: scrubbed_content
|
2019-01-16 23:18:46 -07:00
|
|
|
], []},
|
2021-06-08 16:14:30 -06:00
|
|
|
{:meta, [property: "og:type", content: "article"], []}
|
2019-01-16 08:02:46 -07:00
|
|
|
] ++
|
2019-01-17 23:32:52 -07:00
|
|
|
if attachments == [] or Metadata.activity_nsfw?(object) do
|
2024-04-11 22:16:47 -06:00
|
|
|
user_avatar_tags(user)
|
2019-01-16 08:02:46 -07:00
|
|
|
else
|
|
|
|
attachments
|
|
|
|
end
|
2019-01-15 09:43:52 -07:00
|
|
|
end
|
|
|
|
|
|
|
|
@impl Provider
|
|
|
|
def build_tags(%{user: user}) do
|
2024-04-11 22:16:47 -06:00
|
|
|
if Utils.visible?(user) do
|
|
|
|
truncated_bio = Utils.scrub_html_and_truncate(user.bio)
|
|
|
|
|
2019-01-15 09:43:52 -07:00
|
|
|
[
|
|
|
|
{:meta,
|
|
|
|
[
|
|
|
|
property: "og:title",
|
2019-02-11 16:59:04 -07:00
|
|
|
content: Utils.user_name_string(user)
|
2019-01-15 09:43:52 -07:00
|
|
|
], []},
|
2020-03-29 18:01:09 -06:00
|
|
|
{:meta, [property: "og:url", content: user.uri || user.ap_id], []},
|
2019-01-15 09:43:52 -07:00
|
|
|
{:meta, [property: "og:description", content: truncated_bio], []},
|
2024-04-11 22:16:47 -06:00
|
|
|
{:meta, [property: "og:type", content: "article"], []}
|
|
|
|
] ++ user_avatar_tags(user)
|
|
|
|
else
|
|
|
|
[]
|
2019-01-15 09:43:52 -07:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-01-17 23:32:52 -07:00
|
|
|
defp build_attachments(%{data: %{"attachment" => attachments}}) do
|
2019-01-16 06:52:01 -07:00
|
|
|
Enum.reduce(attachments, [], fn attachment, acc ->
|
2019-01-15 11:17:56 -07:00
|
|
|
rendered_tags =
|
2019-01-16 06:52:01 -07:00
|
|
|
Enum.reduce(attachment["url"], [], fn url, acc ->
|
2021-06-08 15:58:33 -06:00
|
|
|
# TODO: Whatsapp only wants JPEG or PNGs. It seems that if we add a second og:image
|
2019-03-04 21:37:33 -07:00
|
|
|
# object when a Video or GIF is attached it will display that in Whatsapp Rich Preview.
|
2019-07-12 10:42:54 -06:00
|
|
|
case Utils.fetch_media_type(@media_types, url["mediaType"]) do
|
2019-01-16 23:18:46 -07:00
|
|
|
"audio" ->
|
|
|
|
[
|
2021-06-09 08:58:29 -06:00
|
|
|
{:meta, [property: "og:audio", content: MediaProxy.url(url["href"])], []}
|
2019-01-16 23:18:46 -07:00
|
|
|
| acc
|
|
|
|
]
|
|
|
|
|
2021-06-09 10:09:14 -06: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 10:58:51 -06: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-16 23:18:46 -07:00
|
|
|
"image" ->
|
|
|
|
[
|
2021-06-09 08:58:29 -06:00
|
|
|
{:meta, [property: "og:image", content: MediaProxy.url(url["href"])], []},
|
2021-06-03 22:15:54 -06:00
|
|
|
{:meta, [property: "og:image:alt", content: attachment["name"]], []}
|
2019-01-16 23:18:46 -07:00
|
|
|
| acc
|
|
|
|
]
|
2021-06-08 15:58:33 -06:00
|
|
|
|> maybe_add_dimensions(url)
|
2019-01-16 23:18:46 -07:00
|
|
|
|
|
|
|
"video" ->
|
|
|
|
[
|
2021-06-09 08:58:29 -06:00
|
|
|
{:meta, [property: "og:video", content: MediaProxy.url(url["href"])], []}
|
2019-01-16 23:18:46 -07:00
|
|
|
| acc
|
|
|
|
]
|
2021-06-08 15:58:33 -06:00
|
|
|
|> maybe_add_dimensions(url)
|
2021-06-09 09:02:41 -06:00
|
|
|
|> maybe_add_video_thumbnail(url)
|
2019-01-16 23:18:46 -07:00
|
|
|
|
|
|
|
_ ->
|
|
|
|
acc
|
2019-01-15 11:17:56 -07:00
|
|
|
end
|
|
|
|
end)
|
|
|
|
|
|
|
|
acc ++ rendered_tags
|
|
|
|
end)
|
|
|
|
end
|
2019-07-08 03:16:32 -06:00
|
|
|
|
|
|
|
defp build_attachments(_), do: []
|
2021-06-08 15:58:33 -06: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 09:02:41 -06:00
|
|
|
|
2021-06-10 08:56:43 -06: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 10:05:24 -06:00
|
|
|
defp maybe_add_video_thumbnail(metadata, url) do
|
2021-06-09 09:02:41 -06:00
|
|
|
cond do
|
|
|
|
Pleroma.Config.get([:media_preview_proxy, :enabled], false) ->
|
2021-06-09 10:05:24 -06:00
|
|
|
metadata ++
|
2021-06-09 10:06:53 -06: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 09:02:41 -06:00
|
|
|
|
|
|
|
true ->
|
|
|
|
metadata
|
|
|
|
end
|
|
|
|
end
|
2019-01-15 09:43:52 -07:00
|
|
|
end
|