1 defmodule Pleroma.Web.MastodonAPI.MastodonAPIControllerTest do
2 use Pleroma.Web.ConnCase
4 alias Pleroma.Web.TwitterAPI.TwitterAPI
5 alias Pleroma.{Repo, User, Activity, Notification}
6 alias Pleroma.Web.{OStatus, CommonAPI}
9 import ExUnit.CaptureLog
11 test "the home timeline", %{conn: conn} do
13 following = insert(:user)
15 {:ok, _activity} = TwitterAPI.create_status(following, %{"status" => "test"})
19 |> assign(:user, user)
20 |> get("/api/v1/timelines/home")
22 assert length(json_response(conn, 200)) == 0
24 {:ok, user} = User.follow(user, following)
28 |> assign(:user, user)
29 |> get("/api/v1/timelines/home")
31 assert [%{"content" => "test"}] = json_response(conn, 200)
34 test "the public timeline", %{conn: conn} do
35 following = insert(:user)
38 {:ok, _activity} = TwitterAPI.create_status(following, %{"status" => "test"})
41 OStatus.fetch_activity_from_url("https://shitposter.club/notice/2827873")
45 |> get("/api/v1/timelines/public", %{"local" => "False"})
47 assert length(json_response(conn, 200)) == 2
51 |> get("/api/v1/timelines/public", %{"local" => "True"})
53 assert [%{"content" => "test"}] = json_response(conn, 200)
57 |> get("/api/v1/timelines/public", %{"local" => "1"})
59 assert [%{"content" => "test"}] = json_response(conn, 200)
63 test "posting a status", %{conn: conn} do
66 idempotency_key = "Pikachu rocks!"
70 |> assign(:user, user)
71 |> put_req_header("idempotency-key", idempotency_key)
72 |> post("/api/v1/statuses", %{
74 "spoiler_text" => "2hu",
75 "sensitive" => "false"
78 {:ok, ttl} = Cachex.ttl(:idempotency_cache, idempotency_key)
80 assert ttl > :timer.seconds(6 * 60 * 60 - 1)
82 assert %{"content" => "cofe", "id" => id, "spoiler_text" => "2hu", "sensitive" => false} =
83 json_response(conn_one, 200)
85 assert Repo.get(Activity, id)
89 |> assign(:user, user)
90 |> put_req_header("idempotency-key", idempotency_key)
91 |> post("/api/v1/statuses", %{
93 "spoiler_text" => "2hu",
94 "sensitive" => "false"
97 assert %{"id" => second_id} = json_response(conn_two, 200)
99 assert id == second_id
103 |> assign(:user, user)
104 |> post("/api/v1/statuses", %{
106 "spoiler_text" => "2hu",
107 "sensitive" => "false"
110 assert %{"id" => third_id} = json_response(conn_three, 200)
112 refute id == third_id
115 test "posting a sensitive status", %{conn: conn} do
120 |> assign(:user, user)
121 |> post("/api/v1/statuses", %{"status" => "cofe", "sensitive" => true})
123 assert %{"content" => "cofe", "id" => id, "sensitive" => true} = json_response(conn, 200)
124 assert Repo.get(Activity, id)
127 test "posting a direct status", %{conn: conn} do
128 user1 = insert(:user)
129 user2 = insert(:user)
130 content = "direct cofe @#{user2.nickname}"
134 |> assign(:user, user1)
135 |> post("api/v1/statuses", %{"status" => content, "visibility" => "direct"})
137 assert %{"id" => id, "visibility" => "direct"} = json_response(conn, 200)
138 assert activity = Repo.get(Activity, id)
139 assert activity.recipients == [user2.ap_id]
140 assert activity.data["to"] == [user2.ap_id]
141 assert activity.data["cc"] == []
144 test "direct timeline", %{conn: conn} do
145 user_one = insert(:user)
146 user_two = insert(:user)
148 {:ok, user_two} = User.follow(user_two, user_one)
151 CommonAPI.post(user_one, %{
152 "status" => "Hi @#{user_two.nickname}!",
153 "visibility" => "direct"
156 {:ok, _follower_only} =
157 CommonAPI.post(user_one, %{
158 "status" => "Hi @#{user_two.nickname}!",
159 "visibility" => "private"
162 # Only direct should be visible here
165 |> assign(:user, user_two)
166 |> get("api/v1/timelines/direct")
168 [status] = json_response(res_conn, 200)
170 assert %{"visibility" => "direct"} = status
171 assert status["url"] != direct.data["id"]
173 # Both should be visible here
176 |> assign(:user, user_two)
177 |> get("api/v1/timelines/home")
179 [_s1, _s2] = json_response(res_conn, 200)
182 test "replying to a status", %{conn: conn} do
185 {:ok, replied_to} = TwitterAPI.create_status(user, %{"status" => "cofe"})
189 |> assign(:user, user)
190 |> post("/api/v1/statuses", %{"status" => "xD", "in_reply_to_id" => replied_to.id})
192 assert %{"content" => "xD", "id" => id} = json_response(conn, 200)
194 activity = Repo.get(Activity, id)
196 assert activity.data["context"] == replied_to.data["context"]
197 assert activity.data["object"]["inReplyToStatusId"] == replied_to.id
200 test "verify_credentials", %{conn: conn} do
205 |> assign(:user, user)
206 |> get("/api/v1/accounts/verify_credentials")
208 assert %{"id" => id} = json_response(conn, 200)
209 assert id == to_string(user.id)
212 test "get a status", %{conn: conn} do
213 activity = insert(:note_activity)
217 |> get("/api/v1/statuses/#{activity.id}")
219 assert %{"id" => id} = json_response(conn, 200)
220 assert id == to_string(activity.id)
223 describe "deleting a status" do
224 test "when you created it", %{conn: conn} do
225 activity = insert(:note_activity)
226 author = User.get_by_ap_id(activity.data["actor"])
230 |> assign(:user, author)
231 |> delete("/api/v1/statuses/#{activity.id}")
233 assert %{} = json_response(conn, 200)
235 assert Repo.get(Activity, activity.id) == nil
238 test "when you didn't create it", %{conn: conn} do
239 activity = insert(:note_activity)
244 |> assign(:user, user)
245 |> delete("/api/v1/statuses/#{activity.id}")
247 assert %{"error" => _} = json_response(conn, 403)
249 assert Repo.get(Activity, activity.id) == activity
254 test "creating a list", %{conn: conn} do
259 |> assign(:user, user)
260 |> post("/api/v1/lists", %{"title" => "cuties"})
262 assert %{"title" => title} = json_response(conn, 200)
263 assert title == "cuties"
266 test "adding users to a list", %{conn: conn} do
268 other_user = insert(:user)
269 {:ok, list} = Pleroma.List.create("name", user)
273 |> assign(:user, user)
274 |> post("/api/v1/lists/#{list.id}/accounts", %{"account_ids" => [other_user.id]})
276 assert %{} == json_response(conn, 200)
277 %Pleroma.List{following: following} = Pleroma.List.get(list.id, user)
278 assert following == [other_user.follower_address]
281 test "removing users from a list", %{conn: conn} do
283 other_user = insert(:user)
284 third_user = insert(:user)
285 {:ok, list} = Pleroma.List.create("name", user)
286 {:ok, list} = Pleroma.List.follow(list, other_user)
287 {:ok, list} = Pleroma.List.follow(list, third_user)
291 |> assign(:user, user)
292 |> delete("/api/v1/lists/#{list.id}/accounts", %{"account_ids" => [other_user.id]})
294 assert %{} == json_response(conn, 200)
295 %Pleroma.List{following: following} = Pleroma.List.get(list.id, user)
296 assert following == [third_user.follower_address]
299 test "listing users in a list", %{conn: conn} do
301 other_user = insert(:user)
302 {:ok, list} = Pleroma.List.create("name", user)
303 {:ok, list} = Pleroma.List.follow(list, other_user)
307 |> assign(:user, user)
308 |> get("/api/v1/lists/#{list.id}/accounts", %{"account_ids" => [other_user.id]})
310 assert [%{"id" => id}] = json_response(conn, 200)
311 assert id == to_string(other_user.id)
314 test "retrieving a list", %{conn: conn} do
316 {:ok, list} = Pleroma.List.create("name", user)
320 |> assign(:user, user)
321 |> get("/api/v1/lists/#{list.id}")
323 assert %{"id" => id} = json_response(conn, 200)
324 assert id == to_string(list.id)
327 test "renaming a list", %{conn: conn} do
329 {:ok, list} = Pleroma.List.create("name", user)
333 |> assign(:user, user)
334 |> put("/api/v1/lists/#{list.id}", %{"title" => "newname"})
336 assert %{"title" => name} = json_response(conn, 200)
337 assert name == "newname"
340 test "deleting a list", %{conn: conn} do
342 {:ok, list} = Pleroma.List.create("name", user)
346 |> assign(:user, user)
347 |> delete("/api/v1/lists/#{list.id}")
349 assert %{} = json_response(conn, 200)
350 assert is_nil(Repo.get(Pleroma.List, list.id))
353 test "list timeline", %{conn: conn} do
355 other_user = insert(:user)
356 {:ok, _activity_one} = TwitterAPI.create_status(user, %{"status" => "Marisa is cute."})
357 {:ok, activity_two} = TwitterAPI.create_status(other_user, %{"status" => "Marisa is cute."})
358 {:ok, list} = Pleroma.List.create("name", user)
359 {:ok, list} = Pleroma.List.follow(list, other_user)
363 |> assign(:user, user)
364 |> get("/api/v1/timelines/list/#{list.id}")
366 assert [%{"id" => id}] = json_response(conn, 200)
368 assert id == to_string(activity_two.id)
372 describe "notifications" do
373 test "list of notifications", %{conn: conn} do
375 other_user = insert(:user)
378 TwitterAPI.create_status(other_user, %{"status" => "hi @#{user.nickname}"})
380 {:ok, [_notification]} = Notification.create_notifications(activity)
384 |> assign(:user, user)
385 |> get("/api/v1/notifications")
388 "hi <span><a href=\"#{user.ap_id}\">@<span>#{user.nickname}</span></a></span>"
390 assert [%{"status" => %{"content" => response}} | _rest] = json_response(conn, 200)
391 assert response == expected_response
394 test "getting a single notification", %{conn: conn} do
396 other_user = insert(:user)
399 TwitterAPI.create_status(other_user, %{"status" => "hi @#{user.nickname}"})
401 {:ok, [notification]} = Notification.create_notifications(activity)
405 |> assign(:user, user)
406 |> get("/api/v1/notifications/#{notification.id}")
409 "hi <span><a href=\"#{user.ap_id}\">@<span>#{user.nickname}</span></a></span>"
411 assert %{"status" => %{"content" => response}} = json_response(conn, 200)
412 assert response == expected_response
415 test "dismissing a single notification", %{conn: conn} do
417 other_user = insert(:user)
420 TwitterAPI.create_status(other_user, %{"status" => "hi @#{user.nickname}"})
422 {:ok, [notification]} = Notification.create_notifications(activity)
426 |> assign(:user, user)
427 |> post("/api/v1/notifications/dismiss", %{"id" => notification.id})
429 assert %{} = json_response(conn, 200)
432 test "clearing all notifications", %{conn: conn} do
434 other_user = insert(:user)
437 TwitterAPI.create_status(other_user, %{"status" => "hi @#{user.nickname}"})
439 {:ok, [_notification]} = Notification.create_notifications(activity)
443 |> assign(:user, user)
444 |> post("/api/v1/notifications/clear")
446 assert %{} = json_response(conn, 200)
450 |> assign(:user, user)
451 |> get("/api/v1/notifications")
453 assert all = json_response(conn, 200)
458 describe "reblogging" do
459 test "reblogs and returns the reblogged status", %{conn: conn} do
460 activity = insert(:note_activity)
465 |> assign(:user, user)
466 |> post("/api/v1/statuses/#{activity.id}/reblog")
468 assert %{"reblog" => %{"id" => id, "reblogged" => true, "reblogs_count" => 1}} =
469 json_response(conn, 200)
471 assert to_string(activity.id) == id
475 describe "unreblogging" do
476 test "unreblogs and returns the unreblogged status", %{conn: conn} do
477 activity = insert(:note_activity)
480 {:ok, _, _} = CommonAPI.repeat(activity.id, user)
484 |> assign(:user, user)
485 |> post("/api/v1/statuses/#{activity.id}/unreblog")
487 assert %{"id" => id, "reblogged" => false, "reblogs_count" => 0} = json_response(conn, 200)
489 assert to_string(activity.id) == id
493 describe "favoriting" do
494 test "favs a status and returns it", %{conn: conn} do
495 activity = insert(:note_activity)
500 |> assign(:user, user)
501 |> post("/api/v1/statuses/#{activity.id}/favourite")
503 assert %{"id" => id, "favourites_count" => 1, "favourited" => true} =
504 json_response(conn, 200)
506 assert to_string(activity.id) == id
510 describe "unfavoriting" do
511 test "unfavorites a status and returns it", %{conn: conn} do
512 activity = insert(:note_activity)
515 {:ok, _, _} = CommonAPI.favorite(activity.id, user)
519 |> assign(:user, user)
520 |> post("/api/v1/statuses/#{activity.id}/unfavourite")
522 assert %{"id" => id, "favourites_count" => 0, "favourited" => false} =
523 json_response(conn, 200)
525 assert to_string(activity.id) == id
529 describe "user timelines" do
530 test "gets a users statuses", %{conn: conn} do
531 user_one = insert(:user)
532 user_two = insert(:user)
533 user_three = insert(:user)
535 {:ok, user_three} = User.follow(user_three, user_one)
537 {:ok, activity} = CommonAPI.post(user_one, %{"status" => "HI!!!"})
539 {:ok, direct_activity} =
540 CommonAPI.post(user_one, %{
541 "status" => "Hi, @#{user_two.nickname}.",
542 "visibility" => "direct"
545 {:ok, private_activity} =
546 CommonAPI.post(user_one, %{"status" => "private", "visibility" => "private"})
550 |> get("/api/v1/accounts/#{user_one.id}/statuses")
552 assert [%{"id" => id}] = json_response(resp, 200)
553 assert id == to_string(activity.id)
557 |> assign(:user, user_two)
558 |> get("/api/v1/accounts/#{user_one.id}/statuses")
560 assert [%{"id" => id_one}, %{"id" => id_two}] = json_response(resp, 200)
561 assert id_one == to_string(direct_activity.id)
562 assert id_two == to_string(activity.id)
566 |> assign(:user, user_three)
567 |> get("/api/v1/accounts/#{user_one.id}/statuses")
569 assert [%{"id" => id_one}, %{"id" => id_two}] = json_response(resp, 200)
570 assert id_one == to_string(private_activity.id)
571 assert id_two == to_string(activity.id)
574 test "unimplemented pinned statuses feature", %{conn: conn} do
575 note = insert(:note_activity)
576 user = User.get_by_ap_id(note.data["actor"])
580 |> get("/api/v1/accounts/#{user.id}/statuses?pinned=true")
582 assert json_response(conn, 200) == []
585 test "gets an users media", %{conn: conn} do
586 note = insert(:note_activity)
587 user = User.get_by_ap_id(note.data["actor"])
590 content_type: "image/jpg",
591 path: Path.absname("test/fixtures/image.jpg"),
592 filename: "an_image.jpg"
596 TwitterAPI.upload(file, "json")
600 TwitterAPI.create_status(user, %{"status" => "cofe", "media_ids" => [media["media_id"]]})
604 |> get("/api/v1/accounts/#{user.id}/statuses", %{"only_media" => "true"})
606 assert [%{"id" => id}] = json_response(conn, 200)
607 assert id == to_string(image_post.id)
611 |> get("/api/v1/accounts/#{user.id}/statuses", %{"only_media" => "1"})
613 assert [%{"id" => id}] = json_response(conn, 200)
614 assert id == to_string(image_post.id)
618 describe "user relationships" do
619 test "returns the relationships for the current user", %{conn: conn} do
621 other_user = insert(:user)
622 {:ok, user} = User.follow(user, other_user)
626 |> assign(:user, user)
627 |> get("/api/v1/accounts/relationships", %{"id" => [other_user.id]})
629 assert [relationship] = json_response(conn, 200)
631 assert to_string(other_user.id) == relationship["id"]
635 test "account fetching", %{conn: conn} do
640 |> get("/api/v1/accounts/#{user.id}")
642 assert %{"id" => id} = json_response(conn, 200)
643 assert id == to_string(user.id)
647 |> get("/api/v1/accounts/-1")
649 assert %{"error" => "Can't find user"} = json_response(conn, 404)
652 test "media upload", %{conn: conn} do
654 content_type: "image/jpg",
655 path: Path.absname("test/fixtures/image.jpg"),
656 filename: "an_image.jpg"
663 |> assign(:user, user)
664 |> post("/api/v1/media", %{"file" => file})
666 assert media = json_response(conn, 200)
668 assert media["type"] == "image"
671 test "hashtag timeline", %{conn: conn} do
672 following = insert(:user)
675 {:ok, activity} = TwitterAPI.create_status(following, %{"status" => "test #2hu"})
678 OStatus.fetch_activity_from_url("https://shitposter.club/notice/2827873")
682 |> get("/api/v1/timelines/tag/2hu")
684 assert [%{"id" => id}] = json_response(conn, 200)
686 assert id == to_string(activity.id)
690 test "getting followers", %{conn: conn} do
692 other_user = insert(:user)
693 {:ok, user} = User.follow(user, other_user)
697 |> get("/api/v1/accounts/#{other_user.id}/followers")
699 assert [%{"id" => id}] = json_response(conn, 200)
700 assert id == to_string(user.id)
703 test "getting following", %{conn: conn} do
705 other_user = insert(:user)
706 {:ok, user} = User.follow(user, other_user)
710 |> get("/api/v1/accounts/#{user.id}/following")
712 assert [%{"id" => id}] = json_response(conn, 200)
713 assert id == to_string(other_user.id)
716 test "following / unfollowing a user", %{conn: conn} do
718 other_user = insert(:user)
722 |> assign(:user, user)
723 |> post("/api/v1/accounts/#{other_user.id}/follow")
725 assert %{"id" => _id, "following" => true} = json_response(conn, 200)
727 user = Repo.get(User, user.id)
731 |> assign(:user, user)
732 |> post("/api/v1/accounts/#{other_user.id}/unfollow")
734 assert %{"id" => _id, "following" => false} = json_response(conn, 200)
736 user = Repo.get(User, user.id)
740 |> assign(:user, user)
741 |> post("/api/v1/follows", %{"uri" => other_user.nickname})
743 assert %{"id" => id} = json_response(conn, 200)
744 assert id == to_string(other_user.id)
747 test "blocking / unblocking a user", %{conn: conn} do
749 other_user = insert(:user)
753 |> assign(:user, user)
754 |> post("/api/v1/accounts/#{other_user.id}/block")
756 assert %{"id" => _id, "blocking" => true} = json_response(conn, 200)
758 user = Repo.get(User, user.id)
762 |> assign(:user, user)
763 |> post("/api/v1/accounts/#{other_user.id}/unblock")
765 assert %{"id" => _id, "blocking" => false} = json_response(conn, 200)
768 test "getting a list of blocks", %{conn: conn} do
770 other_user = insert(:user)
772 {:ok, user} = User.block(user, other_user)
776 |> assign(:user, user)
777 |> get("/api/v1/blocks")
779 other_user_id = to_string(other_user.id)
780 assert [%{"id" => ^other_user_id}] = json_response(conn, 200)
783 test "unimplemented mute endpoints" do
785 other_user = insert(:user)
788 |> Enum.each(fn endpoint ->
791 |> assign(:user, user)
792 |> post("/api/v1/accounts/#{other_user.id}/#{endpoint}")
794 assert %{"id" => id} = json_response(conn, 200)
795 assert id == to_string(other_user.id)
799 test "unimplemented mutes, follow_requests, blocks, domain blocks" do
802 ["blocks", "domain_blocks", "mutes", "follow_requests"]
803 |> Enum.each(fn endpoint ->
806 |> assign(:user, user)
807 |> get("/api/v1/#{endpoint}")
809 assert [] = json_response(conn, 200)
813 test "account search", %{conn: conn} do
815 user_two = insert(:user, %{nickname: "shp@shitposter.club"})
816 user_three = insert(:user, %{nickname: "shp@heldscal.la", name: "I love 2hu"})
820 |> assign(:user, user)
821 |> get("/api/v1/accounts/search", %{"q" => "shp"})
822 |> json_response(200)
824 result_ids = for result <- results, do: result["acct"]
826 assert user_two.nickname in result_ids
827 assert user_three.nickname in result_ids
831 |> assign(:user, user)
832 |> get("/api/v1/accounts/search", %{"q" => "2hu"})
833 |> json_response(200)
835 result_ids = for result <- results, do: result["acct"]
837 assert user_three.nickname in result_ids
840 test "search", %{conn: conn} do
842 user_two = insert(:user, %{nickname: "shp@shitposter.club"})
843 user_three = insert(:user, %{nickname: "shp@heldscal.la", name: "I love 2hu"})
845 {:ok, activity} = CommonAPI.post(user, %{"status" => "This is about 2hu"})
848 CommonAPI.post(user, %{
849 "status" => "This is about 2hu, but private",
850 "visibility" => "private"
853 {:ok, _} = CommonAPI.post(user_two, %{"status" => "This isn't"})
857 |> get("/api/v1/search", %{"q" => "2hu"})
859 assert results = json_response(conn, 200)
861 [account | _] = results["accounts"]
862 assert account["id"] == to_string(user_three.id)
864 assert results["hashtags"] == []
866 [status] = results["statuses"]
867 assert status["id"] == to_string(activity.id)
870 test "search fetches remote statuses", %{conn: conn} do
874 |> get("/api/v1/search", %{"q" => "https://shitposter.club/notice/2827873"})
876 assert results = json_response(conn, 200)
878 [status] = results["statuses"]
879 assert status["uri"] == "tag:shitposter.club,2017-05-05:noticeId=2827873:objectType=comment"
883 test "search fetches remote accounts", %{conn: conn} do
886 |> get("/api/v1/search", %{"q" => "shp@social.heldscal.la", "resolve" => "true"})
888 assert results = json_response(conn, 200)
889 [account] = results["accounts"]
890 assert account["acct"] == "shp@social.heldscal.la"
893 test "returns the favorites of a user", %{conn: conn} do
895 other_user = insert(:user)
897 {:ok, _} = CommonAPI.post(other_user, %{"status" => "bla"})
898 {:ok, activity} = CommonAPI.post(other_user, %{"status" => "traps are happy"})
900 {:ok, _, _} = CommonAPI.favorite(activity.id, user)
904 |> assign(:user, user)
905 |> get("/api/v1/favourites")
907 assert [status] = json_response(conn, 200)
908 assert status["id"] == to_string(activity.id)
911 describe "updating credentials" do
912 test "updates the user's bio", %{conn: conn} do
917 |> assign(:user, user)
918 |> patch("/api/v1/accounts/update_credentials", %{"note" => "I drink #cofe"})
920 assert user = json_response(conn, 200)
921 assert user["note"] == "I drink #cofe"
924 test "updates the user's name", %{conn: conn} do
929 |> assign(:user, user)
930 |> patch("/api/v1/accounts/update_credentials", %{"display_name" => "markorepairs"})
932 assert user = json_response(conn, 200)
933 assert user["display_name"] == "markorepairs"
936 test "updates the user's avatar", %{conn: conn} do
939 new_avatar = %Plug.Upload{
940 content_type: "image/jpg",
941 path: Path.absname("test/fixtures/image.jpg"),
942 filename: "an_image.jpg"
947 |> assign(:user, user)
948 |> patch("/api/v1/accounts/update_credentials", %{"avatar" => new_avatar})
950 assert user = json_response(conn, 200)
951 assert user["avatar"] != "https://placehold.it/48x48"
954 test "updates the user's banner", %{conn: conn} do
957 new_header = %Plug.Upload{
958 content_type: "image/jpg",
959 path: Path.absname("test/fixtures/image.jpg"),
960 filename: "an_image.jpg"
965 |> assign(:user, user)
966 |> patch("/api/v1/accounts/update_credentials", %{"header" => new_header})
968 assert user = json_response(conn, 200)
969 assert user["header"] != "https://placehold.it/700x335"
973 test "get instance information", %{conn: conn} do
974 insert(:user, %{local: true})
975 user = insert(:user, %{local: true})
976 insert(:user, %{local: false})
978 {:ok, _} = TwitterAPI.create_status(user, %{"status" => "cofe"})
980 Pleroma.Stats.update_stats()
984 |> get("/api/v1/instance")
986 assert result = json_response(conn, 200)
988 assert result["stats"]["user_count"] == 2
989 assert result["stats"]["status_count"] == 1