2018-12-23 15:11:29 -05:00
|
|
|
# Pleroma: A lightweight social networking server
|
|
|
|
# Copyright © 2017-2018 Pleroma Authors <https://pleroma.social/>
|
|
|
|
# SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
2017-09-09 06:10:29 -04:00
|
|
|
defmodule Pleroma.Web.MastodonAPI.StatusViewTest do
|
|
|
|
use Pleroma.DataCase
|
|
|
|
|
2019-03-04 21:52:23 -05:00
|
|
|
alias Pleroma.Activity
|
2019-04-27 16:06:46 -04:00
|
|
|
alias Pleroma.Bookmark
|
2019-04-17 05:22:32 -04:00
|
|
|
alias Pleroma.Object
|
2019-04-17 07:52:01 -04:00
|
|
|
alias Pleroma.Repo
|
|
|
|
alias Pleroma.User
|
2017-09-17 07:54:14 -04:00
|
|
|
alias Pleroma.Web.CommonAPI
|
2019-03-21 19:25:41 -04:00
|
|
|
alias Pleroma.Web.CommonAPI.Utils
|
2019-02-10 16:57:38 -05:00
|
|
|
alias Pleroma.Web.MastodonAPI.AccountView
|
|
|
|
alias Pleroma.Web.MastodonAPI.StatusView
|
2017-09-09 06:10:29 -04:00
|
|
|
alias Pleroma.Web.OStatus
|
|
|
|
import Pleroma.Factory
|
2018-12-03 13:37:55 -05:00
|
|
|
import Tesla.Mock
|
|
|
|
|
|
|
|
setup do
|
|
|
|
mock(fn env -> apply(HttpRequestMock, :request, [env]) end)
|
|
|
|
:ok
|
|
|
|
end
|
2017-09-09 06:10:29 -04:00
|
|
|
|
2019-01-16 09:13:09 -05:00
|
|
|
test "returns a temporary ap_id based user for activities missing db users" do
|
|
|
|
user = insert(:user)
|
|
|
|
|
|
|
|
{:ok, activity} = CommonAPI.post(user, %{"status" => "Hey @shp!", "visibility" => "direct"})
|
|
|
|
|
|
|
|
Repo.delete(user)
|
|
|
|
Cachex.clear(:user_cache)
|
|
|
|
|
|
|
|
%{account: ms_user} = StatusView.render("status.json", activity: activity)
|
|
|
|
|
|
|
|
assert ms_user.acct == "erroruser@example.com"
|
|
|
|
end
|
|
|
|
|
|
|
|
test "tries to get a user by nickname if fetching by ap_id doesn't work" do
|
|
|
|
user = insert(:user)
|
|
|
|
|
|
|
|
{:ok, activity} = CommonAPI.post(user, %{"status" => "Hey @shp!", "visibility" => "direct"})
|
|
|
|
|
|
|
|
{:ok, user} =
|
|
|
|
user
|
|
|
|
|> Ecto.Changeset.change(%{ap_id: "#{user.ap_id}/extension/#{user.nickname}"})
|
|
|
|
|> Repo.update()
|
|
|
|
|
|
|
|
Cachex.clear(:user_cache)
|
|
|
|
|
|
|
|
result = StatusView.render("status.json", activity: activity)
|
|
|
|
|
|
|
|
assert result[:account][:id] == to_string(user.id)
|
|
|
|
end
|
2017-09-09 06:10:29 -04:00
|
|
|
|
2018-11-03 11:28:29 -04:00
|
|
|
test "a note with null content" do
|
|
|
|
note = insert(:note_activity)
|
2018-11-25 16:08:55 -05:00
|
|
|
note_object = Object.normalize(note.data["object"])
|
2018-11-03 11:40:57 -04:00
|
|
|
|
|
|
|
data =
|
2018-11-25 16:08:55 -05:00
|
|
|
note_object.data
|
|
|
|
|> Map.put("content", nil)
|
2018-11-03 11:40:57 -04:00
|
|
|
|
2018-11-25 16:08:55 -05:00
|
|
|
Object.change(note_object, %{data: data})
|
2019-04-17 08:46:59 -04:00
|
|
|
|> Object.update_and_set_cache()
|
2018-11-03 11:28:29 -04:00
|
|
|
|
2018-12-06 13:50:34 -05:00
|
|
|
User.get_cached_by_ap_id(note.data["actor"])
|
2018-11-03 11:28:29 -04:00
|
|
|
|
|
|
|
status = StatusView.render("status.json", %{activity: note})
|
|
|
|
|
|
|
|
assert status.content == ""
|
|
|
|
end
|
|
|
|
|
2017-09-09 06:10:29 -04:00
|
|
|
test "a note activity" do
|
|
|
|
note = insert(:note_activity)
|
|
|
|
user = User.get_cached_by_ap_id(note.data["actor"])
|
|
|
|
|
2019-03-21 19:25:41 -04:00
|
|
|
convo_id = Utils.context_to_conversation_id(note.data["object"]["context"])
|
|
|
|
|
2017-09-09 06:10:29 -04:00
|
|
|
status = StatusView.render("status.json", %{activity: note})
|
|
|
|
|
2018-03-30 09:01:53 -04:00
|
|
|
created_at =
|
|
|
|
(note.data["object"]["published"] || "")
|
|
|
|
|> String.replace(~r/\.\d+Z/, ".000Z")
|
2017-09-12 07:31:17 -04:00
|
|
|
|
2017-09-09 06:10:29 -04:00
|
|
|
expected = %{
|
2017-10-31 09:51:41 -04:00
|
|
|
id: to_string(note.id),
|
2017-09-09 06:10:29 -04:00
|
|
|
uri: note.data["object"]["id"],
|
2019-02-20 11:36:16 -05:00
|
|
|
url: Pleroma.Web.Router.Helpers.o_status_url(Pleroma.Web.Endpoint, :notice, note),
|
2017-09-09 06:10:29 -04:00
|
|
|
account: AccountView.render("account.json", %{user: user}),
|
|
|
|
in_reply_to_id: nil,
|
|
|
|
in_reply_to_account_id: nil,
|
2019-01-27 07:21:51 -05:00
|
|
|
card: nil,
|
2017-09-09 06:10:29 -04:00
|
|
|
reblog: nil,
|
|
|
|
content: HtmlSanitizeEx.basic_html(note.data["object"]["content"]),
|
2017-09-12 07:31:17 -04:00
|
|
|
created_at: created_at,
|
2017-09-09 06:10:29 -04:00
|
|
|
reblogs_count: 0,
|
2018-09-20 10:10:46 -04:00
|
|
|
replies_count: 0,
|
2017-09-09 06:10:29 -04:00
|
|
|
favourites_count: 0,
|
|
|
|
reblogged: false,
|
2018-09-18 20:04:56 -04:00
|
|
|
bookmarked: false,
|
2017-09-09 06:10:29 -04:00
|
|
|
favourited: false,
|
|
|
|
muted: false,
|
2019-01-08 03:27:02 -05:00
|
|
|
pinned: false,
|
2017-09-09 06:10:29 -04:00
|
|
|
sensitive: false,
|
2016-12-31 19:10:08 -05:00
|
|
|
spoiler_text: HtmlSanitizeEx.basic_html(note.data["object"]["summary"]),
|
2017-09-09 06:10:29 -04:00
|
|
|
visibility: "public",
|
|
|
|
media_attachments: [],
|
|
|
|
mentions: [],
|
2018-12-13 07:13:02 -05:00
|
|
|
tags: [
|
|
|
|
%{
|
|
|
|
name: "#{note.data["object"]["tag"]}",
|
|
|
|
url: "/tag/#{note.data["object"]["tag"]}"
|
|
|
|
}
|
|
|
|
],
|
2017-09-14 03:14:08 -04:00
|
|
|
application: %{
|
|
|
|
name: "Web",
|
|
|
|
website: nil
|
|
|
|
},
|
2017-10-23 10:27:51 -04:00
|
|
|
language: nil,
|
|
|
|
emojis: [
|
|
|
|
%{
|
|
|
|
shortcode: "2hu",
|
|
|
|
url: "corndog.png",
|
2018-09-10 19:40:29 -04:00
|
|
|
static_url: "corndog.png",
|
|
|
|
visible_in_picker: false
|
2017-10-23 10:27:51 -04:00
|
|
|
}
|
2019-03-11 08:48:27 -04:00
|
|
|
],
|
|
|
|
pleroma: %{
|
2019-03-21 19:25:41 -04:00
|
|
|
local: true,
|
2016-12-31 19:10:08 -05:00
|
|
|
conversation_id: convo_id,
|
2019-04-22 04:53:37 -04:00
|
|
|
in_reply_to_account_acct: nil,
|
2016-12-31 19:10:08 -05:00
|
|
|
content: %{"text/plain" => HtmlSanitizeEx.strip_tags(note.data["object"]["content"])},
|
|
|
|
spoiler_text: %{"text/plain" => HtmlSanitizeEx.strip_tags(note.data["object"]["summary"])}
|
2019-03-11 08:48:27 -04:00
|
|
|
}
|
2017-09-09 06:10:29 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
assert status == expected
|
|
|
|
end
|
|
|
|
|
2019-02-27 10:46:47 -05:00
|
|
|
test "tells if the message is muted for some reason" do
|
|
|
|
user = insert(:user)
|
|
|
|
other_user = insert(:user)
|
|
|
|
|
|
|
|
{:ok, user} = User.mute(user, other_user)
|
|
|
|
|
|
|
|
{:ok, activity} = CommonAPI.post(other_user, %{"status" => "test"})
|
|
|
|
status = StatusView.render("status.json", %{activity: activity})
|
|
|
|
|
|
|
|
assert status.muted == false
|
|
|
|
|
|
|
|
status = StatusView.render("status.json", %{activity: activity, for: user})
|
|
|
|
|
|
|
|
assert status.muted == true
|
|
|
|
end
|
|
|
|
|
2019-04-27 16:06:46 -04:00
|
|
|
test "tells if the status is bookmarked" do
|
|
|
|
user = insert(:user)
|
|
|
|
|
|
|
|
{:ok, activity} = CommonAPI.post(user, %{"status" => "Cute girls doing cute things"})
|
|
|
|
status = StatusView.render("status.json", %{activity: activity})
|
|
|
|
|
|
|
|
assert status.bookmarked == false
|
|
|
|
|
|
|
|
status = StatusView.render("status.json", %{activity: activity, for: user})
|
|
|
|
|
|
|
|
assert status.bookmarked == false
|
|
|
|
|
|
|
|
{:ok, _bookmark} = Bookmark.create(user.id, activity.id)
|
|
|
|
|
|
|
|
status = StatusView.render("status.json", %{activity: activity, for: user})
|
|
|
|
|
|
|
|
assert status.bookmarked == true
|
|
|
|
end
|
|
|
|
|
2018-03-27 11:43:08 -04:00
|
|
|
test "a reply" do
|
|
|
|
note = insert(:note_activity)
|
|
|
|
user = insert(:user)
|
2018-03-30 09:01:53 -04:00
|
|
|
|
|
|
|
{:ok, activity} =
|
|
|
|
CommonAPI.post(user, %{"status" => "he", "in_reply_to_status_id" => note.id})
|
2018-03-27 11:43:08 -04:00
|
|
|
|
2018-03-27 12:18:24 -04:00
|
|
|
status = StatusView.render("status.json", %{activity: activity})
|
|
|
|
|
2018-06-15 15:37:29 -04:00
|
|
|
assert status.in_reply_to_id == to_string(note.id)
|
2018-03-27 12:18:24 -04:00
|
|
|
|
|
|
|
[status] = StatusView.render("index.json", %{activities: [activity], as: :activity})
|
|
|
|
|
2018-06-15 15:37:29 -04:00
|
|
|
assert status.in_reply_to_id == to_string(note.id)
|
2018-03-27 11:43:08 -04:00
|
|
|
end
|
|
|
|
|
2017-09-09 06:10:29 -04:00
|
|
|
test "contains mentions" do
|
|
|
|
incoming = File.read!("test/fixtures/incoming_reply_mastodon.xml")
|
2017-12-11 14:01:36 -05:00
|
|
|
# a user with this ap id might be in the cache.
|
|
|
|
recipient = "https://pleroma.soykaf.com/users/lain"
|
2018-03-22 07:44:32 -04:00
|
|
|
user = insert(:user, %{ap_id: recipient})
|
2018-03-16 01:30:28 -04:00
|
|
|
|
2017-09-09 06:10:29 -04:00
|
|
|
{:ok, [activity]} = OStatus.handle_incoming(incoming)
|
|
|
|
|
|
|
|
status = StatusView.render("status.json", %{activity: activity})
|
|
|
|
|
2019-04-22 03:20:43 -04:00
|
|
|
actor = User.get_cached_by_ap_id(activity.actor)
|
2019-01-18 20:26:52 -05:00
|
|
|
|
|
|
|
assert status.mentions ==
|
|
|
|
Enum.map([user, actor], fn u -> AccountView.render("mention.json", %{user: u}) end)
|
2017-09-09 06:10:29 -04:00
|
|
|
end
|
2017-09-10 05:51:01 -04:00
|
|
|
|
|
|
|
test "attachments" do
|
|
|
|
object = %{
|
|
|
|
"type" => "Image",
|
|
|
|
"url" => [
|
|
|
|
%{
|
|
|
|
"mediaType" => "image/png",
|
|
|
|
"href" => "someurl"
|
|
|
|
}
|
|
|
|
],
|
|
|
|
"uuid" => 6
|
|
|
|
}
|
|
|
|
|
|
|
|
expected = %{
|
2017-11-15 12:58:13 -05:00
|
|
|
id: "1638338801",
|
2017-09-10 05:51:01 -04:00
|
|
|
type: "image",
|
|
|
|
url: "someurl",
|
|
|
|
remote_url: "someurl",
|
2017-09-14 02:38:48 -04:00
|
|
|
preview_url: "someurl",
|
2018-07-16 23:37:40 -04:00
|
|
|
text_url: "someurl",
|
2019-03-15 04:58:12 -04:00
|
|
|
description: nil,
|
|
|
|
pleroma: %{mime_type: "image/png"}
|
2017-09-10 05:51:01 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
assert expected == StatusView.render("attachment.json", %{attachment: object})
|
2017-09-14 02:08:32 -04:00
|
|
|
|
|
|
|
# If theres a "id", use that instead of the generated one
|
|
|
|
object = Map.put(object, "id", 2)
|
2017-11-15 12:58:13 -05:00
|
|
|
assert %{id: "2"} = StatusView.render("attachment.json", %{attachment: object})
|
2017-09-10 05:51:01 -04:00
|
|
|
end
|
2017-09-17 07:54:14 -04:00
|
|
|
|
|
|
|
test "a reblog" do
|
|
|
|
user = insert(:user)
|
|
|
|
activity = insert(:note_activity)
|
|
|
|
|
|
|
|
{:ok, reblog, _} = CommonAPI.repeat(activity.id, user)
|
|
|
|
|
|
|
|
represented = StatusView.render("status.json", %{for: user, activity: reblog})
|
|
|
|
|
2017-10-31 09:51:41 -04:00
|
|
|
assert represented[:id] == to_string(reblog.id)
|
|
|
|
assert represented[:reblog][:id] == to_string(activity.id)
|
2017-11-11 05:18:05 -05:00
|
|
|
assert represented[:emojis] == []
|
2017-09-17 07:54:14 -04:00
|
|
|
end
|
2018-12-13 07:13:02 -05:00
|
|
|
|
2018-12-23 08:42:42 -05:00
|
|
|
test "a peertube video" do
|
|
|
|
user = insert(:user)
|
|
|
|
|
|
|
|
{:ok, object} =
|
2019-04-17 07:21:39 -04:00
|
|
|
Pleroma.Object.Fetcher.fetch_object_from_id(
|
2018-12-23 08:42:42 -05:00
|
|
|
"https://peertube.moe/videos/watch/df5f464b-be8d-46fb-ad81-2d4c2d1630e3"
|
|
|
|
)
|
|
|
|
|
2019-01-21 01:14:20 -05:00
|
|
|
%Activity{} = activity = Activity.get_create_by_object_ap_id(object.data["id"])
|
2018-12-23 08:42:42 -05:00
|
|
|
|
|
|
|
represented = StatusView.render("status.json", %{for: user, activity: activity})
|
|
|
|
|
|
|
|
assert represented[:id] == to_string(activity.id)
|
|
|
|
assert length(represented[:media_attachments]) == 1
|
|
|
|
end
|
|
|
|
|
2018-12-13 07:13:02 -05:00
|
|
|
describe "build_tags/1" do
|
|
|
|
test "it returns a a dictionary tags" do
|
2018-12-14 14:56:37 -05:00
|
|
|
object_tags = [
|
|
|
|
"fediverse",
|
|
|
|
"mastodon",
|
|
|
|
"nextcloud",
|
|
|
|
%{
|
|
|
|
"href" => "https://kawen.space/users/lain",
|
|
|
|
"name" => "@lain@kawen.space",
|
|
|
|
"type" => "Mention"
|
|
|
|
}
|
|
|
|
]
|
|
|
|
|
|
|
|
assert StatusView.build_tags(object_tags) == [
|
2018-12-13 07:13:02 -05:00
|
|
|
%{name: "fediverse", url: "/tag/fediverse"},
|
|
|
|
%{name: "mastodon", url: "/tag/mastodon"},
|
|
|
|
%{name: "nextcloud", url: "/tag/nextcloud"}
|
|
|
|
]
|
|
|
|
end
|
|
|
|
end
|
2019-02-06 13:12:26 -05:00
|
|
|
|
|
|
|
describe "rich media cards" do
|
|
|
|
test "a rich media card without a site name renders correctly" do
|
|
|
|
page_url = "http://example.com"
|
|
|
|
|
|
|
|
card = %{
|
|
|
|
url: page_url,
|
|
|
|
image: page_url <> "/example.jpg",
|
|
|
|
title: "Example website"
|
|
|
|
}
|
|
|
|
|
|
|
|
%{provider_name: "example.com"} =
|
|
|
|
StatusView.render("card.json", %{page_url: page_url, rich_media: card})
|
|
|
|
end
|
|
|
|
|
|
|
|
test "a rich media card without a site name or image renders correctly" do
|
|
|
|
page_url = "http://example.com"
|
|
|
|
|
|
|
|
card = %{
|
|
|
|
url: page_url,
|
|
|
|
title: "Example website"
|
|
|
|
}
|
|
|
|
|
|
|
|
%{provider_name: "example.com"} =
|
|
|
|
StatusView.render("card.json", %{page_url: page_url, rich_media: card})
|
|
|
|
end
|
|
|
|
|
|
|
|
test "a rich media card without an image renders correctly" do
|
|
|
|
page_url = "http://example.com"
|
|
|
|
|
|
|
|
card = %{
|
|
|
|
url: page_url,
|
|
|
|
site_name: "Example site name",
|
|
|
|
title: "Example website"
|
|
|
|
}
|
|
|
|
|
|
|
|
%{provider_name: "Example site name"} =
|
|
|
|
StatusView.render("card.json", %{page_url: page_url, rich_media: card})
|
|
|
|
end
|
2019-02-06 13:27:55 -05:00
|
|
|
|
|
|
|
test "a rich media card with all relevant data renders correctly" do
|
|
|
|
page_url = "http://example.com"
|
|
|
|
|
|
|
|
card = %{
|
|
|
|
url: page_url,
|
|
|
|
site_name: "Example site name",
|
|
|
|
title: "Example website",
|
|
|
|
image: page_url <> "/example.jpg",
|
|
|
|
description: "Example description"
|
|
|
|
}
|
|
|
|
|
|
|
|
%{provider_name: "Example site name"} =
|
|
|
|
StatusView.render("card.json", %{page_url: page_url, rich_media: card})
|
|
|
|
end
|
2019-02-06 13:12:26 -05:00
|
|
|
end
|
2017-09-09 06:10:29 -04:00
|
|
|
end
|