ac7ef0999d
Twitter cards were not passing any useful metadata. A few things were being handled on Twitter's end by trying to match OpenGraph tags with their own, but it wasn't working at all for media. This is an attempt to fix that. Common functions have been pulled out of opengraph and put into utils. Twitter's functionality was entirely replaced with a direct copy of Opengraph's and then modified as needed. Profiles are now represented as Summary Cards Posts with images are now represented as Summart with Large Image Cards Posts with video and audio attachments are represented as Player Cards. This now passes the Twitter Card Validator. Validator and Docs are below https://cards-dev.twitter.com/validator https://developer.twitter.com/en/docs/tweets/optimize-with-cards/overview/abouts-cards
124 lines
4.1 KiB
Elixir
124 lines
4.1 KiB
Elixir
# Pleroma: A lightweight social networking server
|
|
# Copyright © 2017-2019 Pleroma Authors <https://pleroma.social/>
|
|
# SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
defmodule Pleroma.Web.Metadata.Providers.OpenGraph do
|
|
alias Pleroma.User
|
|
alias Pleroma.Web.Metadata
|
|
alias Pleroma.Web.Metadata.Providers.Provider
|
|
alias Pleroma.Web.Metadata.Utils
|
|
|
|
@behaviour Provider
|
|
|
|
@impl Provider
|
|
def build_tags(%{
|
|
object: object,
|
|
url: url,
|
|
user: user
|
|
}) do
|
|
attachments = build_attachments(object)
|
|
scrubbed_content = Utils.scrub_html_and_truncate(object)
|
|
# Zero width space
|
|
content =
|
|
if scrubbed_content != "" and scrubbed_content != "\u200B" do
|
|
": “" <> scrubbed_content <> "”"
|
|
else
|
|
""
|
|
end
|
|
|
|
# Most previews only show og:title which is inconvenient. Instagram
|
|
# hacks this by putting the description in the title and making the
|
|
# description longer prefixed by how many likes and shares the post
|
|
# has. Here we use the descriptive nickname in the title, and expand
|
|
# the full account & nickname in the description. We also use the cute^Wevil
|
|
# smart quotes around the status text like Instagram, too.
|
|
[
|
|
{:meta,
|
|
[
|
|
property: "og:title",
|
|
content: "#{user.name}" <> content
|
|
], []},
|
|
{:meta, [property: "og:url", content: url], []},
|
|
{:meta,
|
|
[
|
|
property: "og:description",
|
|
content: "#{Utils.user_name_string(user)}" <> content
|
|
], []},
|
|
{:meta, [property: "og:type", content: "website"], []}
|
|
] ++
|
|
if attachments == [] or Metadata.activity_nsfw?(object) do
|
|
[
|
|
{:meta, [property: "og:image", content: Utils.attachment_url(User.avatar_url(user))],
|
|
[]},
|
|
{:meta, [property: "og:image:width", content: 150], []},
|
|
{:meta, [property: "og:image:height", content: 150], []}
|
|
]
|
|
else
|
|
attachments
|
|
end
|
|
end
|
|
|
|
@impl Provider
|
|
def build_tags(%{user: user}) do
|
|
with truncated_bio = Utils.scrub_html_and_truncate(user.bio || "") do
|
|
[
|
|
{:meta,
|
|
[
|
|
property: "og:title",
|
|
content: Utils.user_name_string(user)
|
|
], []},
|
|
{:meta, [property: "og:url", content: User.profile_url(user)], []},
|
|
{:meta, [property: "og:description", content: truncated_bio], []},
|
|
{:meta, [property: "og:type", content: "website"], []},
|
|
{:meta, [property: "og:image", content: Utils.attachment_url(User.avatar_url(user))], []},
|
|
{:meta, [property: "og:image:width", content: 150], []},
|
|
{:meta, [property: "og:image:height", content: 150], []}
|
|
]
|
|
end
|
|
end
|
|
|
|
defp build_attachments(%{data: %{"attachment" => attachments}}) do
|
|
Enum.reduce(attachments, [], fn attachment, acc ->
|
|
rendered_tags =
|
|
Enum.reduce(attachment["url"], [], fn url, acc ->
|
|
media_type =
|
|
Enum.find(["image", "audio", "video"], fn media_type ->
|
|
String.starts_with?(url["mediaType"], media_type)
|
|
end)
|
|
|
|
# TODO: Add additional properties to objects when we have the data available.
|
|
# Also, Whatsapp only wants JPEG or PNGs. It seems that if we add a second og:image
|
|
# object when a Video or GIF is attached it will display that in the Whatsapp Rich Preview.
|
|
case media_type do
|
|
"audio" ->
|
|
[
|
|
{:meta,
|
|
[property: "og:" <> media_type, content: Utils.attachment_url(url["href"])], []}
|
|
| acc
|
|
]
|
|
|
|
"image" ->
|
|
[
|
|
{:meta, [property: "og:" <> media_type, content: Utils.attachment_url(url["href"])],
|
|
[]},
|
|
{:meta, [property: "og:image:width", content: 150], []},
|
|
{:meta, [property: "og:image:height", content: 150], []}
|
|
| acc
|
|
]
|
|
|
|
"video" ->
|
|
[
|
|
{:meta,
|
|
[property: "og:" <> media_type, content: Utils.attachment_url(url["href"])], []}
|
|
| acc
|
|
]
|
|
|
|
_ ->
|
|
acc
|
|
end
|
|
end)
|
|
|
|
acc ++ rendered_tags
|
|
end)
|
|
end
|
|
end
|