1 # Pleroma: A lightweight social networking server
2 # Copyright © 2017-2019 Pleroma Authors <https://pleroma.social/>
3 # SPDX-License-Identifier: AGPL-3.0-only
5 defmodule Pleroma.Web.AdminAPI.AdminAPIControllerTest do
6 use Pleroma.Web.ConnCase
7 use Oban.Testing, repo: Pleroma.Repo
11 alias Pleroma.ModerationLog
13 alias Pleroma.ReportNote
14 alias Pleroma.Tests.ObanHelpers
16 alias Pleroma.UserInviteToken
17 alias Pleroma.Web.ActivityPub.Relay
18 alias Pleroma.Web.CommonAPI
19 alias Pleroma.Web.MastodonAPI.StatusView
20 alias Pleroma.Web.MediaProxy
21 import Pleroma.Factory
24 Tesla.Mock.mock_global(fn env -> apply(HttpRequestMock, :request, [env]) end)
29 clear_config([:auth, :enforce_oauth_admin_scope_usage]) do
30 Pleroma.Config.put([:auth, :enforce_oauth_admin_scope_usage], false)
33 describe "with [:auth, :enforce_oauth_admin_scope_usage]," do
34 clear_config([:auth, :enforce_oauth_admin_scope_usage]) do
35 Pleroma.Config.put([:auth, :enforce_oauth_admin_scope_usage], true)
38 test "GET /api/pleroma/admin/users/:nickname requires admin:read:accounts or broader scope" do
40 admin = insert(:user, is_admin: true)
41 url = "/api/pleroma/admin/users/#{user.nickname}"
43 good_token1 = insert(:oauth_token, user: admin, scopes: ["admin"])
44 good_token2 = insert(:oauth_token, user: admin, scopes: ["admin:read"])
45 good_token3 = insert(:oauth_token, user: admin, scopes: ["admin:read:accounts"])
47 bad_token1 = insert(:oauth_token, user: admin, scopes: ["read:accounts"])
48 bad_token2 = insert(:oauth_token, user: admin, scopes: ["admin:read:accounts:partial"])
51 for good_token <- [good_token1, good_token2, good_token3] do
54 |> assign(:user, admin)
55 |> assign(:token, good_token)
58 assert json_response(conn, 200)
61 for good_token <- [good_token1, good_token2, good_token3] do
65 |> assign(:token, good_token)
68 assert json_response(conn, :forbidden)
71 for bad_token <- [bad_token1, bad_token2, bad_token3] do
74 |> assign(:user, admin)
75 |> assign(:token, bad_token)
78 assert json_response(conn, :forbidden)
83 describe "DELETE /api/pleroma/admin/users" do
85 admin = insert(:user, is_admin: true)
90 |> assign(:user, admin)
91 |> put_req_header("accept", "application/json")
92 |> delete("/api/pleroma/admin/users?nickname=#{user.nickname}")
94 log_entry = Repo.one(ModerationLog)
96 assert ModerationLog.get_log_entry_message(log_entry) ==
97 "@#{admin.nickname} deleted users: @#{user.nickname}"
99 assert json_response(conn, 200) == user.nickname
102 test "multiple users" do
103 admin = insert(:user, is_admin: true)
104 user_one = insert(:user)
105 user_two = insert(:user)
109 |> assign(:user, admin)
110 |> put_req_header("accept", "application/json")
111 |> delete("/api/pleroma/admin/users", %{
112 nicknames: [user_one.nickname, user_two.nickname]
115 log_entry = Repo.one(ModerationLog)
117 assert ModerationLog.get_log_entry_message(log_entry) ==
118 "@#{admin.nickname} deleted users: @#{user_one.nickname}, @#{user_two.nickname}"
120 response = json_response(conn, 200)
121 assert response -- [user_one.nickname, user_two.nickname] == []
125 describe "/api/pleroma/admin/users" do
127 admin = insert(:user, is_admin: true)
131 |> assign(:user, admin)
132 |> put_req_header("accept", "application/json")
133 |> post("/api/pleroma/admin/users", %{
136 "nickname" => "lain",
137 "email" => "lain@example.org",
141 "nickname" => "lain2",
142 "email" => "lain2@example.org",
148 response = json_response(conn, 200) |> Enum.map(&Map.get(&1, "type"))
149 assert response == ["success", "success"]
151 log_entry = Repo.one(ModerationLog)
153 assert ["lain", "lain2"] -- Enum.map(log_entry.data["subjects"], & &1["nickname"]) == []
156 test "Cannot create user with existing email" do
157 admin = insert(:user, is_admin: true)
162 |> assign(:user, admin)
163 |> put_req_header("accept", "application/json")
164 |> post("/api/pleroma/admin/users", %{
167 "nickname" => "lain",
168 "email" => user.email,
174 assert json_response(conn, 409) == [
178 "email" => user.email,
181 "error" => "email has already been taken",
187 test "Cannot create user with existing nickname" do
188 admin = insert(:user, is_admin: true)
193 |> assign(:user, admin)
194 |> put_req_header("accept", "application/json")
195 |> post("/api/pleroma/admin/users", %{
198 "nickname" => user.nickname,
199 "email" => "someuser@plerama.social",
205 assert json_response(conn, 409) == [
209 "email" => "someuser@plerama.social",
210 "nickname" => user.nickname
212 "error" => "nickname has already been taken",
218 test "Multiple user creation works in transaction" do
219 admin = insert(:user, is_admin: true)
224 |> assign(:user, admin)
225 |> put_req_header("accept", "application/json")
226 |> post("/api/pleroma/admin/users", %{
229 "nickname" => "newuser",
230 "email" => "newuser@pleroma.social",
234 "nickname" => "lain",
235 "email" => user.email,
241 assert json_response(conn, 409) == [
245 "email" => user.email,
248 "error" => "email has already been taken",
254 "email" => "newuser@pleroma.social",
255 "nickname" => "newuser"
262 assert User.get_by_nickname("newuser") === nil
266 describe "/api/pleroma/admin/users/:nickname" do
267 test "Show", %{conn: conn} do
268 admin = insert(:user, is_admin: true)
273 |> assign(:user, admin)
274 |> get("/api/pleroma/admin/users/#{user.nickname}")
277 "deactivated" => false,
278 "id" => to_string(user.id),
280 "nickname" => user.nickname,
281 "roles" => %{"admin" => false, "moderator" => false},
283 "avatar" => User.avatar_url(user) |> MediaProxy.url(),
284 "display_name" => HTML.strip_tags(user.name || user.nickname),
285 "confirmation_pending" => false
288 assert expected == json_response(conn, 200)
291 test "when the user doesn't exist", %{conn: conn} do
292 admin = insert(:user, is_admin: true)
297 |> assign(:user, admin)
298 |> get("/api/pleroma/admin/users/#{user.nickname}")
300 assert "Not found" == json_response(conn, 404)
304 describe "/api/pleroma/admin/users/follow" do
305 test "allows to force-follow another user" do
306 admin = insert(:user, is_admin: true)
308 follower = insert(:user)
311 |> assign(:user, admin)
312 |> put_req_header("accept", "application/json")
313 |> post("/api/pleroma/admin/users/follow", %{
314 "follower" => follower.nickname,
315 "followed" => user.nickname
318 user = User.get_cached_by_id(user.id)
319 follower = User.get_cached_by_id(follower.id)
321 assert User.following?(follower, user)
323 log_entry = Repo.one(ModerationLog)
325 assert ModerationLog.get_log_entry_message(log_entry) ==
326 "@#{admin.nickname} made @#{follower.nickname} follow @#{user.nickname}"
330 describe "/api/pleroma/admin/users/unfollow" do
331 test "allows to force-unfollow another user" do
332 admin = insert(:user, is_admin: true)
334 follower = insert(:user)
336 User.follow(follower, user)
339 |> assign(:user, admin)
340 |> put_req_header("accept", "application/json")
341 |> post("/api/pleroma/admin/users/unfollow", %{
342 "follower" => follower.nickname,
343 "followed" => user.nickname
346 user = User.get_cached_by_id(user.id)
347 follower = User.get_cached_by_id(follower.id)
349 refute User.following?(follower, user)
351 log_entry = Repo.one(ModerationLog)
353 assert ModerationLog.get_log_entry_message(log_entry) ==
354 "@#{admin.nickname} made @#{follower.nickname} unfollow @#{user.nickname}"
358 describe "PUT /api/pleroma/admin/users/tag" do
360 admin = insert(:user, is_admin: true)
361 user1 = insert(:user, %{tags: ["x"]})
362 user2 = insert(:user, %{tags: ["y"]})
363 user3 = insert(:user, %{tags: ["unchanged"]})
367 |> assign(:user, admin)
368 |> put_req_header("accept", "application/json")
370 "/api/pleroma/admin/users/tag?nicknames[]=#{user1.nickname}&nicknames[]=#{
372 }&tags[]=foo&tags[]=bar"
375 %{conn: conn, admin: admin, user1: user1, user2: user2, user3: user3}
378 test "it appends specified tags to users with specified nicknames", %{
384 assert json_response(conn, :no_content)
385 assert User.get_cached_by_id(user1.id).tags == ["x", "foo", "bar"]
386 assert User.get_cached_by_id(user2.id).tags == ["y", "foo", "bar"]
388 log_entry = Repo.one(ModerationLog)
391 [user1.nickname, user2.nickname]
392 |> Enum.map(&"@#{&1}")
395 tags = ["foo", "bar"] |> Enum.join(", ")
397 assert ModerationLog.get_log_entry_message(log_entry) ==
398 "@#{admin.nickname} added tags: #{tags} to users: #{users}"
401 test "it does not modify tags of not specified users", %{conn: conn, user3: user3} do
402 assert json_response(conn, :no_content)
403 assert User.get_cached_by_id(user3.id).tags == ["unchanged"]
407 describe "DELETE /api/pleroma/admin/users/tag" do
409 admin = insert(:user, is_admin: true)
410 user1 = insert(:user, %{tags: ["x"]})
411 user2 = insert(:user, %{tags: ["y", "z"]})
412 user3 = insert(:user, %{tags: ["unchanged"]})
416 |> assign(:user, admin)
417 |> put_req_header("accept", "application/json")
419 "/api/pleroma/admin/users/tag?nicknames[]=#{user1.nickname}&nicknames[]=#{
424 %{conn: conn, admin: admin, user1: user1, user2: user2, user3: user3}
427 test "it removes specified tags from users with specified nicknames", %{
433 assert json_response(conn, :no_content)
434 assert User.get_cached_by_id(user1.id).tags == []
435 assert User.get_cached_by_id(user2.id).tags == ["y"]
437 log_entry = Repo.one(ModerationLog)
440 [user1.nickname, user2.nickname]
441 |> Enum.map(&"@#{&1}")
444 tags = ["x", "z"] |> Enum.join(", ")
446 assert ModerationLog.get_log_entry_message(log_entry) ==
447 "@#{admin.nickname} removed tags: #{tags} from users: #{users}"
450 test "it does not modify tags of not specified users", %{conn: conn, user3: user3} do
451 assert json_response(conn, :no_content)
452 assert User.get_cached_by_id(user3.id).tags == ["unchanged"]
456 describe "/api/pleroma/admin/users/:nickname/permission_group" do
457 test "GET is giving user_info" do
458 admin = insert(:user, is_admin: true)
462 |> assign(:user, admin)
463 |> put_req_header("accept", "application/json")
464 |> get("/api/pleroma/admin/users/#{admin.nickname}/permission_group/")
466 assert json_response(conn, 200) == %{
468 "is_moderator" => false
472 test "/:right POST, can add to a permission group" do
473 admin = insert(:user, is_admin: true)
478 |> assign(:user, admin)
479 |> put_req_header("accept", "application/json")
480 |> post("/api/pleroma/admin/users/#{user.nickname}/permission_group/admin")
482 assert json_response(conn, 200) == %{
486 log_entry = Repo.one(ModerationLog)
488 assert ModerationLog.get_log_entry_message(log_entry) ==
489 "@#{admin.nickname} made @#{user.nickname} admin"
492 test "/:right POST, can add to a permission group (multiple)" do
493 admin = insert(:user, is_admin: true)
494 user_one = insert(:user)
495 user_two = insert(:user)
499 |> assign(:user, admin)
500 |> put_req_header("accept", "application/json")
501 |> post("/api/pleroma/admin/users/permission_group/admin", %{
502 nicknames: [user_one.nickname, user_two.nickname]
505 assert json_response(conn, 200) == %{
509 log_entry = Repo.one(ModerationLog)
511 assert ModerationLog.get_log_entry_message(log_entry) ==
512 "@#{admin.nickname} made @#{user_one.nickname}, @#{user_two.nickname} admin"
515 test "/:right DELETE, can remove from a permission group" do
516 admin = insert(:user, is_admin: true)
517 user = insert(:user, is_admin: true)
521 |> assign(:user, admin)
522 |> put_req_header("accept", "application/json")
523 |> delete("/api/pleroma/admin/users/#{user.nickname}/permission_group/admin")
525 assert json_response(conn, 200) == %{
529 log_entry = Repo.one(ModerationLog)
531 assert ModerationLog.get_log_entry_message(log_entry) ==
532 "@#{admin.nickname} revoked admin role from @#{user.nickname}"
535 test "/:right DELETE, can remove from a permission group (multiple)" do
536 admin = insert(:user, is_admin: true)
537 user_one = insert(:user, is_admin: true)
538 user_two = insert(:user, is_admin: true)
542 |> assign(:user, admin)
543 |> put_req_header("accept", "application/json")
544 |> delete("/api/pleroma/admin/users/permission_group/admin", %{
545 nicknames: [user_one.nickname, user_two.nickname]
548 assert json_response(conn, 200) == %{
552 log_entry = Repo.one(ModerationLog)
554 assert ModerationLog.get_log_entry_message(log_entry) ==
555 "@#{admin.nickname} revoked admin role from @#{user_one.nickname}, @#{
561 describe "POST /api/pleroma/admin/email_invite, with valid config" do
563 [user: insert(:user, is_admin: true)]
566 clear_config([:instance, :registrations_open]) do
567 Pleroma.Config.put([:instance, :registrations_open], false)
570 clear_config([:instance, :invites_enabled]) do
571 Pleroma.Config.put([:instance, :invites_enabled], true)
574 test "sends invitation and returns 204", %{conn: conn, user: user} do
575 recipient_email = "foo@bar.com"
576 recipient_name = "J. D."
580 |> assign(:user, user)
582 "/api/pleroma/admin/users/email_invite?email=#{recipient_email}&name=#{recipient_name}"
585 assert json_response(conn, :no_content)
587 token_record = List.last(Pleroma.Repo.all(Pleroma.UserInviteToken))
589 refute token_record.used
591 notify_email = Pleroma.Config.get([:instance, :notify_email])
592 instance_name = Pleroma.Config.get([:instance, :name])
595 Pleroma.Emails.UserEmail.user_invitation_email(
602 Swoosh.TestAssertions.assert_email_sent(
603 from: {instance_name, notify_email},
604 to: {recipient_name, recipient_email},
605 html_body: email.html_body
609 test "it returns 403 if requested by a non-admin", %{conn: conn} do
610 non_admin_user = insert(:user)
614 |> assign(:user, non_admin_user)
615 |> post("/api/pleroma/admin/users/email_invite?email=foo@bar.com&name=JD")
617 assert json_response(conn, :forbidden)
621 describe "POST /api/pleroma/admin/users/email_invite, with invalid config" do
623 [user: insert(:user, is_admin: true)]
626 clear_config([:instance, :registrations_open])
627 clear_config([:instance, :invites_enabled])
629 test "it returns 500 if `invites_enabled` is not enabled", %{conn: conn, user: user} do
630 Pleroma.Config.put([:instance, :registrations_open], false)
631 Pleroma.Config.put([:instance, :invites_enabled], false)
635 |> assign(:user, user)
636 |> post("/api/pleroma/admin/users/email_invite?email=foo@bar.com&name=JD")
638 assert json_response(conn, :internal_server_error)
641 test "it returns 500 if `registrations_open` is enabled", %{conn: conn, user: user} do
642 Pleroma.Config.put([:instance, :registrations_open], true)
643 Pleroma.Config.put([:instance, :invites_enabled], true)
647 |> assign(:user, user)
648 |> post("/api/pleroma/admin/users/email_invite?email=foo@bar.com&name=JD")
650 assert json_response(conn, :internal_server_error)
654 test "/api/pleroma/admin/users/:nickname/password_reset" do
655 admin = insert(:user, is_admin: true)
660 |> assign(:user, admin)
661 |> put_req_header("accept", "application/json")
662 |> get("/api/pleroma/admin/users/#{user.nickname}/password_reset")
664 resp = json_response(conn, 200)
666 assert Regex.match?(~r/(http:\/\/|https:\/\/)/, resp["link"])
669 describe "GET /api/pleroma/admin/users" do
671 admin = insert(:user, is_admin: true)
675 |> assign(:user, admin)
677 {:ok, conn: conn, admin: admin}
680 test "renders users array for the first page", %{conn: conn, admin: admin} do
681 user = insert(:user, local: false, tags: ["foo", "bar"])
682 conn = get(conn, "/api/pleroma/admin/users?page=1")
687 "deactivated" => admin.deactivated,
689 "nickname" => admin.nickname,
690 "roles" => %{"admin" => true, "moderator" => false},
693 "avatar" => User.avatar_url(admin) |> MediaProxy.url(),
694 "display_name" => HTML.strip_tags(admin.name || admin.nickname),
695 "confirmation_pending" => false
698 "deactivated" => user.deactivated,
700 "nickname" => user.nickname,
701 "roles" => %{"admin" => false, "moderator" => false},
703 "tags" => ["foo", "bar"],
704 "avatar" => User.avatar_url(user) |> MediaProxy.url(),
705 "display_name" => HTML.strip_tags(user.name || user.nickname),
706 "confirmation_pending" => false
709 |> Enum.sort_by(& &1["nickname"])
711 assert json_response(conn, 200) == %{
718 test "renders empty array for the second page", %{conn: conn} do
721 conn = get(conn, "/api/pleroma/admin/users?page=2")
723 assert json_response(conn, 200) == %{
730 test "regular search", %{conn: conn} do
731 user = insert(:user, nickname: "bob")
733 conn = get(conn, "/api/pleroma/admin/users?query=bo")
735 assert json_response(conn, 200) == %{
740 "deactivated" => user.deactivated,
742 "nickname" => user.nickname,
743 "roles" => %{"admin" => false, "moderator" => false},
746 "avatar" => User.avatar_url(user) |> MediaProxy.url(),
747 "display_name" => HTML.strip_tags(user.name || user.nickname),
748 "confirmation_pending" => false
754 test "search by domain", %{conn: conn} do
755 user = insert(:user, nickname: "nickname@domain.com")
758 conn = get(conn, "/api/pleroma/admin/users?query=domain.com")
760 assert json_response(conn, 200) == %{
765 "deactivated" => user.deactivated,
767 "nickname" => user.nickname,
768 "roles" => %{"admin" => false, "moderator" => false},
771 "avatar" => User.avatar_url(user) |> MediaProxy.url(),
772 "display_name" => HTML.strip_tags(user.name || user.nickname),
773 "confirmation_pending" => false
779 test "search by full nickname", %{conn: conn} do
780 user = insert(:user, nickname: "nickname@domain.com")
783 conn = get(conn, "/api/pleroma/admin/users?query=nickname@domain.com")
785 assert json_response(conn, 200) == %{
790 "deactivated" => user.deactivated,
792 "nickname" => user.nickname,
793 "roles" => %{"admin" => false, "moderator" => false},
796 "avatar" => User.avatar_url(user) |> MediaProxy.url(),
797 "display_name" => HTML.strip_tags(user.name || user.nickname),
798 "confirmation_pending" => false
804 test "search by display name", %{conn: conn} do
805 user = insert(:user, name: "Display name")
808 conn = get(conn, "/api/pleroma/admin/users?name=display")
810 assert json_response(conn, 200) == %{
815 "deactivated" => user.deactivated,
817 "nickname" => user.nickname,
818 "roles" => %{"admin" => false, "moderator" => false},
821 "avatar" => User.avatar_url(user) |> MediaProxy.url(),
822 "display_name" => HTML.strip_tags(user.name || user.nickname),
823 "confirmation_pending" => false
829 test "search by email", %{conn: conn} do
830 user = insert(:user, email: "email@example.com")
833 conn = get(conn, "/api/pleroma/admin/users?email=email@example.com")
835 assert json_response(conn, 200) == %{
840 "deactivated" => user.deactivated,
842 "nickname" => user.nickname,
843 "roles" => %{"admin" => false, "moderator" => false},
846 "avatar" => User.avatar_url(user) |> MediaProxy.url(),
847 "display_name" => HTML.strip_tags(user.name || user.nickname),
848 "confirmation_pending" => false
854 test "regular search with page size", %{conn: conn} do
855 user = insert(:user, nickname: "aalice")
856 user2 = insert(:user, nickname: "alice")
858 conn1 = get(conn, "/api/pleroma/admin/users?query=a&page_size=1&page=1")
860 assert json_response(conn1, 200) == %{
865 "deactivated" => user.deactivated,
867 "nickname" => user.nickname,
868 "roles" => %{"admin" => false, "moderator" => false},
871 "avatar" => User.avatar_url(user) |> MediaProxy.url(),
872 "display_name" => HTML.strip_tags(user.name || user.nickname),
873 "confirmation_pending" => false
878 conn2 = get(conn, "/api/pleroma/admin/users?query=a&page_size=1&page=2")
880 assert json_response(conn2, 200) == %{
885 "deactivated" => user2.deactivated,
887 "nickname" => user2.nickname,
888 "roles" => %{"admin" => false, "moderator" => false},
891 "avatar" => User.avatar_url(user2) |> MediaProxy.url(),
892 "display_name" => HTML.strip_tags(user2.name || user2.nickname),
893 "confirmation_pending" => false
899 test "only local users" do
900 admin = insert(:user, is_admin: true, nickname: "john")
901 user = insert(:user, nickname: "bob")
903 insert(:user, nickname: "bobb", local: false)
907 |> assign(:user, admin)
908 |> get("/api/pleroma/admin/users?query=bo&filters=local")
910 assert json_response(conn, 200) == %{
915 "deactivated" => user.deactivated,
917 "nickname" => user.nickname,
918 "roles" => %{"admin" => false, "moderator" => false},
921 "avatar" => User.avatar_url(user) |> MediaProxy.url(),
922 "display_name" => HTML.strip_tags(user.name || user.nickname),
923 "confirmation_pending" => false
929 test "only local users with no query", %{admin: old_admin} do
930 admin = insert(:user, is_admin: true, nickname: "john")
931 user = insert(:user, nickname: "bob")
933 insert(:user, nickname: "bobb", local: false)
937 |> assign(:user, admin)
938 |> get("/api/pleroma/admin/users?filters=local")
943 "deactivated" => user.deactivated,
945 "nickname" => user.nickname,
946 "roles" => %{"admin" => false, "moderator" => false},
949 "avatar" => User.avatar_url(user) |> MediaProxy.url(),
950 "display_name" => HTML.strip_tags(user.name || user.nickname),
951 "confirmation_pending" => false
954 "deactivated" => admin.deactivated,
956 "nickname" => admin.nickname,
957 "roles" => %{"admin" => true, "moderator" => false},
960 "avatar" => User.avatar_url(admin) |> MediaProxy.url(),
961 "display_name" => HTML.strip_tags(admin.name || admin.nickname),
962 "confirmation_pending" => false
965 "deactivated" => false,
966 "id" => old_admin.id,
968 "nickname" => old_admin.nickname,
969 "roles" => %{"admin" => true, "moderator" => false},
971 "avatar" => User.avatar_url(old_admin) |> MediaProxy.url(),
972 "display_name" => HTML.strip_tags(old_admin.name || old_admin.nickname),
973 "confirmation_pending" => false
976 |> Enum.sort_by(& &1["nickname"])
978 assert json_response(conn, 200) == %{
985 test "load only admins", %{conn: conn, admin: admin} do
986 second_admin = insert(:user, is_admin: true)
990 conn = get(conn, "/api/pleroma/admin/users?filters=is_admin")
995 "deactivated" => false,
997 "nickname" => admin.nickname,
998 "roles" => %{"admin" => true, "moderator" => false},
999 "local" => admin.local,
1001 "avatar" => User.avatar_url(admin) |> MediaProxy.url(),
1002 "display_name" => HTML.strip_tags(admin.name || admin.nickname),
1003 "confirmation_pending" => false
1006 "deactivated" => false,
1007 "id" => second_admin.id,
1008 "nickname" => second_admin.nickname,
1009 "roles" => %{"admin" => true, "moderator" => false},
1010 "local" => second_admin.local,
1012 "avatar" => User.avatar_url(second_admin) |> MediaProxy.url(),
1013 "display_name" => HTML.strip_tags(second_admin.name || second_admin.nickname),
1014 "confirmation_pending" => false
1017 |> Enum.sort_by(& &1["nickname"])
1019 assert json_response(conn, 200) == %{
1026 test "load only moderators", %{conn: conn} do
1027 moderator = insert(:user, is_moderator: true)
1031 conn = get(conn, "/api/pleroma/admin/users?filters=is_moderator")
1033 assert json_response(conn, 200) == %{
1038 "deactivated" => false,
1039 "id" => moderator.id,
1040 "nickname" => moderator.nickname,
1041 "roles" => %{"admin" => false, "moderator" => true},
1042 "local" => moderator.local,
1044 "avatar" => User.avatar_url(moderator) |> MediaProxy.url(),
1045 "display_name" => HTML.strip_tags(moderator.name || moderator.nickname),
1046 "confirmation_pending" => false
1052 test "load users with tags list", %{conn: conn} do
1053 user1 = insert(:user, tags: ["first"])
1054 user2 = insert(:user, tags: ["second"])
1058 conn = get(conn, "/api/pleroma/admin/users?tags[]=first&tags[]=second")
1063 "deactivated" => false,
1065 "nickname" => user1.nickname,
1066 "roles" => %{"admin" => false, "moderator" => false},
1067 "local" => user1.local,
1068 "tags" => ["first"],
1069 "avatar" => User.avatar_url(user1) |> MediaProxy.url(),
1070 "display_name" => HTML.strip_tags(user1.name || user1.nickname),
1071 "confirmation_pending" => false
1074 "deactivated" => false,
1076 "nickname" => user2.nickname,
1077 "roles" => %{"admin" => false, "moderator" => false},
1078 "local" => user2.local,
1079 "tags" => ["second"],
1080 "avatar" => User.avatar_url(user2) |> MediaProxy.url(),
1081 "display_name" => HTML.strip_tags(user2.name || user2.nickname),
1082 "confirmation_pending" => false
1085 |> Enum.sort_by(& &1["nickname"])
1087 assert json_response(conn, 200) == %{
1094 test "it works with multiple filters" do
1095 admin = insert(:user, nickname: "john", is_admin: true)
1096 user = insert(:user, nickname: "bob", local: false, deactivated: true)
1098 insert(:user, nickname: "ken", local: true, deactivated: true)
1099 insert(:user, nickname: "bobb", local: false, deactivated: false)
1103 |> assign(:user, admin)
1104 |> get("/api/pleroma/admin/users?filters=deactivated,external")
1106 assert json_response(conn, 200) == %{
1111 "deactivated" => user.deactivated,
1113 "nickname" => user.nickname,
1114 "roles" => %{"admin" => false, "moderator" => false},
1115 "local" => user.local,
1117 "avatar" => User.avatar_url(user) |> MediaProxy.url(),
1118 "display_name" => HTML.strip_tags(user.name || user.nickname),
1119 "confirmation_pending" => false
1125 test "it omits relay user", %{admin: admin} do
1126 assert %User{} = Relay.get_actor()
1130 |> assign(:user, admin)
1131 |> get("/api/pleroma/admin/users")
1133 assert json_response(conn, 200) == %{
1138 "deactivated" => admin.deactivated,
1140 "nickname" => admin.nickname,
1141 "roles" => %{"admin" => true, "moderator" => false},
1144 "avatar" => User.avatar_url(admin) |> MediaProxy.url(),
1145 "display_name" => HTML.strip_tags(admin.name || admin.nickname),
1146 "confirmation_pending" => false
1153 test "PATCH /api/pleroma/admin/users/activate" do
1154 admin = insert(:user, is_admin: true)
1155 user_one = insert(:user, deactivated: true)
1156 user_two = insert(:user, deactivated: true)
1160 |> assign(:user, admin)
1162 "/api/pleroma/admin/users/activate",
1163 %{nicknames: [user_one.nickname, user_two.nickname]}
1166 response = json_response(conn, 200)
1167 assert Enum.map(response["users"], & &1["deactivated"]) == [false, false]
1169 log_entry = Repo.one(ModerationLog)
1171 assert ModerationLog.get_log_entry_message(log_entry) ==
1172 "@#{admin.nickname} activated users: @#{user_one.nickname}, @#{user_two.nickname}"
1175 test "PATCH /api/pleroma/admin/users/deactivate" do
1176 admin = insert(:user, is_admin: true)
1177 user_one = insert(:user, deactivated: false)
1178 user_two = insert(:user, deactivated: false)
1182 |> assign(:user, admin)
1184 "/api/pleroma/admin/users/deactivate",
1185 %{nicknames: [user_one.nickname, user_two.nickname]}
1188 response = json_response(conn, 200)
1189 assert Enum.map(response["users"], & &1["deactivated"]) == [true, true]
1191 log_entry = Repo.one(ModerationLog)
1193 assert ModerationLog.get_log_entry_message(log_entry) ==
1194 "@#{admin.nickname} deactivated users: @#{user_one.nickname}, @#{user_two.nickname}"
1197 test "PATCH /api/pleroma/admin/users/:nickname/toggle_activation" do
1198 admin = insert(:user, is_admin: true)
1199 user = insert(:user)
1203 |> assign(:user, admin)
1204 |> patch("/api/pleroma/admin/users/#{user.nickname}/toggle_activation")
1206 assert json_response(conn, 200) ==
1208 "deactivated" => !user.deactivated,
1210 "nickname" => user.nickname,
1211 "roles" => %{"admin" => false, "moderator" => false},
1214 "avatar" => User.avatar_url(user) |> MediaProxy.url(),
1215 "display_name" => HTML.strip_tags(user.name || user.nickname),
1216 "confirmation_pending" => false
1219 log_entry = Repo.one(ModerationLog)
1221 assert ModerationLog.get_log_entry_message(log_entry) ==
1222 "@#{admin.nickname} deactivated users: @#{user.nickname}"
1225 describe "POST /api/pleroma/admin/users/invite_token" do
1227 admin = insert(:user, is_admin: true)
1231 |> assign(:user, admin)
1236 test "without options", %{conn: conn} do
1237 conn = post(conn, "/api/pleroma/admin/users/invite_token")
1239 invite_json = json_response(conn, 200)
1240 invite = UserInviteToken.find_by_token!(invite_json["token"])
1242 refute invite.expires_at
1243 refute invite.max_use
1244 assert invite.invite_type == "one_time"
1247 test "with expires_at", %{conn: conn} do
1249 post(conn, "/api/pleroma/admin/users/invite_token", %{
1250 "expires_at" => Date.to_string(Date.utc_today())
1253 invite_json = json_response(conn, 200)
1254 invite = UserInviteToken.find_by_token!(invite_json["token"])
1257 assert invite.expires_at == Date.utc_today()
1258 refute invite.max_use
1259 assert invite.invite_type == "date_limited"
1262 test "with max_use", %{conn: conn} do
1263 conn = post(conn, "/api/pleroma/admin/users/invite_token", %{"max_use" => 150})
1265 invite_json = json_response(conn, 200)
1266 invite = UserInviteToken.find_by_token!(invite_json["token"])
1268 refute invite.expires_at
1269 assert invite.max_use == 150
1270 assert invite.invite_type == "reusable"
1273 test "with max use and expires_at", %{conn: conn} do
1275 post(conn, "/api/pleroma/admin/users/invite_token", %{
1277 "expires_at" => Date.to_string(Date.utc_today())
1280 invite_json = json_response(conn, 200)
1281 invite = UserInviteToken.find_by_token!(invite_json["token"])
1283 assert invite.expires_at == Date.utc_today()
1284 assert invite.max_use == 150
1285 assert invite.invite_type == "reusable_date_limited"
1289 describe "GET /api/pleroma/admin/users/invites" do
1291 admin = insert(:user, is_admin: true)
1295 |> assign(:user, admin)
1300 test "no invites", %{conn: conn} do
1301 conn = get(conn, "/api/pleroma/admin/users/invites")
1303 assert json_response(conn, 200) == %{"invites" => []}
1306 test "with invite", %{conn: conn} do
1307 {:ok, invite} = UserInviteToken.create_invite()
1309 conn = get(conn, "/api/pleroma/admin/users/invites")
1311 assert json_response(conn, 200) == %{
1314 "expires_at" => nil,
1316 "invite_type" => "one_time",
1318 "token" => invite.token,
1327 describe "POST /api/pleroma/admin/users/revoke_invite" do
1328 test "with token" do
1329 admin = insert(:user, is_admin: true)
1330 {:ok, invite} = UserInviteToken.create_invite()
1334 |> assign(:user, admin)
1335 |> post("/api/pleroma/admin/users/revoke_invite", %{"token" => invite.token})
1337 assert json_response(conn, 200) == %{
1338 "expires_at" => nil,
1340 "invite_type" => "one_time",
1342 "token" => invite.token,
1348 test "with invalid token" do
1349 admin = insert(:user, is_admin: true)
1353 |> assign(:user, admin)
1354 |> post("/api/pleroma/admin/users/revoke_invite", %{"token" => "foo"})
1356 assert json_response(conn, :not_found) == "Not found"
1360 describe "GET /api/pleroma/admin/reports/:id" do
1361 setup %{conn: conn} do
1362 admin = insert(:user, is_admin: true)
1364 %{conn: assign(conn, :user, admin)}
1367 test "returns report by its id", %{conn: conn} do
1368 [reporter, target_user] = insert_pair(:user)
1369 activity = insert(:note_activity, user: target_user)
1371 {:ok, %{id: report_id}} =
1372 CommonAPI.report(reporter, %{
1373 "account_id" => target_user.id,
1374 "comment" => "I feel offended",
1375 "status_ids" => [activity.id]
1380 |> get("/api/pleroma/admin/reports/#{report_id}")
1381 |> json_response(:ok)
1383 assert response["id"] == report_id
1386 test "returns 404 when report id is invalid", %{conn: conn} do
1387 conn = get(conn, "/api/pleroma/admin/reports/test")
1389 assert json_response(conn, :not_found) == "Not found"
1393 describe "PATCH /api/pleroma/admin/reports" do
1394 setup %{conn: conn} do
1395 admin = insert(:user, is_admin: true)
1396 [reporter, target_user] = insert_pair(:user)
1397 activity = insert(:note_activity, user: target_user)
1399 {:ok, %{id: report_id}} =
1400 CommonAPI.report(reporter, %{
1401 "account_id" => target_user.id,
1402 "comment" => "I feel offended",
1403 "status_ids" => [activity.id]
1406 {:ok, %{id: second_report_id}} =
1407 CommonAPI.report(reporter, %{
1408 "account_id" => target_user.id,
1409 "comment" => "I feel very offended",
1410 "status_ids" => [activity.id]
1414 conn: assign(conn, :user, admin),
1417 second_report_id: second_report_id
1421 test "mark report as resolved", %{conn: conn, id: id, admin: admin} do
1423 |> patch("/api/pleroma/admin/reports", %{
1425 %{"state" => "resolved", "id" => id}
1428 |> json_response(:no_content)
1430 activity = Activity.get_by_id(id)
1431 assert activity.data["state"] == "resolved"
1433 log_entry = Repo.one(ModerationLog)
1435 assert ModerationLog.get_log_entry_message(log_entry) ==
1436 "@#{admin.nickname} updated report ##{id} with 'resolved' state"
1439 test "closes report", %{conn: conn, id: id, admin: admin} do
1441 |> patch("/api/pleroma/admin/reports", %{
1443 %{"state" => "closed", "id" => id}
1446 |> json_response(:no_content)
1448 activity = Activity.get_by_id(id)
1449 assert activity.data["state"] == "closed"
1451 log_entry = Repo.one(ModerationLog)
1453 assert ModerationLog.get_log_entry_message(log_entry) ==
1454 "@#{admin.nickname} updated report ##{id} with 'closed' state"
1457 test "returns 400 when state is unknown", %{conn: conn, id: id} do
1460 |> patch("/api/pleroma/admin/reports", %{
1462 %{"state" => "test", "id" => id}
1466 assert hd(json_response(conn, :bad_request))["error"] == "Unsupported state"
1469 test "returns 404 when report is not exist", %{conn: conn} do
1472 |> patch("/api/pleroma/admin/reports", %{
1474 %{"state" => "closed", "id" => "test"}
1478 assert hd(json_response(conn, :bad_request))["error"] == "not_found"
1481 test "updates state of multiple reports", %{
1485 second_report_id: second_report_id
1488 |> patch("/api/pleroma/admin/reports", %{
1490 %{"state" => "resolved", "id" => id},
1491 %{"state" => "closed", "id" => second_report_id}
1494 |> json_response(:no_content)
1496 activity = Activity.get_by_id(id)
1497 second_activity = Activity.get_by_id(second_report_id)
1498 assert activity.data["state"] == "resolved"
1499 assert second_activity.data["state"] == "closed"
1501 [first_log_entry, second_log_entry] = Repo.all(ModerationLog)
1503 assert ModerationLog.get_log_entry_message(first_log_entry) ==
1504 "@#{admin.nickname} updated report ##{id} with 'resolved' state"
1506 assert ModerationLog.get_log_entry_message(second_log_entry) ==
1507 "@#{admin.nickname} updated report ##{second_report_id} with 'closed' state"
1511 describe "GET /api/pleroma/admin/reports" do
1512 setup %{conn: conn} do
1513 admin = insert(:user, is_admin: true)
1515 %{conn: assign(conn, :user, admin)}
1518 test "returns empty response when no reports created", %{conn: conn} do
1521 |> get("/api/pleroma/admin/reports")
1522 |> json_response(:ok)
1524 assert Enum.empty?(response["reports"])
1525 assert response["total"] == 0
1528 test "returns reports", %{conn: conn} do
1529 [reporter, target_user] = insert_pair(:user)
1530 activity = insert(:note_activity, user: target_user)
1532 {:ok, %{id: report_id}} =
1533 CommonAPI.report(reporter, %{
1534 "account_id" => target_user.id,
1535 "comment" => "I feel offended",
1536 "status_ids" => [activity.id]
1541 |> get("/api/pleroma/admin/reports")
1542 |> json_response(:ok)
1544 [report] = response["reports"]
1546 assert length(response["reports"]) == 1
1547 assert report["id"] == report_id
1549 assert response["total"] == 1
1552 test "returns reports with specified state", %{conn: conn} do
1553 [reporter, target_user] = insert_pair(:user)
1554 activity = insert(:note_activity, user: target_user)
1556 {:ok, %{id: first_report_id}} =
1557 CommonAPI.report(reporter, %{
1558 "account_id" => target_user.id,
1559 "comment" => "I feel offended",
1560 "status_ids" => [activity.id]
1563 {:ok, %{id: second_report_id}} =
1564 CommonAPI.report(reporter, %{
1565 "account_id" => target_user.id,
1566 "comment" => "I don't like this user"
1569 CommonAPI.update_report_state(second_report_id, "closed")
1573 |> get("/api/pleroma/admin/reports", %{
1576 |> json_response(:ok)
1578 [open_report] = response["reports"]
1580 assert length(response["reports"]) == 1
1581 assert open_report["id"] == first_report_id
1583 assert response["total"] == 1
1587 |> get("/api/pleroma/admin/reports", %{
1590 |> json_response(:ok)
1592 [closed_report] = response["reports"]
1594 assert length(response["reports"]) == 1
1595 assert closed_report["id"] == second_report_id
1597 assert response["total"] == 1
1601 |> get("/api/pleroma/admin/reports", %{
1602 "state" => "resolved"
1604 |> json_response(:ok)
1606 assert Enum.empty?(response["reports"])
1607 assert response["total"] == 0
1610 test "returns 403 when requested by a non-admin" do
1611 user = insert(:user)
1615 |> assign(:user, user)
1616 |> get("/api/pleroma/admin/reports")
1618 assert json_response(conn, :forbidden) ==
1619 %{"error" => "User is not an admin or OAuth admin scope is not granted."}
1622 test "returns 403 when requested by anonymous" do
1625 |> get("/api/pleroma/admin/reports")
1627 assert json_response(conn, :forbidden) == %{"error" => "Invalid credentials."}
1631 describe "GET /api/pleroma/admin/grouped_reports" do
1632 setup %{conn: conn} do
1633 admin = insert(:user, is_admin: true)
1634 [reporter, target_user] = insert_pair(:user)
1636 date1 = (DateTime.to_unix(DateTime.utc_now()) + 1000) |> DateTime.from_unix!()
1637 date2 = (DateTime.to_unix(DateTime.utc_now()) + 2000) |> DateTime.from_unix!()
1638 date3 = (DateTime.to_unix(DateTime.utc_now()) + 3000) |> DateTime.from_unix!()
1641 insert(:note_activity, user: target_user, data_attrs: %{"published" => date1})
1644 insert(:note_activity, user: target_user, data_attrs: %{"published" => date2})
1647 insert(:note_activity, user: target_user, data_attrs: %{"published" => date3})
1649 {:ok, first_report} =
1650 CommonAPI.report(reporter, %{
1651 "account_id" => target_user.id,
1652 "status_ids" => [first_status.id, second_status.id, third_status.id]
1655 {:ok, second_report} =
1656 CommonAPI.report(reporter, %{
1657 "account_id" => target_user.id,
1658 "status_ids" => [first_status.id, second_status.id]
1661 {:ok, third_report} =
1662 CommonAPI.report(reporter, %{
1663 "account_id" => target_user.id,
1664 "status_ids" => [first_status.id]
1668 conn: assign(conn, :user, admin),
1669 first_status: Activity.get_by_ap_id_with_object(first_status.data["id"]),
1670 second_status: Activity.get_by_ap_id_with_object(second_status.data["id"]),
1671 third_status: Activity.get_by_ap_id_with_object(third_status.data["id"]),
1672 first_report: first_report,
1673 first_status_reports: [first_report, second_report, third_report],
1674 second_status_reports: [first_report, second_report],
1675 third_status_reports: [first_report],
1676 target_user: target_user,
1681 test "returns reports grouped by status", %{
1683 first_status: first_status,
1684 second_status: second_status,
1685 third_status: third_status,
1686 first_status_reports: first_status_reports,
1687 second_status_reports: second_status_reports,
1688 third_status_reports: third_status_reports,
1689 target_user: target_user,
1694 |> get("/api/pleroma/admin/grouped_reports")
1695 |> json_response(:ok)
1697 assert length(response["reports"]) == 3
1699 first_group = Enum.find(response["reports"], &(&1["status"]["id"] == first_status.id))
1701 second_group = Enum.find(response["reports"], &(&1["status"]["id"] == second_status.id))
1703 third_group = Enum.find(response["reports"], &(&1["status"]["id"] == third_status.id))
1705 assert length(first_group["reports"]) == 3
1706 assert length(second_group["reports"]) == 2
1707 assert length(third_group["reports"]) == 1
1709 assert first_group["date"] ==
1710 Enum.max_by(first_status_reports, fn act ->
1711 NaiveDateTime.from_iso8601!(act.data["published"])
1712 end).data["published"]
1714 assert first_group["status"] ==
1716 stringify_keys(StatusView.render("show.json", %{activity: first_status})),
1721 assert(first_group["account"]["id"] == target_user.id)
1723 assert length(first_group["actors"]) == 1
1724 assert hd(first_group["actors"])["id"] == reporter.id
1726 assert Enum.map(first_group["reports"], & &1["id"]) --
1727 Enum.map(first_status_reports, & &1.id) == []
1729 assert second_group["date"] ==
1730 Enum.max_by(second_status_reports, fn act ->
1731 NaiveDateTime.from_iso8601!(act.data["published"])
1732 end).data["published"]
1734 assert second_group["status"] ==
1736 stringify_keys(StatusView.render("show.json", %{activity: second_status})),
1741 assert second_group["account"]["id"] == target_user.id
1743 assert length(second_group["actors"]) == 1
1744 assert hd(second_group["actors"])["id"] == reporter.id
1746 assert Enum.map(second_group["reports"], & &1["id"]) --
1747 Enum.map(second_status_reports, & &1.id) == []
1749 assert third_group["date"] ==
1750 Enum.max_by(third_status_reports, fn act ->
1751 NaiveDateTime.from_iso8601!(act.data["published"])
1752 end).data["published"]
1754 assert third_group["status"] ==
1756 stringify_keys(StatusView.render("show.json", %{activity: third_status})),
1761 assert third_group["account"]["id"] == target_user.id
1763 assert length(third_group["actors"]) == 1
1764 assert hd(third_group["actors"])["id"] == reporter.id
1766 assert Enum.map(third_group["reports"], & &1["id"]) --
1767 Enum.map(third_status_reports, & &1.id) == []
1770 test "reopened report renders status data", %{
1772 first_report: first_report,
1773 first_status: first_status
1775 {:ok, _} = CommonAPI.update_report_state(first_report.id, "resolved")
1779 |> get("/api/pleroma/admin/grouped_reports")
1780 |> json_response(:ok)
1782 first_group = Enum.find(response["reports"], &(&1["status"]["id"] == first_status.id))
1784 assert first_group["status"] ==
1786 stringify_keys(StatusView.render("show.json", %{activity: first_status})),
1792 test "reopened report does not render status data if status has been deleted", %{
1794 first_report: first_report,
1795 first_status: first_status,
1796 target_user: target_user
1798 {:ok, _} = CommonAPI.update_report_state(first_report.id, "resolved")
1799 {:ok, _} = CommonAPI.delete(first_status.id, target_user)
1801 refute Activity.get_by_ap_id(first_status.id)
1805 |> get("/api/pleroma/admin/grouped_reports")
1806 |> json_response(:ok)
1808 assert Enum.find(response["reports"], &(&1["status"]["deleted"] == true))["status"][
1812 assert length(Enum.filter(response["reports"], &(&1["status"]["deleted"] == false))) == 2
1815 test "account not empty if status was deleted", %{
1817 first_report: first_report,
1818 first_status: first_status,
1819 target_user: target_user
1821 {:ok, _} = CommonAPI.update_report_state(first_report.id, "resolved")
1822 {:ok, _} = CommonAPI.delete(first_status.id, target_user)
1824 refute Activity.get_by_ap_id(first_status.id)
1828 |> get("/api/pleroma/admin/grouped_reports")
1829 |> json_response(:ok)
1831 assert Enum.find(response["reports"], &(&1["status"]["deleted"] == true))["account"]
1835 describe "PUT /api/pleroma/admin/statuses/:id" do
1836 setup %{conn: conn} do
1837 admin = insert(:user, is_admin: true)
1838 activity = insert(:note_activity)
1840 %{conn: assign(conn, :user, admin), id: activity.id, admin: admin}
1843 test "toggle sensitive flag", %{conn: conn, id: id, admin: admin} do
1846 |> put("/api/pleroma/admin/statuses/#{id}", %{"sensitive" => "true"})
1847 |> json_response(:ok)
1849 assert response["sensitive"]
1851 log_entry = Repo.one(ModerationLog)
1853 assert ModerationLog.get_log_entry_message(log_entry) ==
1854 "@#{admin.nickname} updated status ##{id}, set sensitive: 'true'"
1858 |> put("/api/pleroma/admin/statuses/#{id}", %{"sensitive" => "false"})
1859 |> json_response(:ok)
1861 refute response["sensitive"]
1864 test "change visibility flag", %{conn: conn, id: id, admin: admin} do
1867 |> put("/api/pleroma/admin/statuses/#{id}", %{"visibility" => "public"})
1868 |> json_response(:ok)
1870 assert response["visibility"] == "public"
1872 log_entry = Repo.one(ModerationLog)
1874 assert ModerationLog.get_log_entry_message(log_entry) ==
1875 "@#{admin.nickname} updated status ##{id}, set visibility: 'public'"
1879 |> put("/api/pleroma/admin/statuses/#{id}", %{"visibility" => "private"})
1880 |> json_response(:ok)
1882 assert response["visibility"] == "private"
1886 |> put("/api/pleroma/admin/statuses/#{id}", %{"visibility" => "unlisted"})
1887 |> json_response(:ok)
1889 assert response["visibility"] == "unlisted"
1892 test "returns 400 when visibility is unknown", %{conn: conn, id: id} do
1895 |> put("/api/pleroma/admin/statuses/#{id}", %{"visibility" => "test"})
1897 assert json_response(conn, :bad_request) == "Unsupported visibility"
1901 describe "DELETE /api/pleroma/admin/statuses/:id" do
1902 setup %{conn: conn} do
1903 admin = insert(:user, is_admin: true)
1904 activity = insert(:note_activity)
1906 %{conn: assign(conn, :user, admin), id: activity.id, admin: admin}
1909 test "deletes status", %{conn: conn, id: id, admin: admin} do
1911 |> delete("/api/pleroma/admin/statuses/#{id}")
1912 |> json_response(:ok)
1914 refute Activity.get_by_id(id)
1916 log_entry = Repo.one(ModerationLog)
1918 assert ModerationLog.get_log_entry_message(log_entry) ==
1919 "@#{admin.nickname} deleted status ##{id}"
1922 test "returns error when status is not exist", %{conn: conn} do
1925 |> delete("/api/pleroma/admin/statuses/test")
1927 assert json_response(conn, :bad_request) == "Could not delete"
1931 describe "GET /api/pleroma/admin/config" do
1932 setup %{conn: conn} do
1933 admin = insert(:user, is_admin: true)
1935 %{conn: assign(conn, :user, admin)}
1938 test "without any settings in db", %{conn: conn} do
1939 conn = get(conn, "/api/pleroma/admin/config")
1941 assert json_response(conn, 200) == %{"configs" => []}
1944 test "with settings in db", %{conn: conn} do
1945 config1 = insert(:config)
1946 config2 = insert(:config)
1948 conn = get(conn, "/api/pleroma/admin/config")
1953 "group" => ":pleroma",
1962 } = json_response(conn, 200)
1964 assert key1 == config1.key
1965 assert key2 == config2.key
1969 describe "POST /api/pleroma/admin/config" do
1970 setup %{conn: conn} do
1971 admin = insert(:user, is_admin: true)
1974 Application.delete_env(:pleroma, :key1)
1975 Application.delete_env(:pleroma, :key2)
1976 Application.delete_env(:pleroma, :key3)
1977 Application.delete_env(:pleroma, :key4)
1978 Application.delete_env(:pleroma, :keyaa1)
1979 Application.delete_env(:pleroma, :keyaa2)
1980 Application.delete_env(:pleroma, Pleroma.Web.Endpoint.NotReal)
1981 Application.delete_env(:pleroma, Pleroma.Captcha.NotReal)
1982 Application.put_env(:tesla, :adapter, Tesla.Mock)
1983 :ok = File.rm("config/test.exported_from_db.secret.exs")
1986 %{conn: assign(conn, :user, admin)}
1989 clear_config([:instance, :dynamic_configuration]) do
1990 Pleroma.Config.put([:instance, :dynamic_configuration], true)
1993 @tag capture_log: true
1994 test "create new config setting in db", %{conn: conn} do
1996 post(conn, "/api/pleroma/admin/config", %{
1998 %{group: ":pleroma", key: ":key1", value: "value1"},
2000 group: ":ueberauth",
2001 key: "Ueberauth.Strategy.Twitter.OAuth",
2002 value: [%{"tuple" => [":consumer_secret", "aaaa"]}]
2008 ":nested_1" => "nested_value1",
2010 %{":nested_22" => "nested_value222"},
2011 %{":nested_33" => %{":nested_44" => "nested_444"}}
2019 %{"nested_3" => ":nested_3", "nested_33" => "nested_33"},
2020 %{"nested_4" => true}
2026 value: %{":nested_5" => ":upload", "endpoint" => "https://example.com"}
2031 value: %{"tuple" => ["string", "Pleroma.Captcha.NotReal", []]}
2036 assert json_response(conn, 200) == %{
2039 "group" => ":pleroma",
2044 "group" => ":ueberauth",
2045 "key" => "Ueberauth.Strategy.Twitter.OAuth",
2046 "value" => [%{"tuple" => [":consumer_secret", "aaaa"]}]
2049 "group" => ":pleroma",
2052 ":nested_1" => "nested_value1",
2054 %{":nested_22" => "nested_value222"},
2055 %{":nested_33" => %{":nested_44" => "nested_444"}}
2060 "group" => ":pleroma",
2063 %{"nested_3" => ":nested_3", "nested_33" => "nested_33"},
2064 %{"nested_4" => true}
2068 "group" => ":pleroma",
2070 "value" => %{"endpoint" => "https://example.com", ":nested_5" => ":upload"}
2075 "value" => %{"tuple" => ["string", "Pleroma.Captcha.NotReal", []]}
2080 assert Application.get_env(:pleroma, :key1) == "value1"
2082 assert Application.get_env(:pleroma, :key2) == %{
2083 nested_1: "nested_value1",
2085 %{nested_22: "nested_value222"},
2086 %{nested_33: %{nested_44: "nested_444"}}
2090 assert Application.get_env(:pleroma, :key3) == [
2091 %{"nested_3" => :nested_3, "nested_33" => "nested_33"},
2092 %{"nested_4" => true}
2095 assert Application.get_env(:pleroma, :key4) == %{
2096 "endpoint" => "https://example.com",
2100 assert Application.get_env(:idna, :key5) == {"string", Pleroma.Captcha.NotReal, []}
2103 test "save config setting without key", %{conn: conn} do
2104 initial = Application.get_all_env(:quack)
2105 on_exit(fn -> Application.put_all_env([{:quack, initial}]) end)
2108 post(conn, "/api/pleroma/admin/config", %{
2122 key: ":webhook_url",
2123 value: "https://hooks.slack.com/services/KEY"
2128 assert json_response(conn, 200) == %{
2130 %{"group" => ":quack", "key" => ":level", "value" => ":info"},
2131 %{"group" => ":quack", "key" => ":meta", "value" => [":none"]},
2133 "group" => ":quack",
2134 "key" => ":webhook_url",
2135 "value" => "https://hooks.slack.com/services/KEY"
2140 assert Application.get_env(:quack, :level) == :info
2141 assert Application.get_env(:quack, :meta) == [:none]
2142 assert Application.get_env(:quack, :webhook_url) == "https://hooks.slack.com/services/KEY"
2145 test "saving config with partial update", %{conn: conn} do
2146 config = insert(:config, key: ":key1", value: :erlang.term_to_binary(key1: 1, key2: 2))
2149 post(conn, "/api/pleroma/admin/config", %{
2151 %{group: config.group, key: config.key, value: [%{"tuple" => [":key3", 3]}]}
2155 assert json_response(conn, 200) == %{
2158 "group" => ":pleroma",
2161 %{"tuple" => [":key1", 1]},
2162 %{"tuple" => [":key2", 2]},
2163 %{"tuple" => [":key3", 3]}
2170 test "saving full setting if value is not keyword", %{conn: conn} do
2175 value: :erlang.term_to_binary(Tesla.Adapter.Hackey)
2179 post(conn, "/api/pleroma/admin/config", %{
2181 %{group: config.group, key: config.key, value: "Tesla.Adapter.Httpc"}
2185 assert json_response(conn, 200) == %{
2188 "group" => ":tesla",
2189 "key" => ":adapter",
2190 "value" => "Tesla.Adapter.Httpc"
2196 test "update config setting & delete", %{conn: conn} do
2197 config1 = insert(:config, key: ":keyaa1")
2198 config2 = insert(:config, key: ":keyaa2")
2202 key: "Ueberauth.Strategy.Microsoft.OAuth"
2206 post(conn, "/api/pleroma/admin/config", %{
2208 %{group: config1.group, key: config1.key, value: "another_value"},
2209 %{group: config2.group, key: config2.key, delete: "true"},
2212 key: "Ueberauth.Strategy.Microsoft.OAuth",
2218 assert json_response(conn, 200) == %{
2221 "group" => ":pleroma",
2222 "key" => config1.key,
2223 "value" => "another_value"
2228 assert Application.get_env(:pleroma, :keyaa1) == "another_value"
2229 refute Application.get_env(:pleroma, :keyaa2)
2232 test "common config example", %{conn: conn} do
2233 adapter = Application.get_env(:tesla, :adapter)
2234 on_exit(fn -> Application.put_env(:tesla, :adapter, adapter) end)
2237 post(conn, "/api/pleroma/admin/config", %{
2240 "group" => ":pleroma",
2241 "key" => "Pleroma.Captcha.NotReal",
2243 %{"tuple" => [":enabled", false]},
2244 %{"tuple" => [":method", "Pleroma.Captcha.Kocaptcha"]},
2245 %{"tuple" => [":seconds_valid", 60]},
2246 %{"tuple" => [":path", ""]},
2247 %{"tuple" => [":key1", nil]},
2248 %{"tuple" => [":partial_chain", "&:hackney_connect.partial_chain/1"]},
2249 %{"tuple" => [":regex1", "~r/https:\/\/example.com/"]},
2250 %{"tuple" => [":regex2", "~r/https:\/\/example.com/u"]},
2251 %{"tuple" => [":regex3", "~r/https:\/\/example.com/i"]},
2252 %{"tuple" => [":regex4", "~r/https:\/\/example.com/s"]},
2253 %{"tuple" => [":name", "Pleroma"]}
2256 %{"group" => ":tesla", "key" => ":adapter", "value" => "Tesla.Adapter.Httpc"}
2260 assert Application.get_env(:tesla, :adapter) == Tesla.Adapter.Httpc
2261 assert Pleroma.Config.get([Pleroma.Captcha.NotReal, :name]) == "Pleroma"
2263 assert json_response(conn, 200) == %{
2266 "group" => ":pleroma",
2267 "key" => "Pleroma.Captcha.NotReal",
2269 %{"tuple" => [":enabled", false]},
2270 %{"tuple" => [":method", "Pleroma.Captcha.Kocaptcha"]},
2271 %{"tuple" => [":seconds_valid", 60]},
2272 %{"tuple" => [":path", ""]},
2273 %{"tuple" => [":key1", nil]},
2274 %{"tuple" => [":partial_chain", "&:hackney_connect.partial_chain/1"]},
2275 %{"tuple" => [":regex1", "~r/https:\\/\\/example.com/"]},
2276 %{"tuple" => [":regex2", "~r/https:\\/\\/example.com/u"]},
2277 %{"tuple" => [":regex3", "~r/https:\\/\\/example.com/i"]},
2278 %{"tuple" => [":regex4", "~r/https:\\/\\/example.com/s"]},
2279 %{"tuple" => [":name", "Pleroma"]}
2282 %{"group" => ":tesla", "key" => ":adapter", "value" => "Tesla.Adapter.Httpc"}
2287 test "tuples with more than two values", %{conn: conn} do
2289 post(conn, "/api/pleroma/admin/config", %{
2292 "group" => ":pleroma",
2293 "key" => "Pleroma.Web.Endpoint.NotReal",
2309 "/api/v1/streaming",
2310 "Pleroma.Web.MastodonAPI.WebsocketHandler",
2317 "Phoenix.Endpoint.CowboyWebSocket",
2320 "Phoenix.Transports.WebSocket",
2323 "Pleroma.Web.Endpoint",
2324 "Pleroma.Web.UserSocket",
2335 "Phoenix.Endpoint.Cowboy2Handler",
2336 %{"tuple" => ["Pleroma.Web.Endpoint", []]}
2353 assert json_response(conn, 200) == %{
2356 "group" => ":pleroma",
2357 "key" => "Pleroma.Web.Endpoint.NotReal",
2373 "/api/v1/streaming",
2374 "Pleroma.Web.MastodonAPI.WebsocketHandler",
2381 "Phoenix.Endpoint.CowboyWebSocket",
2384 "Phoenix.Transports.WebSocket",
2387 "Pleroma.Web.Endpoint",
2388 "Pleroma.Web.UserSocket",
2399 "Phoenix.Endpoint.Cowboy2Handler",
2400 %{"tuple" => ["Pleroma.Web.Endpoint", []]}
2418 test "settings with nesting map", %{conn: conn} do
2420 post(conn, "/api/pleroma/admin/config", %{
2423 "group" => ":pleroma",
2426 %{"tuple" => [":key2", "some_val"]},
2431 ":max_options" => 20,
2432 ":max_option_chars" => 200,
2433 ":min_expiration" => 0,
2434 ":max_expiration" => 31_536_000,
2436 ":max_options" => 20,
2437 ":max_option_chars" => 200,
2438 ":min_expiration" => 0,
2439 ":max_expiration" => 31_536_000
2449 assert json_response(conn, 200) ==
2453 "group" => ":pleroma",
2456 %{"tuple" => [":key2", "some_val"]},
2461 ":max_expiration" => 31_536_000,
2462 ":max_option_chars" => 200,
2463 ":max_options" => 20,
2464 ":min_expiration" => 0,
2466 ":max_expiration" => 31_536_000,
2467 ":max_option_chars" => 200,
2468 ":max_options" => 20,
2469 ":min_expiration" => 0
2480 test "value as map", %{conn: conn} do
2482 post(conn, "/api/pleroma/admin/config", %{
2485 "group" => ":pleroma",
2487 "value" => %{"key" => "some_val"}
2492 assert json_response(conn, 200) ==
2496 "group" => ":pleroma",
2498 "value" => %{"key" => "some_val"}
2504 test "dispatch setting", %{conn: conn} do
2506 post(conn, "/api/pleroma/admin/config", %{
2509 "group" => ":pleroma",
2510 "key" => "Pleroma.Web.Endpoint.NotReal",
2516 %{"tuple" => [":ip", %{"tuple" => [127, 0, 0, 1]}]},
2517 %{"tuple" => [":dispatch", ["{:_,
2519 {\"/api/v1/streaming\", Pleroma.Web.MastodonAPI.WebsocketHandler, []},
2520 {\"/websocket\", Phoenix.Endpoint.CowboyWebSocket,
2521 {Phoenix.Transports.WebSocket,
2522 {Pleroma.Web.Endpoint, Pleroma.Web.UserSocket, [path: \"/websocket\"]}}},
2523 {:_, Phoenix.Endpoint.Cowboy2Handler, {Pleroma.Web.Endpoint, []}}
2534 "{:_, [{\"/api/v1/streaming\", Pleroma.Web.MastodonAPI.WebsocketHandler, []}, " <>
2535 "{\"/websocket\", Phoenix.Endpoint.CowboyWebSocket, {Phoenix.Transports.WebSocket, " <>
2536 "{Pleroma.Web.Endpoint, Pleroma.Web.UserSocket, [path: \"/websocket\"]}}}, " <>
2537 "{:_, Phoenix.Endpoint.Cowboy2Handler, {Pleroma.Web.Endpoint, []}}]}"
2539 assert json_response(conn, 200) == %{
2542 "group" => ":pleroma",
2543 "key" => "Pleroma.Web.Endpoint.NotReal",
2549 %{"tuple" => [":ip", %{"tuple" => [127, 0, 0, 1]}]},
2567 test "queues key as atom", %{conn: conn} do
2569 post(conn, "/api/pleroma/admin/config", %{
2575 %{"tuple" => [":federator_incoming", 50]},
2576 %{"tuple" => [":federator_outgoing", 50]},
2577 %{"tuple" => [":web_push", 50]},
2578 %{"tuple" => [":mailer", 10]},
2579 %{"tuple" => [":transmogrifier", 20]},
2580 %{"tuple" => [":scheduled_activities", 10]},
2581 %{"tuple" => [":background", 5]}
2587 assert json_response(conn, 200) == %{
2593 %{"tuple" => [":federator_incoming", 50]},
2594 %{"tuple" => [":federator_outgoing", 50]},
2595 %{"tuple" => [":web_push", 50]},
2596 %{"tuple" => [":mailer", 10]},
2597 %{"tuple" => [":transmogrifier", 20]},
2598 %{"tuple" => [":scheduled_activities", 10]},
2599 %{"tuple" => [":background", 5]}
2606 test "delete part of settings by atom subkeys", %{conn: conn} do
2610 value: :erlang.term_to_binary(subkey1: "val1", subkey2: "val2", subkey3: "val3")
2614 post(conn, "/api/pleroma/admin/config", %{
2617 group: config.group,
2619 subkeys: [":subkey1", ":subkey3"],
2626 json_response(conn, 200) == %{
2629 "group" => ":pleroma",
2631 "value" => [%{"tuple" => [":subkey2", "val2"]}]
2639 describe "config mix tasks run" do
2640 setup %{conn: conn} do
2641 admin = insert(:user, is_admin: true)
2643 temp_file = "config/test.exported_from_db.secret.exs"
2645 Mix.shell(Mix.Shell.Quiet)
2648 Mix.shell(Mix.Shell.IO)
2649 :ok = File.rm(temp_file)
2652 %{conn: assign(conn, :user, admin), admin: admin}
2655 clear_config([:instance, :dynamic_configuration]) do
2656 Pleroma.Config.put([:instance, :dynamic_configuration], true)
2659 clear_config([:feed, :post_title]) do
2660 Pleroma.Config.put([:feed, :post_title], %{max_length: 100, omission: "…"})
2663 test "transfer settings to DB and to file", %{conn: conn, admin: admin} do
2664 assert Pleroma.Repo.all(Pleroma.Web.AdminAPI.Config) == []
2665 conn = get(conn, "/api/pleroma/admin/config/migrate_to_db")
2666 assert json_response(conn, 200) == %{}
2667 assert Pleroma.Repo.all(Pleroma.Web.AdminAPI.Config) > 0
2671 |> assign(:user, admin)
2672 |> get("/api/pleroma/admin/config/migrate_from_db")
2674 assert json_response(conn, 200) == %{}
2675 assert Pleroma.Repo.all(Pleroma.Web.AdminAPI.Config) == []
2679 describe "GET /api/pleroma/admin/users/:nickname/statuses" do
2681 admin = insert(:user, is_admin: true)
2682 user = insert(:user)
2684 date1 = (DateTime.to_unix(DateTime.utc_now()) + 2000) |> DateTime.from_unix!()
2685 date2 = (DateTime.to_unix(DateTime.utc_now()) + 1000) |> DateTime.from_unix!()
2686 date3 = (DateTime.to_unix(DateTime.utc_now()) + 3000) |> DateTime.from_unix!()
2688 insert(:note_activity, user: user, published: date1)
2689 insert(:note_activity, user: user, published: date2)
2690 insert(:note_activity, user: user, published: date3)
2694 |> assign(:user, admin)
2696 {:ok, conn: conn, user: user}
2699 test "renders user's statuses", %{conn: conn, user: user} do
2700 conn = get(conn, "/api/pleroma/admin/users/#{user.nickname}/statuses")
2702 assert json_response(conn, 200) |> length() == 3
2705 test "renders user's statuses with a limit", %{conn: conn, user: user} do
2706 conn = get(conn, "/api/pleroma/admin/users/#{user.nickname}/statuses?page_size=2")
2708 assert json_response(conn, 200) |> length() == 2
2711 test "doesn't return private statuses by default", %{conn: conn, user: user} do
2712 {:ok, _private_status} =
2713 CommonAPI.post(user, %{"status" => "private", "visibility" => "private"})
2715 {:ok, _public_status} =
2716 CommonAPI.post(user, %{"status" => "public", "visibility" => "public"})
2718 conn = get(conn, "/api/pleroma/admin/users/#{user.nickname}/statuses")
2720 assert json_response(conn, 200) |> length() == 4
2723 test "returns private statuses with godmode on", %{conn: conn, user: user} do
2724 {:ok, _private_status} =
2725 CommonAPI.post(user, %{"status" => "private", "visibility" => "private"})
2727 {:ok, _public_status} =
2728 CommonAPI.post(user, %{"status" => "public", "visibility" => "public"})
2730 conn = get(conn, "/api/pleroma/admin/users/#{user.nickname}/statuses?godmode=true")
2732 assert json_response(conn, 200) |> length() == 5
2736 describe "GET /api/pleroma/admin/moderation_log" do
2737 setup %{conn: conn} do
2738 admin = insert(:user, is_admin: true)
2739 moderator = insert(:user, is_moderator: true)
2741 %{conn: assign(conn, :user, admin), admin: admin, moderator: moderator}
2744 test "returns the log", %{conn: conn, admin: admin} do
2745 Repo.insert(%ModerationLog{
2749 "nickname" => admin.nickname,
2752 action: "relay_follow",
2753 target: "https://example.org/relay"
2755 inserted_at: NaiveDateTime.truncate(~N[2017-08-15 15:47:06.597036], :second)
2758 Repo.insert(%ModerationLog{
2762 "nickname" => admin.nickname,
2765 action: "relay_unfollow",
2766 target: "https://example.org/relay"
2768 inserted_at: NaiveDateTime.truncate(~N[2017-08-16 15:47:06.597036], :second)
2771 conn = get(conn, "/api/pleroma/admin/moderation_log")
2773 response = json_response(conn, 200)
2774 [first_entry, second_entry] = response["items"]
2776 assert response["total"] == 2
2777 assert first_entry["data"]["action"] == "relay_unfollow"
2779 assert first_entry["message"] ==
2780 "@#{admin.nickname} unfollowed relay: https://example.org/relay"
2782 assert second_entry["data"]["action"] == "relay_follow"
2784 assert second_entry["message"] ==
2785 "@#{admin.nickname} followed relay: https://example.org/relay"
2788 test "returns the log with pagination", %{conn: conn, admin: admin} do
2789 Repo.insert(%ModerationLog{
2793 "nickname" => admin.nickname,
2796 action: "relay_follow",
2797 target: "https://example.org/relay"
2799 inserted_at: NaiveDateTime.truncate(~N[2017-08-15 15:47:06.597036], :second)
2802 Repo.insert(%ModerationLog{
2806 "nickname" => admin.nickname,
2809 action: "relay_unfollow",
2810 target: "https://example.org/relay"
2812 inserted_at: NaiveDateTime.truncate(~N[2017-08-16 15:47:06.597036], :second)
2815 conn1 = get(conn, "/api/pleroma/admin/moderation_log?page_size=1&page=1")
2817 response1 = json_response(conn1, 200)
2818 [first_entry] = response1["items"]
2820 assert response1["total"] == 2
2821 assert response1["items"] |> length() == 1
2822 assert first_entry["data"]["action"] == "relay_unfollow"
2824 assert first_entry["message"] ==
2825 "@#{admin.nickname} unfollowed relay: https://example.org/relay"
2827 conn2 = get(conn, "/api/pleroma/admin/moderation_log?page_size=1&page=2")
2829 response2 = json_response(conn2, 200)
2830 [second_entry] = response2["items"]
2832 assert response2["total"] == 2
2833 assert response2["items"] |> length() == 1
2834 assert second_entry["data"]["action"] == "relay_follow"
2836 assert second_entry["message"] ==
2837 "@#{admin.nickname} followed relay: https://example.org/relay"
2840 test "filters log by date", %{conn: conn, admin: admin} do
2841 first_date = "2017-08-15T15:47:06Z"
2842 second_date = "2017-08-20T15:47:06Z"
2844 Repo.insert(%ModerationLog{
2848 "nickname" => admin.nickname,
2851 action: "relay_follow",
2852 target: "https://example.org/relay"
2854 inserted_at: NaiveDateTime.from_iso8601!(first_date)
2857 Repo.insert(%ModerationLog{
2861 "nickname" => admin.nickname,
2864 action: "relay_unfollow",
2865 target: "https://example.org/relay"
2867 inserted_at: NaiveDateTime.from_iso8601!(second_date)
2873 "/api/pleroma/admin/moderation_log?start_date=#{second_date}"
2876 response1 = json_response(conn1, 200)
2877 [first_entry] = response1["items"]
2879 assert response1["total"] == 1
2880 assert first_entry["data"]["action"] == "relay_unfollow"
2882 assert first_entry["message"] ==
2883 "@#{admin.nickname} unfollowed relay: https://example.org/relay"
2886 test "returns log filtered by user", %{conn: conn, admin: admin, moderator: moderator} do
2887 Repo.insert(%ModerationLog{
2891 "nickname" => admin.nickname,
2894 action: "relay_follow",
2895 target: "https://example.org/relay"
2899 Repo.insert(%ModerationLog{
2902 "id" => moderator.id,
2903 "nickname" => moderator.nickname,
2906 action: "relay_unfollow",
2907 target: "https://example.org/relay"
2911 conn1 = get(conn, "/api/pleroma/admin/moderation_log?user_id=#{moderator.id}")
2913 response1 = json_response(conn1, 200)
2914 [first_entry] = response1["items"]
2916 assert response1["total"] == 1
2917 assert get_in(first_entry, ["data", "actor", "id"]) == moderator.id
2920 test "returns log filtered by search", %{conn: conn, moderator: moderator} do
2921 ModerationLog.insert_log(%{
2923 action: "relay_follow",
2924 target: "https://example.org/relay"
2927 ModerationLog.insert_log(%{
2929 action: "relay_unfollow",
2930 target: "https://example.org/relay"
2933 conn1 = get(conn, "/api/pleroma/admin/moderation_log?search=unfo")
2935 response1 = json_response(conn1, 200)
2936 [first_entry] = response1["items"]
2938 assert response1["total"] == 1
2940 assert get_in(first_entry, ["data", "message"]) ==
2941 "@#{moderator.nickname} unfollowed relay: https://example.org/relay"
2945 describe "PATCH /users/:nickname/force_password_reset" do
2946 setup %{conn: conn} do
2947 admin = insert(:user, is_admin: true)
2948 user = insert(:user)
2950 %{conn: assign(conn, :user, admin), admin: admin, user: user}
2953 test "sets password_reset_pending to true", %{admin: admin, user: user} do
2954 assert user.password_reset_pending == false
2958 |> assign(:user, admin)
2959 |> patch("/api/pleroma/admin/users/force_password_reset", %{nicknames: [user.nickname]})
2961 assert json_response(conn, 204) == ""
2963 ObanHelpers.perform_all()
2965 assert User.get_by_id(user.id).password_reset_pending == true
2969 describe "relays" do
2970 setup %{conn: conn} do
2971 admin = insert(:user, is_admin: true)
2973 %{conn: assign(conn, :user, admin), admin: admin}
2976 test "POST /relay", %{admin: admin} do
2979 |> assign(:user, admin)
2980 |> post("/api/pleroma/admin/relay", %{
2981 relay_url: "http://mastodon.example.org/users/admin"
2984 assert json_response(conn, 200) == "http://mastodon.example.org/users/admin"
2986 log_entry = Repo.one(ModerationLog)
2988 assert ModerationLog.get_log_entry_message(log_entry) ==
2989 "@#{admin.nickname} followed relay: http://mastodon.example.org/users/admin"
2992 test "GET /relay", %{admin: admin} do
2993 relay_user = Pleroma.Web.ActivityPub.Relay.get_actor()
2995 ["http://mastodon.example.org/users/admin", "https://mstdn.io/users/mayuutann"]
2996 |> Enum.each(fn ap_id ->
2997 {:ok, user} = User.get_or_fetch_by_ap_id(ap_id)
2998 User.follow(relay_user, user)
3003 |> assign(:user, admin)
3004 |> get("/api/pleroma/admin/relay")
3006 assert json_response(conn, 200)["relays"] -- ["mastodon.example.org", "mstdn.io"] == []
3009 test "DELETE /relay", %{admin: admin} do
3011 |> assign(:user, admin)
3012 |> post("/api/pleroma/admin/relay", %{
3013 relay_url: "http://mastodon.example.org/users/admin"
3018 |> assign(:user, admin)
3019 |> delete("/api/pleroma/admin/relay", %{
3020 relay_url: "http://mastodon.example.org/users/admin"
3023 assert json_response(conn, 200) == "http://mastodon.example.org/users/admin"
3025 [log_entry_one, log_entry_two] = Repo.all(ModerationLog)
3027 assert ModerationLog.get_log_entry_message(log_entry_one) ==
3028 "@#{admin.nickname} followed relay: http://mastodon.example.org/users/admin"
3030 assert ModerationLog.get_log_entry_message(log_entry_two) ==
3031 "@#{admin.nickname} unfollowed relay: http://mastodon.example.org/users/admin"
3035 describe "instances" do
3036 test "GET /instances/:instance/statuses" do
3037 admin = insert(:user, is_admin: true)
3038 user = insert(:user, local: false, nickname: "archaeme@archae.me")
3039 user2 = insert(:user, local: false, nickname: "test@test.com")
3040 insert_pair(:note_activity, user: user)
3041 insert(:note_activity, user: user2)
3045 |> assign(:user, admin)
3046 |> get("/api/pleroma/admin/instances/archae.me/statuses")
3048 response = json_response(conn, 200)
3050 assert length(response) == 2
3054 |> assign(:user, admin)
3055 |> get("/api/pleroma/admin/instances/test.com/statuses")
3057 response = json_response(conn, 200)
3059 assert length(response) == 1
3063 |> assign(:user, admin)
3064 |> get("/api/pleroma/admin/instances/nonexistent.com/statuses")
3066 response = json_response(conn, 200)
3068 assert length(response) == 0
3072 describe "PATCH /confirm_email" do
3073 setup %{conn: conn} do
3074 admin = insert(:user, is_admin: true)
3076 %{conn: assign(conn, :user, admin), admin: admin}
3079 test "it confirms emails of two users", %{admin: admin} do
3080 [first_user, second_user] = insert_pair(:user, confirmation_pending: true)
3082 assert first_user.confirmation_pending == true
3083 assert second_user.confirmation_pending == true
3086 |> assign(:user, admin)
3087 |> patch("/api/pleroma/admin/users/confirm_email", %{
3089 first_user.nickname,
3090 second_user.nickname
3094 assert first_user.confirmation_pending == true
3095 assert second_user.confirmation_pending == true
3097 log_entry = Repo.one(ModerationLog)
3099 assert ModerationLog.get_log_entry_message(log_entry) ==
3100 "@#{admin.nickname} confirmed email for users: @#{first_user.nickname}, @#{
3101 second_user.nickname
3106 describe "PATCH /resend_confirmation_email" do
3107 setup %{conn: conn} do
3108 admin = insert(:user, is_admin: true)
3110 %{conn: assign(conn, :user, admin), admin: admin}
3113 test "it resend emails for two users", %{admin: admin} do
3114 [first_user, second_user] = insert_pair(:user, confirmation_pending: true)
3117 |> assign(:user, admin)
3118 |> patch("/api/pleroma/admin/users/resend_confirmation_email", %{
3120 first_user.nickname,
3121 second_user.nickname
3125 log_entry = Repo.one(ModerationLog)
3127 assert ModerationLog.get_log_entry_message(log_entry) ==
3128 "@#{admin.nickname} re-sent confirmation email for users: @#{first_user.nickname}, @#{
3129 second_user.nickname
3134 describe "POST /reports/:id/notes" do
3136 admin = insert(:user, is_admin: true)
3137 [reporter, target_user] = insert_pair(:user)
3138 activity = insert(:note_activity, user: target_user)
3140 {:ok, %{id: report_id}} =
3141 CommonAPI.report(reporter, %{
3142 "account_id" => target_user.id,
3143 "comment" => "I feel offended",
3144 "status_ids" => [activity.id]
3148 |> assign(:user, admin)
3149 |> post("/api/pleroma/admin/reports/#{report_id}/notes", %{
3150 content: "this is disgusting!"
3154 |> assign(:user, admin)
3155 |> post("/api/pleroma/admin/reports/#{report_id}/notes", %{
3156 content: "this is disgusting2!"
3161 report_id: report_id,
3166 test "it creates report note", %{admin_id: admin_id, report_id: report_id} do
3167 [note, _] = Repo.all(ReportNote)
3170 activity_id: ^report_id,
3171 content: "this is disgusting!",
3176 test "it returns reports with notes", %{admin: admin} do
3179 |> assign(:user, admin)
3180 |> get("/api/pleroma/admin/reports")
3182 response = json_response(conn, 200)
3183 notes = hd(response["reports"])["notes"]
3186 assert note["user"]["nickname"] == admin.nickname
3187 assert note["content"] == "this is disgusting!"
3188 assert note["created_at"]
3189 assert response["total"] == 1
3192 test "it deletes the note", %{admin: admin, report_id: report_id} do
3193 assert ReportNote |> Repo.all() |> length() == 2
3195 [note, _] = Repo.all(ReportNote)
3198 |> assign(:user, admin)
3199 |> delete("/api/pleroma/admin/reports/#{report_id}/notes/#{note.id}")
3201 assert ReportNote |> Repo.all() |> length() == 1
3205 test "GET /api/pleroma/admin/config/descriptions", %{conn: conn} do
3206 admin = insert(:user, is_admin: true)
3209 assign(conn, :user, admin)
3210 |> get("/api/pleroma/admin/config/descriptions")
3212 assert [child | _others] = json_response(conn, 200)
3214 assert child["children"]
3216 assert String.starts_with?(child["group"], ":")
3217 assert child["description"]
3221 # Needed for testing
3222 defmodule Pleroma.Web.Endpoint.NotReal do
3225 defmodule Pleroma.Captcha.NotReal do