2018-12-23 20:04:54 +00:00
|
|
|
# Pleroma: A lightweight social networking server
|
2018-12-31 15:41:47 +00:00
|
|
|
# Copyright © 2017-2019 Pleroma Authors <https://pleroma.social/>
|
2018-12-23 20:04:54 +00:00
|
|
|
# SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
2017-05-20 11:35:22 +00:00
|
|
|
defmodule Pleroma.Web.OStatus.NoteHandler do
|
|
|
|
require Logger
|
2019-03-05 02:52:23 +00:00
|
|
|
|
2019-02-09 15:16:26 +00:00
|
|
|
alias Pleroma.Activity
|
|
|
|
alias Pleroma.Object
|
|
|
|
alias Pleroma.Web.ActivityPub.ActivityPub
|
|
|
|
alias Pleroma.Web.ActivityPub.Utils
|
2017-09-15 12:17:36 +00:00
|
|
|
alias Pleroma.Web.CommonAPI
|
2019-03-05 02:52:23 +00:00
|
|
|
alias Pleroma.Web.OStatus
|
|
|
|
alias Pleroma.Web.XML
|
2017-05-20 11:35:22 +00: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.
|
|
|
|
"""
|
2019-03-05 03:36:19 +00:00
|
|
|
def get_context(entry, in_reply_to) do
|
2018-03-30 13:01:53 +00:00
|
|
|
context =
|
|
|
|
(XML.string_from_xpath("//ostatus:conversation[1]", entry) ||
|
|
|
|
XML.string_from_xpath("//ostatus:conversation[1]/@ref", entry) || "")
|
|
|
|
|> String.trim()
|
2017-05-20 11:35:22 +00:00
|
|
|
|
2019-03-05 03:36:19 +00:00
|
|
|
with %{data: %{"context" => context}} <- Object.get_cached_by_ap_id(in_reply_to) do
|
2017-05-20 11:35:22 +00:00
|
|
|
context
|
2018-03-30 13:01:53 +00:00
|
|
|
else
|
|
|
|
_e ->
|
|
|
|
if String.length(context) > 0 do
|
|
|
|
context
|
|
|
|
else
|
|
|
|
Utils.generate_context_id()
|
|
|
|
end
|
2017-05-20 11:35:22 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-06-30 13:54:32 +00:00
|
|
|
def get_people_mentions(entry) do
|
2018-03-30 13:01:53 +00: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 11:35:22 +00:00
|
|
|
end
|
|
|
|
|
2017-06-30 13:54:32 +00:00
|
|
|
def get_collection_mentions(entry) do
|
|
|
|
transmogrify = fn
|
2018-03-30 13:01:53 +00:00
|
|
|
"http://activityschema.org/collection/public" ->
|
2017-06-30 13:54:32 +00:00
|
|
|
"https://www.w3.org/ns/activitystreams#Public"
|
2018-03-30 13:01:53 +00:00
|
|
|
|
|
|
|
group ->
|
2017-06-30 13:54:32 +00:00
|
|
|
group
|
|
|
|
end
|
|
|
|
|
2018-03-30 13:01:53 +00: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)
|
2017-06-30 13:54:32 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
def get_mentions(entry) do
|
2018-03-30 13:01:53 +00:00
|
|
|
(get_people_mentions(entry) ++ get_collection_mentions(entry))
|
|
|
|
|> Enum.filter(& &1)
|
2017-06-30 13:54:32 +00:00
|
|
|
end
|
|
|
|
|
2017-09-16 14:27:48 +00:00
|
|
|
def get_emoji(entry) do
|
|
|
|
try do
|
|
|
|
:xmerl_xpath.string('//link[@rel="emoji"]', entry)
|
2018-03-30 13:01:53 +00:00
|
|
|
|> Enum.reduce(%{}, fn emoji, acc ->
|
2017-09-16 14:27:48 +00: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 11:35:22 +00:00
|
|
|
def make_to_list(actor, mentions) do
|
|
|
|
[
|
2017-07-19 17:06:49 +00:00
|
|
|
actor.follower_address
|
2017-05-20 11:35:22 +00:00
|
|
|
] ++ mentions
|
|
|
|
end
|
|
|
|
|
2017-06-25 09:57:34 +00: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
|
|
|
|
|
2019-03-05 03:36:19 +00:00
|
|
|
def fetch_replied_to_activity(entry, in_reply_to) do
|
|
|
|
with %Activity{} = activity <- Activity.get_create_by_object_ap_id(in_reply_to) do
|
2017-09-05 12:01:37 +00:00
|
|
|
activity
|
|
|
|
else
|
|
|
|
_e ->
|
2019-03-05 03:36:19 +00:00
|
|
|
with in_reply_to_href when not is_nil(in_reply_to_href) <-
|
2018-03-30 13:01:53 +00:00
|
|
|
XML.string_from_xpath("//thr:in-reply-to[1]/@href", entry),
|
2019-03-05 03:36:19 +00:00
|
|
|
{:ok, [activity | _]} <- OStatus.fetch_activity_from_url(in_reply_to_href) do
|
2017-09-05 12:01:37 +00:00
|
|
|
activity
|
|
|
|
else
|
|
|
|
_e -> nil
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-03-07 14:45:13 +00:00
|
|
|
# TODO: Clean this up a bit.
|
2017-05-20 11:35:22 +00:00
|
|
|
def handle_note(entry, doc \\ nil) do
|
|
|
|
with id <- XML.string_from_xpath("//id", entry),
|
2019-01-21 06:14:20 +00:00
|
|
|
activity when is_nil(activity) <- Activity.get_create_by_object_ap_id(id),
|
2017-05-20 11:35:22 +00:00
|
|
|
[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 16:30:46 +00:00
|
|
|
cw <- OStatus.get_cw(entry),
|
2019-03-05 03:36:19 +00:00
|
|
|
in_reply_to <- XML.string_from_xpath("//thr:in-reply-to[1]/@ref", entry),
|
|
|
|
in_reply_to_activity <- fetch_replied_to_activity(entry, in_reply_to),
|
|
|
|
in_reply_to <-
|
|
|
|
(in_reply_to_activity && in_reply_to_activity.data["object"]["id"]) || in_reply_to,
|
2017-05-20 11:35:22 +00:00
|
|
|
attachments <- OStatus.get_attachments(entry),
|
2019-03-05 03:36:19 +00:00
|
|
|
context <- get_context(entry, in_reply_to),
|
2017-05-20 11:35:22 +00:00
|
|
|
tags <- OStatus.get_tags(entry),
|
|
|
|
mentions <- get_mentions(entry),
|
|
|
|
to <- make_to_list(actor, mentions),
|
|
|
|
date <- XML.string_from_xpath("//published", entry),
|
2018-03-07 14:45:13 +00:00
|
|
|
unlisted <- XML.string_from_xpath("//mastodon:scope", entry) == "unlisted",
|
|
|
|
cc <- if(unlisted, do: ["https://www.w3.org/ns/activitystreams#Public"], else: []),
|
2018-03-30 13:01:53 +00:00
|
|
|
note <-
|
|
|
|
CommonAPI.Utils.make_note_data(
|
|
|
|
actor.ap_id,
|
|
|
|
to,
|
|
|
|
context,
|
|
|
|
content_html,
|
|
|
|
attachments,
|
2019-03-05 03:36:19 +00:00
|
|
|
in_reply_to_activity,
|
2018-03-30 13:01:53 +00:00
|
|
|
[],
|
|
|
|
cw
|
|
|
|
),
|
2017-05-20 11:35:22 +00:00
|
|
|
note <- note |> Map.put("id", id) |> Map.put("tag", tags),
|
2017-06-08 14:44:12 +00:00
|
|
|
note <- note |> Map.put("published", date),
|
2017-09-16 14:27:48 +00:00
|
|
|
note <- note |> Map.put("emoji", get_emoji(entry)),
|
2017-06-25 09:57:34 +00:00
|
|
|
note <- add_external_url(note, entry),
|
2018-03-07 14:45:13 +00:00
|
|
|
note <- note |> Map.put("cc", cc),
|
2017-05-20 11:35:22 +00:00
|
|
|
# TODO: Handle this case in make_note_data
|
2018-03-30 13:01:53 +00:00
|
|
|
note <-
|
|
|
|
if(
|
2019-03-05 03:36:19 +00:00
|
|
|
in_reply_to && !in_reply_to_activity,
|
|
|
|
do: note |> Map.put("inReplyTo", in_reply_to),
|
2018-03-30 13:01:53 +00:00
|
|
|
else: note
|
|
|
|
) do
|
2018-04-21 09:58:04 +00:00
|
|
|
ActivityPub.create(%{
|
|
|
|
to: to,
|
|
|
|
actor: actor,
|
|
|
|
context: context,
|
|
|
|
object: note,
|
|
|
|
published: date,
|
|
|
|
local: false,
|
|
|
|
additional: %{"cc" => cc}
|
|
|
|
})
|
2017-05-20 11:35:22 +00:00
|
|
|
else
|
|
|
|
%Activity{} = activity -> {:ok, activity}
|
|
|
|
e -> {:error, e}
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|