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