1 # Pleroma: A lightweight social networking server
2 # Copyright © 2017-2018 Pleroma Authors <https://pleroma.social/>
3 # SPDX-License-Identifier: AGPL-3.0-only
5 defmodule Pleroma.Web.ActivityPub.ActivityPubControllerTest do
6 use Pleroma.Web.ConnCase
8 alias Pleroma.Web.ActivityPub.{UserView, ObjectView}
9 alias Pleroma.{Repo, User}
10 alias Pleroma.Activity
13 Tesla.Mock.mock_global(fn env -> apply(HttpRequestMock, :request, [env]) end)
18 test "with the relay active, it returns the relay user", %{conn: conn} do
21 |> get(activity_pub_path(conn, :relay))
24 assert res["id"] =~ "/relay"
27 test "with the relay disabled, it returns 404", %{conn: conn} do
28 Pleroma.Config.put([:instance, :allow_relay], false)
31 |> get(activity_pub_path(conn, :relay))
35 Pleroma.Config.put([:instance, :allow_relay], true)
39 describe "/users/:nickname" do
40 test "it returns a json representation of the user", %{conn: conn} do
45 |> put_req_header("accept", "application/activity+json")
46 |> get("/users/#{user.nickname}")
48 user = Repo.get(User, user.id)
50 assert json_response(conn, 200) == UserView.render("user.json", %{user: user})
54 describe "/object/:uuid" do
55 test "it returns a json representation of the object", %{conn: conn} do
57 uuid = String.split(note.data["id"], "/") |> List.last()
61 |> put_req_header("accept", "application/activity+json")
62 |> get("/objects/#{uuid}")
64 assert json_response(conn, 200) == ObjectView.render("object.json", %{object: note})
67 test "it returns 404 for non-public messages", %{conn: conn} do
68 note = insert(:direct_note)
69 uuid = String.split(note.data["id"], "/") |> List.last()
73 |> put_req_header("accept", "application/activity+json")
74 |> get("/objects/#{uuid}")
76 assert json_response(conn, 404)
81 test "it inserts an incoming activity into the database", %{conn: conn} do
82 data = File.read!("test/fixtures/mastodon-post-activity.json") |> Poison.decode!()
86 |> assign(:valid_signature, true)
87 |> put_req_header("content-type", "application/activity+json")
88 |> post("/inbox", data)
90 assert "ok" == json_response(conn, 200)
92 assert Activity.get_by_ap_id(data["id"])
96 describe "/users/:nickname/inbox" do
97 test "it inserts an incoming activity into the database", %{conn: conn} do
101 File.read!("test/fixtures/mastodon-post-activity.json")
103 |> Map.put("bcc", [user.ap_id])
107 |> assign(:valid_signature, true)
108 |> put_req_header("content-type", "application/activity+json")
109 |> post("/users/#{user.nickname}/inbox", data)
111 assert "ok" == json_response(conn, 200)
113 assert Activity.get_by_ap_id(data["id"])
116 test "it rejects reads from other users", %{conn: conn} do
118 otheruser = insert(:user)
122 |> assign(:user, otheruser)
123 |> put_req_header("accept", "application/activity+json")
124 |> get("/users/#{user.nickname}/inbox")
126 assert json_response(conn, 403)
129 test "it returns a note activity in a collection", %{conn: conn} do
130 note_activity = insert(:direct_note_activity)
131 user = User.get_cached_by_ap_id(hd(note_activity.data["to"]))
135 |> assign(:user, user)
136 |> put_req_header("accept", "application/activity+json")
137 |> get("/users/#{user.nickname}/inbox")
139 assert response(conn, 200) =~ note_activity.data["object"]["content"]
143 describe "/users/:nickname/outbox" do
144 test "it returns a note activity in a collection", %{conn: conn} do
145 note_activity = insert(:note_activity)
146 user = User.get_cached_by_ap_id(note_activity.data["actor"])
150 |> put_req_header("accept", "application/activity+json")
151 |> get("/users/#{user.nickname}/outbox")
153 assert response(conn, 200) =~ note_activity.data["object"]["content"]
156 test "it returns an announce activity in a collection", %{conn: conn} do
157 announce_activity = insert(:announce_activity)
158 user = User.get_cached_by_ap_id(announce_activity.data["actor"])
162 |> put_req_header("accept", "application/activity+json")
163 |> get("/users/#{user.nickname}/outbox")
165 assert response(conn, 200) =~ announce_activity.data["object"]
168 test "it rejects posts from other users", %{conn: conn} do
169 data = File.read!("test/fixtures/activitypub-client-post-activity.json") |> Poison.decode!()
171 otheruser = insert(:user)
175 |> assign(:user, otheruser)
176 |> put_req_header("content-type", "application/activity+json")
177 |> post("/users/#{user.nickname}/outbox", data)
179 assert json_response(conn, 403)
182 test "it inserts an incoming activity into the database", %{conn: conn} do
183 data = File.read!("test/fixtures/activitypub-client-post-activity.json") |> Poison.decode!()
188 |> assign(:user, user)
189 |> put_req_header("content-type", "application/activity+json")
190 |> post("/users/#{user.nickname}/outbox", data)
192 result = json_response(conn, 201)
193 assert Activity.get_by_ap_id(result["id"])
196 test "it rejects an incoming activity with bogus type", %{conn: conn} do
197 data = File.read!("test/fixtures/activitypub-client-post-activity.json") |> Poison.decode!()
202 |> Map.put("type", "BadType")
206 |> assign(:user, user)
207 |> put_req_header("content-type", "application/activity+json")
208 |> post("/users/#{user.nickname}/outbox", data)
210 assert json_response(conn, 400)
214 describe "/users/:nickname/followers" do
215 test "it returns the followers in a collection", %{conn: conn} do
217 user_two = insert(:user)
218 User.follow(user, user_two)
222 |> get("/users/#{user_two.nickname}/followers")
223 |> json_response(200)
225 assert result["first"]["orderedItems"] == [user.ap_id]
228 test "it returns returns empty if the user has 'hide_network' set", %{conn: conn} do
230 user_two = insert(:user, %{info: %{hide_network: true}})
231 User.follow(user, user_two)
235 |> get("/users/#{user_two.nickname}/followers")
236 |> json_response(200)
238 assert result["first"]["orderedItems"] == []
239 assert result["totalItems"] == 1
242 test "it works for more than 10 users", %{conn: conn} do
245 Enum.each(1..15, fn _ ->
246 other_user = insert(:user)
247 User.follow(other_user, user)
252 |> get("/users/#{user.nickname}/followers")
253 |> json_response(200)
255 assert length(result["first"]["orderedItems"]) == 10
256 assert result["first"]["totalItems"] == 15
257 assert result["totalItems"] == 15
261 |> get("/users/#{user.nickname}/followers?page=2")
262 |> json_response(200)
264 assert length(result["orderedItems"]) == 5
265 assert result["totalItems"] == 15
269 describe "/users/:nickname/following" do
270 test "it returns the following in a collection", %{conn: conn} do
272 user_two = insert(:user)
273 User.follow(user, user_two)
277 |> get("/users/#{user.nickname}/following")
278 |> json_response(200)
280 assert result["first"]["orderedItems"] == [user_two.ap_id]
283 test "it returns returns empty if the user has 'hide_network' set", %{conn: conn} do
284 user = insert(:user, %{info: %{hide_network: true}})
285 user_two = insert(:user)
286 User.follow(user, user_two)
290 |> get("/users/#{user.nickname}/following")
291 |> json_response(200)
293 assert result["first"]["orderedItems"] == []
294 assert result["totalItems"] == 1
297 test "it works for more than 10 users", %{conn: conn} do
300 Enum.each(1..15, fn _ ->
301 user = Repo.get(User, user.id)
302 other_user = insert(:user)
303 User.follow(user, other_user)
308 |> get("/users/#{user.nickname}/following")
309 |> json_response(200)
311 assert length(result["first"]["orderedItems"]) == 10
312 assert result["first"]["totalItems"] == 15
313 assert result["totalItems"] == 15
317 |> get("/users/#{user.nickname}/following?page=2")
318 |> json_response(200)
320 assert length(result["orderedItems"]) == 5
321 assert result["totalItems"] == 15