pleroma/lib/pleroma/web/ostatus/handlers/note_handler.ex

156 lines
4.9 KiB
Elixir
Raw Normal View History

2017-05-20 13:35:22 +02:00
defmodule Pleroma.Web.OStatus.NoteHandler do
require Logger
alias Pleroma.Web.{XML, OStatus}
2018-05-04 23:25:31 +02:00
alias Pleroma.{Object, Activity}
2017-05-20 13:35:22 +02:00
alias Pleroma.Web.ActivityPub.ActivityPub
alias Pleroma.Web.ActivityPub.Utils
alias Pleroma.Web.CommonAPI
2017-05-20 13:35:22 +02:00
@doc """
Get the context for this note. Uses this:
1. The context of the parent activity
2. The conversation reference in the ostatus xml
3. A newly generated context id.
"""
def get_context(entry, inReplyTo) do
2018-03-30 15:01:53 +02:00
context =
(XML.string_from_xpath("//ostatus:conversation[1]", entry) ||
XML.string_from_xpath("//ostatus:conversation[1]/@ref", entry) || "")
|> String.trim()
2017-05-20 13:35:22 +02:00
with %{data: %{"context" => context}} <- Object.get_cached_by_ap_id(inReplyTo) do
context
2018-03-30 15:01:53 +02:00
else
_e ->
if String.length(context) > 0 do
context
else
Utils.generate_context_id()
end
2017-05-20 13:35:22 +02:00
end
end
def get_people_mentions(entry) do
2018-03-30 15:01:53 +02:00
:xmerl_xpath.string(
'//link[@rel="mentioned" and @ostatus:object-type="http://activitystrea.ms/schema/1.0/person"]',
entry
)
|> Enum.map(fn person -> XML.string_from_xpath("@href", person) end)
2017-05-20 13:35:22 +02:00
end
def get_collection_mentions(entry) do
transmogrify = fn
2018-03-30 15:01:53 +02:00
"http://activityschema.org/collection/public" ->
"https://www.w3.org/ns/activitystreams#Public"
2018-03-30 15:01:53 +02:00
group ->
group
end
2018-03-30 15:01:53 +02:00
:xmerl_xpath.string(
'//link[@rel="mentioned" and @ostatus:object-type="http://activitystrea.ms/schema/1.0/collection"]',
entry
)
|> Enum.map(fn collection -> XML.string_from_xpath("@href", collection) |> transmogrify.() end)
end
def get_mentions(entry) do
2018-03-30 15:01:53 +02:00
(get_people_mentions(entry) ++ get_collection_mentions(entry))
|> Enum.filter(& &1)
end
2017-09-16 16:27:48 +02:00
def get_emoji(entry) do
try do
:xmerl_xpath.string('//link[@rel="emoji"]', entry)
2018-03-30 15:01:53 +02:00
|> Enum.reduce(%{}, fn emoji, acc ->
2017-09-16 16:27:48 +02:00
Map.put(acc, XML.string_from_xpath("@name", emoji), XML.string_from_xpath("@href", emoji))
end)
rescue
_e -> nil
end
end
2017-05-20 13:35:22 +02:00
def make_to_list(actor, mentions) do
[
2017-07-19 19:06:49 +02:00
actor.follower_address
2017-05-20 13:35:22 +02:00
] ++ mentions
end
2017-06-25 11:57:34 +02:00
def add_external_url(note, entry) do
url = XML.string_from_xpath("//link[@rel='alternate' and @type='text/html']/@href", entry)
Map.put(note, "external_url", url)
end
2017-09-05 14:01:37 +02:00
def fetch_replied_to_activity(entry, inReplyTo) do
with %Activity{} = activity <- Activity.get_create_activity_by_object_ap_id(inReplyTo) do
activity
else
_e ->
2018-03-30 15:01:53 +02:00
with inReplyToHref when not is_nil(inReplyToHref) <-
XML.string_from_xpath("//thr:in-reply-to[1]/@href", entry),
2017-09-05 14:01:37 +02:00
{:ok, [activity | _]} <- OStatus.fetch_activity_from_url(inReplyToHref) do
activity
else
_e -> nil
end
end
end
# TODO: Clean this up a bit.
2017-05-20 13:35:22 +02:00
def handle_note(entry, doc \\ nil) do
with id <- XML.string_from_xpath("//id", entry),
activity when is_nil(activity) <- Activity.get_create_activity_by_object_ap_id(id),
[author] <- :xmerl_xpath.string('//author[1]', doc),
{:ok, actor} <- OStatus.find_make_or_update_user(author),
content_html <- OStatus.get_content(entry),
2017-10-31 17:30:46 +01:00
cw <- OStatus.get_cw(entry),
2017-05-20 13:35:22 +02:00
inReplyTo <- XML.string_from_xpath("//thr:in-reply-to[1]/@ref", entry),
2017-09-05 14:01:37 +02:00
inReplyToActivity <- fetch_replied_to_activity(entry, inReplyTo),
inReplyTo <- (inReplyToActivity && inReplyToActivity.data["object"]["id"]) || inReplyTo,
2017-05-20 13:35:22 +02:00
attachments <- OStatus.get_attachments(entry),
context <- get_context(entry, inReplyTo),
tags <- OStatus.get_tags(entry),
mentions <- get_mentions(entry),
to <- make_to_list(actor, mentions),
date <- XML.string_from_xpath("//published", entry),
unlisted <- XML.string_from_xpath("//mastodon:scope", entry) == "unlisted",
cc <- if(unlisted, do: ["https://www.w3.org/ns/activitystreams#Public"], else: []),
2018-03-30 15:01:53 +02:00
note <-
CommonAPI.Utils.make_note_data(
actor.ap_id,
to,
context,
content_html,
attachments,
inReplyToActivity,
[],
cw
),
2017-05-20 13:35:22 +02:00
note <- note |> Map.put("id", id) |> Map.put("tag", tags),
2017-06-08 16:44:12 +02:00
note <- note |> Map.put("published", date),
2017-09-16 16:27:48 +02:00
note <- note |> Map.put("emoji", get_emoji(entry)),
2017-06-25 11:57:34 +02:00
note <- add_external_url(note, entry),
note <- note |> Map.put("cc", cc),
2017-05-20 13:35:22 +02:00
# TODO: Handle this case in make_note_data
2018-03-30 15:01:53 +02:00
note <-
if(
inReplyTo && !inReplyToActivity,
do: note |> Map.put("inReplyTo", inReplyTo),
else: note
) do
ActivityPub.create(%{
to: to,
actor: actor,
context: context,
object: note,
published: date,
local: false,
additional: %{"cc" => cc}
})
2017-05-20 13:35:22 +02:00
else
%Activity{} = activity -> {:ok, activity}
e -> {:error, e}
end
end
end