mirror of
https://akkoma.dev/AkkomaGang/akkoma.git
synced 2024-10-30 13:33:12 +00:00
05f8179d08
previously we would uncritically take data and format it into tags for static-fe and the like - however, instances can be configured to disallow unauthenticated access to these resources. this means that OG tags as a vector for information leakage. _technically_ this should only occur if you have both restrict_unauthenticated *AND* you run static-fe, which makes no sense since static-fe is for unauthenticated people in particular, but hey ho.
155 lines
4.8 KiB
Elixir
155 lines
4.8 KiB
Elixir
# Pleroma: A lightweight social networking server
|
|
|
|
# Copyright © 2017-2021 Pleroma Authors <https://pleroma.social/>
|
|
# SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
defmodule Pleroma.Web.Metadata.Providers.TwitterCard do
|
|
alias Pleroma.User
|
|
alias Pleroma.Web.MediaProxy
|
|
alias Pleroma.Web.Metadata
|
|
alias Pleroma.Web.Metadata.Providers.Provider
|
|
alias Pleroma.Web.Metadata.Utils
|
|
|
|
use Pleroma.Web, :verified_routes
|
|
|
|
@behaviour Provider
|
|
@media_types ["image", "audio", "video"]
|
|
|
|
@impl Provider
|
|
def build_tags(%{activity_id: id, object: object, user: user}) do
|
|
attachments =
|
|
if Utils.visible?(object) do
|
|
build_attachments(id, object)
|
|
else
|
|
[]
|
|
end
|
|
|
|
scrubbed_content =
|
|
if Utils.visible?(object) do
|
|
Utils.scrub_html_and_truncate(object)
|
|
else
|
|
"Content cannot be displayed."
|
|
end
|
|
|
|
[
|
|
title_tag(user),
|
|
{:meta, [name: "twitter:description", content: scrubbed_content], []}
|
|
] ++
|
|
if attachments == [] or Metadata.activity_nsfw?(object) do
|
|
[
|
|
image_tag(user),
|
|
{:meta, [name: "twitter:card", content: "summary"], []}
|
|
]
|
|
else
|
|
attachments
|
|
end
|
|
end
|
|
|
|
@impl Provider
|
|
def build_tags(%{user: user}) do
|
|
if Utils.visible?(user) do
|
|
with truncated_bio = Utils.scrub_html_and_truncate(user.bio) do
|
|
[
|
|
title_tag(user),
|
|
{:meta, [name: "twitter:description", content: truncated_bio], []},
|
|
image_tag(user),
|
|
{:meta, [name: "twitter:card", content: "summary"], []}
|
|
]
|
|
end
|
|
else
|
|
[]
|
|
end
|
|
end
|
|
|
|
defp title_tag(user) do
|
|
{:meta, [name: "twitter:title", content: Utils.user_name_string(user)], []}
|
|
end
|
|
|
|
def image_tag(user) do
|
|
if Utils.visible?(user) do
|
|
{:meta, [name: "twitter:image", content: MediaProxy.preview_url(User.avatar_url(user))], []}
|
|
else
|
|
{:meta, [name: "twitter:image", content: ""], []}
|
|
end
|
|
end
|
|
|
|
defp build_attachments(id, %{data: %{"attachment" => attachments}}) do
|
|
Enum.reduce(attachments, [], fn attachment, acc ->
|
|
rendered_tags =
|
|
Enum.reduce(attachment["url"], [], fn url, acc ->
|
|
case Utils.fetch_media_type(@media_types, url["mediaType"]) do
|
|
"audio" ->
|
|
[
|
|
{:meta, [name: "twitter:card", content: "player"], []},
|
|
{:meta, [name: "twitter:player:width", content: "480"], []},
|
|
{:meta, [name: "twitter:player:height", content: "80"], []},
|
|
{:meta, [name: "twitter:player", content: player_url(id)], []}
|
|
| acc
|
|
]
|
|
|
|
# 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
|
|
# 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.
|
|
"image" ->
|
|
[
|
|
{:meta, [name: "twitter:card", content: "summary_large_image"], []},
|
|
{:meta,
|
|
[
|
|
name: "twitter:player",
|
|
content: MediaProxy.url(url["href"])
|
|
], []}
|
|
| acc
|
|
]
|
|
|> maybe_add_dimensions(url)
|
|
|
|
"video" ->
|
|
# fallback to old placeholder values
|
|
height = url["height"] || 480
|
|
width = url["width"] || 480
|
|
|
|
[
|
|
{:meta, [name: "twitter:card", content: "player"], []},
|
|
{:meta, [name: "twitter:player", content: player_url(id)], []},
|
|
{:meta, [name: "twitter:player:width", content: "#{width}"], []},
|
|
{:meta, [name: "twitter:player:height", content: "#{height}"], []},
|
|
{:meta, [name: "twitter:player:stream", content: MediaProxy.url(url["href"])],
|
|
[]},
|
|
{:meta, [name: "twitter:player:stream:content_type", content: url["mediaType"]],
|
|
[]}
|
|
| acc
|
|
]
|
|
|
|
_ ->
|
|
acc
|
|
end
|
|
end)
|
|
|
|
acc ++ rendered_tags
|
|
end)
|
|
end
|
|
|
|
defp build_attachments(_id, _object), do: []
|
|
|
|
defp player_url(id) do
|
|
url(~p[/notice/#{id}/embed_player])
|
|
end
|
|
|
|
# Videos have problems without dimensions, but we used to not provide WxH for images.
|
|
# A default (read: incorrect) fallback for images is likely to cause rendering bugs.
|
|
defp maybe_add_dimensions(metadata, url) do
|
|
cond do
|
|
!is_nil(url["height"]) && !is_nil(url["width"]) ->
|
|
metadata ++
|
|
[
|
|
{:meta, [name: "twitter:player:width", content: "#{url["width"]}"], []},
|
|
{:meta, [name: "twitter:player:height", content: "#{url["height"]}"], []}
|
|
]
|
|
|
|
true ->
|
|
metadata
|
|
end
|
|
end
|
|
end
|