119 lines
		
	
	
	
		
			4 KiB
		
	
	
	
		
			Elixir
		
	
	
	
	
	
			
		
		
	
	
			119 lines
		
	
	
	
		
			4 KiB
		
	
	
	
		
			Elixir
		
	
	
	
	
	
# Pleroma: A lightweight social networking server
 | 
						|
# Copyright © 2017-2020 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
 | 
						|
  @media_types ["image", "audio", "video"]
 | 
						|
 | 
						|
  @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.uri || user.ap_id], []},
 | 
						|
        {: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 ->
 | 
						|
          # 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 Whatsapp Rich Preview.
 | 
						|
          case Utils.fetch_media_type(@media_types, url["mediaType"]) do
 | 
						|
            "audio" ->
 | 
						|
              [
 | 
						|
                {:meta, [property: "og:audio", content: Utils.attachment_url(url["href"])], []}
 | 
						|
                | acc
 | 
						|
              ]
 | 
						|
 | 
						|
            "image" ->
 | 
						|
              [
 | 
						|
                {:meta, [property: "og:image", 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:video", content: Utils.attachment_url(url["href"])], []}
 | 
						|
                | acc
 | 
						|
              ]
 | 
						|
 | 
						|
            _ ->
 | 
						|
              acc
 | 
						|
          end
 | 
						|
        end)
 | 
						|
 | 
						|
      acc ++ rendered_tags
 | 
						|
    end)
 | 
						|
  end
 | 
						|
 | 
						|
  defp build_attachments(_), do: []
 | 
						|
end
 |