1 # Pleroma: A lightweight social networking server
2 # Copyright © 2017-2018 Pleroma Authors <https://pleroma.social/>
3 # SPDX-License-Identifier: AGPL-3.0-only
5 defmodule Pleroma.Web.MastodonAPI.StatusViewTest do
13 alias Pleroma.Web.CommonAPI
14 alias Pleroma.Web.CommonAPI.Utils
15 alias Pleroma.Web.MastodonAPI.AccountView
16 alias Pleroma.Web.MastodonAPI.StatusView
17 alias Pleroma.Web.OStatus
18 import Pleroma.Factory
22 mock(fn env -> apply(HttpRequestMock, :request, [env]) end)
26 test "returns the direct conversation id when given the `with_conversation_id` option" do
29 {:ok, activity} = CommonAPI.post(user, %{"status" => "Hey @shp!", "visibility" => "direct"})
32 StatusView.render("status.json",
34 with_direct_conversation_id: true,
38 assert status[:pleroma][:direct_conversation_id]
41 test "returns a temporary ap_id based user for activities missing db users" do
44 {:ok, activity} = CommonAPI.post(user, %{"status" => "Hey @shp!", "visibility" => "direct"})
47 Cachex.clear(:user_cache)
49 %{account: ms_user} = StatusView.render("status.json", activity: activity)
51 assert ms_user.acct == "erroruser@example.com"
54 test "tries to get a user by nickname if fetching by ap_id doesn't work" do
57 {:ok, activity} = CommonAPI.post(user, %{"status" => "Hey @shp!", "visibility" => "direct"})
61 |> Ecto.Changeset.change(%{ap_id: "#{user.ap_id}/extension/#{user.nickname}"})
64 Cachex.clear(:user_cache)
66 result = StatusView.render("status.json", activity: activity)
68 assert result[:account][:id] == to_string(user.id)
71 test "a note with null content" do
72 note = insert(:note_activity)
73 note_object = Object.normalize(note)
77 |> Map.put("content", nil)
79 Object.change(note_object, %{data: data})
80 |> Object.update_and_set_cache()
82 User.get_cached_by_ap_id(note.data["actor"])
84 status = StatusView.render("status.json", %{activity: note})
86 assert status.content == ""
89 test "a note activity" do
90 note = insert(:note_activity)
91 object_data = Object.normalize(note).data
92 user = User.get_cached_by_ap_id(note.data["actor"])
94 convo_id = Utils.context_to_conversation_id(object_data["context"])
96 status = StatusView.render("status.json", %{activity: note})
99 (object_data["published"] || "")
100 |> String.replace(~r/\.\d+Z/, ".000Z")
103 id: to_string(note.id),
104 uri: object_data["id"],
105 url: Pleroma.Web.Router.Helpers.o_status_url(Pleroma.Web.Endpoint, :notice, note),
106 account: AccountView.render("account.json", %{user: user}),
108 in_reply_to_account_id: nil,
111 content: HtmlSanitizeEx.basic_html(object_data["content"]),
112 created_at: created_at,
123 spoiler_text: HtmlSanitizeEx.basic_html(object_data["summary"]),
124 visibility: "public",
125 media_attachments: [],
129 name: "#{object_data["tag"]}",
130 url: "/tag/#{object_data["tag"]}"
142 static_url: "corndog.png",
143 visible_in_picker: false
148 conversation_id: convo_id,
149 in_reply_to_account_acct: nil,
150 content: %{"text/plain" => HtmlSanitizeEx.strip_tags(object_data["content"])},
151 spoiler_text: %{"text/plain" => HtmlSanitizeEx.strip_tags(object_data["summary"])},
153 direct_conversation_id: nil,
158 assert status == expected
161 test "tells if the message is muted for some reason" do
163 other_user = insert(:user)
165 {:ok, user} = User.mute(user, other_user)
167 {:ok, activity} = CommonAPI.post(other_user, %{"status" => "test"})
168 status = StatusView.render("status.json", %{activity: activity})
170 assert status.muted == false
172 status = StatusView.render("status.json", %{activity: activity, for: user})
174 assert status.muted == true
177 test "tells if the message is thread muted" do
179 other_user = insert(:user)
181 {:ok, user} = User.mute(user, other_user)
183 {:ok, activity} = CommonAPI.post(other_user, %{"status" => "test"})
184 status = StatusView.render("status.json", %{activity: activity, for: user})
186 assert status.pleroma.thread_muted == false
188 {:ok, activity} = CommonAPI.add_mute(user, activity)
190 status = StatusView.render("status.json", %{activity: activity, for: user})
192 assert status.pleroma.thread_muted == true
195 test "tells if the status is bookmarked" do
198 {:ok, activity} = CommonAPI.post(user, %{"status" => "Cute girls doing cute things"})
199 status = StatusView.render("status.json", %{activity: activity})
201 assert status.bookmarked == false
203 status = StatusView.render("status.json", %{activity: activity, for: user})
205 assert status.bookmarked == false
207 {:ok, _bookmark} = Bookmark.create(user.id, activity.id)
209 activity = Activity.get_by_id_with_object(activity.id)
211 status = StatusView.render("status.json", %{activity: activity, for: user})
213 assert status.bookmarked == true
217 note = insert(:note_activity)
221 CommonAPI.post(user, %{"status" => "he", "in_reply_to_status_id" => note.id})
223 status = StatusView.render("status.json", %{activity: activity})
225 assert status.in_reply_to_id == to_string(note.id)
227 [status] = StatusView.render("index.json", %{activities: [activity], as: :activity})
229 assert status.in_reply_to_id == to_string(note.id)
232 test "contains mentions" do
233 incoming = File.read!("test/fixtures/incoming_reply_mastodon.xml")
234 # a user with this ap id might be in the cache.
235 recipient = "https://pleroma.soykaf.com/users/lain"
236 user = insert(:user, %{ap_id: recipient})
238 {:ok, [activity]} = OStatus.handle_incoming(incoming)
240 status = StatusView.render("status.json", %{activity: activity})
242 assert status.mentions ==
243 Enum.map([user], fn u -> AccountView.render("mention.json", %{user: u}) end)
246 test "create mentions from the 'to' field" do
247 %User{ap_id: recipient_ap_id} = insert(:user)
248 cc = insert_pair(:user) |> Enum.map(& &1.ap_id)
253 "to" => [recipient_ap_id],
259 insert(:note_activity, %{
261 recipients: [recipient_ap_id | cc]
264 assert length(activity.recipients) == 3
266 %{mentions: [mention] = mentions} = StatusView.render("status.json", %{activity: activity})
268 assert length(mentions) == 1
269 assert mention.url == recipient_ap_id
272 test "create mentions from the 'tag' field" do
273 recipient = insert(:user)
274 cc = insert_pair(:user) |> Enum.map(& &1.ap_id)
282 "href" => recipient.ap_id,
283 "name" => recipient.nickname,
287 "href" => "https://example.com/search?tag=test",
296 insert(:note_activity, %{
298 recipients: [recipient.ap_id | cc]
301 assert length(activity.recipients) == 3
303 %{mentions: [mention] = mentions} = StatusView.render("status.json", %{activity: activity})
305 assert length(mentions) == 1
306 assert mention.url == recipient.ap_id
309 test "attachments" do
314 "mediaType" => "image/png",
325 remote_url: "someurl",
326 preview_url: "someurl",
329 pleroma: %{mime_type: "image/png"}
332 assert expected == StatusView.render("attachment.json", %{attachment: object})
334 # If theres a "id", use that instead of the generated one
335 object = Map.put(object, "id", 2)
336 assert %{id: "2"} = StatusView.render("attachment.json", %{attachment: object})
339 test "put the url advertised in the Activity in to the url attribute" do
340 id = "https://wedistribute.org/wp-json/pterotype/v1/object/85810"
341 [activity] = Activity.search(nil, id)
343 status = StatusView.render("status.json", %{activity: activity})
345 assert status.uri == id
346 assert status.url == "https://wedistribute.org/2019/07/mastodon-drops-ostatus/"
351 activity = insert(:note_activity)
353 {:ok, reblog, _} = CommonAPI.repeat(activity.id, user)
355 represented = StatusView.render("status.json", %{for: user, activity: reblog})
357 assert represented[:id] == to_string(reblog.id)
358 assert represented[:reblog][:id] == to_string(activity.id)
359 assert represented[:emojis] == []
362 test "a peertube video" do
366 Pleroma.Object.Fetcher.fetch_object_from_id(
367 "https://peertube.moe/videos/watch/df5f464b-be8d-46fb-ad81-2d4c2d1630e3"
370 %Activity{} = activity = Activity.get_create_by_object_ap_id(object.data["id"])
372 represented = StatusView.render("status.json", %{for: user, activity: activity})
374 assert represented[:id] == to_string(activity.id)
375 assert length(represented[:media_attachments]) == 1
378 describe "build_tags/1" do
379 test "it returns a a dictionary tags" do
385 "href" => "https://kawen.space/users/lain",
386 "name" => "@lain@kawen.space",
391 assert StatusView.build_tags(object_tags) == [
392 %{name: "fediverse", url: "/tag/fediverse"},
393 %{name: "mastodon", url: "/tag/mastodon"},
394 %{name: "nextcloud", url: "/tag/nextcloud"}
399 describe "rich media cards" do
400 test "a rich media card without a site name renders correctly" do
401 page_url = "http://example.com"
405 image: page_url <> "/example.jpg",
406 title: "Example website"
409 %{provider_name: "example.com"} =
410 StatusView.render("card.json", %{page_url: page_url, rich_media: card})
413 test "a rich media card without a site name or image renders correctly" do
414 page_url = "http://example.com"
418 title: "Example website"
421 %{provider_name: "example.com"} =
422 StatusView.render("card.json", %{page_url: page_url, rich_media: card})
425 test "a rich media card without an image renders correctly" do
426 page_url = "http://example.com"
430 site_name: "Example site name",
431 title: "Example website"
434 %{provider_name: "Example site name"} =
435 StatusView.render("card.json", %{page_url: page_url, rich_media: card})
438 test "a rich media card with all relevant data renders correctly" do
439 page_url = "http://example.com"
443 site_name: "Example site name",
444 title: "Example website",
445 image: page_url <> "/example.jpg",
446 description: "Example description"
449 %{provider_name: "Example site name"} =
450 StatusView.render("card.json", %{page_url: page_url, rich_media: card})
454 describe "poll view" do
455 test "renders a poll" do
459 CommonAPI.post(user, %{
460 "status" => "Is Tenshi eating a corndog cute?",
462 "options" => ["absolutely!", "sure", "yes", "why are you even asking?"],
467 object = Object.normalize(activity)
472 id: to_string(object.id),
475 %{title: "absolutely!", votes_count: 0},
476 %{title: "sure", votes_count: 0},
477 %{title: "yes", votes_count: 0},
478 %{title: "why are you even asking?", votes_count: 0}
484 result = StatusView.render("poll.json", %{object: object})
485 expires_at = result.expires_at
486 result = Map.delete(result, :expires_at)
488 assert result == expected
490 expires_at = NaiveDateTime.from_iso8601!(expires_at)
491 assert NaiveDateTime.diff(expires_at, NaiveDateTime.utc_now()) in 15..20
494 test "detects if it is multiple choice" do
498 CommonAPI.post(user, %{
499 "status" => "Which Mastodon developer is your favourite?",
501 "options" => ["Gargron", "Eugen"],
507 object = Object.normalize(activity)
509 assert %{multiple: true} = StatusView.render("poll.json", %{object: object})
512 test "detects emoji" do
516 CommonAPI.post(user, %{
517 "status" => "What's with the smug face?",
519 "options" => [":blank: sip", ":blank::blank: sip", ":blank::blank::blank: sip"],
524 object = Object.normalize(activity)
526 assert %{emojis: [%{shortcode: "blank"}]} =
527 StatusView.render("poll.json", %{object: object})
530 test "detects vote status" do
532 other_user = insert(:user)
535 CommonAPI.post(user, %{
536 "status" => "Which input devices do you use?",
538 "options" => ["mouse", "trackball", "trackpoint"],
544 object = Object.normalize(activity)
546 {:ok, _, object} = CommonAPI.vote(other_user, object, [1, 2])
548 result = StatusView.render("poll.json", %{object: object, for: other_user})
550 assert result[:voted] == true
551 assert Enum.at(result[:options], 1)[:votes_count] == 1
552 assert Enum.at(result[:options], 2)[:votes_count] == 1
555 test "does not crash on polls with no end date" do
556 object = Object.normalize("https://skippers-bin.com/notes/7x9tmrp97i")
557 result = StatusView.render("poll.json", %{object: object})
559 assert result[:expires_at] == nil
560 assert result[:expired] == false
564 test "embeds a relationship in the account" do
566 other_user = insert(:user)
569 CommonAPI.post(user, %{
570 "status" => "drink more water"
573 result = StatusView.render("status.json", %{activity: activity, for: other_user})
575 assert result[:account][:pleroma][:relationship] ==
576 AccountView.render("relationship.json", %{user: other_user, target: user})
579 test "embeds a relationship in the account in reposts" do
581 other_user = insert(:user)
584 CommonAPI.post(user, %{
588 {:ok, activity, _object} = CommonAPI.repeat(activity.id, other_user)
590 result = StatusView.render("status.json", %{activity: activity, for: user})
592 assert result[:account][:pleroma][:relationship] ==
593 AccountView.render("relationship.json", %{user: user, target: other_user})
595 assert result[:reblog][:account][:pleroma][:relationship] ==
596 AccountView.render("relationship.json", %{user: user, target: user})
599 test "visibility/list" do
602 {:ok, list} = Pleroma.List.create("foo", user)
605 CommonAPI.post(user, %{"status" => "foobar", "visibility" => "list:#{list.id}"})
607 status = StatusView.render("status.json", activity: activity)
609 assert status.visibility == "list"