activitypub transmogrifier: rewrite non-http URLs using the object's external URL
[akkoma] / test / web / activity_pub / transmogrifier_test.exs
1 defmodule Pleroma.Web.ActivityPub.TransmogrifierTest do
2 use Pleroma.DataCase
3 alias Pleroma.Web.ActivityPub.Transmogrifier
4 alias Pleroma.Web.OStatus
5 alias Pleroma.Activity
6 alias Pleroma.User
7 alias Pleroma.Repo
8 alias Pleroma.Web.Websub.WebsubClientSubscription
9 alias Pleroma.Web.Websub.WebsubServerSubscription
10 import Ecto.Query
11
12 import Pleroma.Factory
13 alias Pleroma.Web.CommonAPI
14
15 describe "handle_incoming" do
16 test "it ignores an incoming notice if we already have it" do
17 activity = insert(:note_activity)
18
19 data = File.read!("test/fixtures/mastodon-post-activity.json")
20 |> Poison.decode!
21 |> Map.put("object", activity.data["object"])
22
23 {:ok, returned_activity} = Transmogrifier.handle_incoming(data)
24
25 assert activity == returned_activity
26 end
27
28 test "it fetches replied-to activities if we don't have them" do
29 data = File.read!("test/fixtures/mastodon-post-activity.json")
30 |> Poison.decode!
31
32 object = data["object"]
33 |> Map.put("inReplyTo", "https://shitposter.club/notice/2827873")
34
35 data = data
36 |> Map.put("object", object)
37
38 {:ok, returned_activity} = Transmogrifier.handle_incoming(data)
39
40 assert activity = Activity.get_create_activity_by_object_ap_id("tag:shitposter.club,2017-05-05:noticeId=2827873:objectType=comment")
41 assert returned_activity.data["object"]["inReplyToAtomUri"] == "https://shitposter.club/notice/2827873"
42 assert returned_activity.data["object"]["inReplyToStatusId"] == activity.id
43 end
44
45 test "it works for incoming notices" do
46 data = File.read!("test/fixtures/mastodon-post-activity.json") |> Poison.decode!
47
48 {:ok, %Activity{data: data, local: false}} = Transmogrifier.handle_incoming(data)
49 assert data["id"] == "http://mastodon.example.org/users/admin/statuses/99512778738411822/activity"
50 assert data["context"] == "tag:mastodon.example.org,2018-02-12:objectId=20:objectType=Conversation"
51 assert data["to"] == ["https://www.w3.org/ns/activitystreams#Public"]
52 assert data["cc"] == [
53 "http://mastodon.example.org/users/admin/followers",
54 "http://localtesting.pleroma.lol/users/lain"
55 ]
56 assert data["actor"] == "http://mastodon.example.org/users/admin"
57
58 object = data["object"]
59 assert object["id"] == "http://mastodon.example.org/users/admin/statuses/99512778738411822"
60
61 assert object["to"] == ["https://www.w3.org/ns/activitystreams#Public"]
62 assert object["cc"] == [
63 "http://mastodon.example.org/users/admin/followers",
64 "http://localtesting.pleroma.lol/users/lain"
65 ]
66 assert object["actor"] == "http://mastodon.example.org/users/admin"
67 assert object["attributedTo"] == "http://mastodon.example.org/users/admin"
68 assert object["context"] == "tag:mastodon.example.org,2018-02-12:objectId=20:objectType=Conversation"
69 assert object["sensitive"] == true
70 end
71
72 test "it works for incoming follow requests" do
73 user = insert(:user)
74 data = File.read!("test/fixtures/mastodon-follow-activity.json") |> Poison.decode!
75 |> Map.put("object", user.ap_id)
76
77 {:ok, %Activity{data: data, local: false}} = Transmogrifier.handle_incoming(data)
78
79 assert data["actor"] == "http://mastodon.example.org/users/admin"
80 assert data["type"] == "Follow"
81 assert data["id"] == "http://mastodon.example.org/users/admin#follows/2"
82 assert User.following?(User.get_by_ap_id(data["actor"]), user)
83 end
84
85 test "it works for incoming likes" do
86 user = insert(:user)
87 {:ok, activity} = CommonAPI.post(user, %{"status" => "hello"})
88
89 data = File.read!("test/fixtures/mastodon-like.json") |> Poison.decode!
90 |> Map.put("object", activity.data["object"]["id"])
91
92 {:ok, %Activity{data: data, local: false}} = Transmogrifier.handle_incoming(data)
93
94 assert data["actor"] == "http://mastodon.example.org/users/admin"
95 assert data["type"] == "Like"
96 assert data["id"] == "http://mastodon.example.org/users/admin#likes/2"
97 assert data["object"] == activity.data["object"]["id"]
98 end
99
100 test "it works for incoming announces" do
101 data = File.read!("test/fixtures/mastodon-announce.json") |> Poison.decode!
102
103 {:ok, %Activity{data: data, local: false}} = Transmogrifier.handle_incoming(data)
104
105 assert data["actor"] == "http://mastodon.example.org/users/admin"
106 assert data["type"] == "Announce"
107 assert data["id"] == "http://mastodon.example.org/users/admin/statuses/99542391527669785/activity"
108 assert data["object"] == "http://mastodon.example.org/users/admin/statuses/99541947525187367"
109
110 assert Activity.get_create_activity_by_object_ap_id(data["object"])
111 end
112
113 test "it works for incoming announces with an existing activity" do
114 user = insert(:user)
115 {:ok, activity} = CommonAPI.post(user, %{"status" => "hey"})
116
117 data = File.read!("test/fixtures/mastodon-announce.json")
118 |> Poison.decode!
119 |> Map.put("object", activity.data["object"]["id"])
120
121 {:ok, %Activity{data: data, local: false}} = Transmogrifier.handle_incoming(data)
122
123 assert data["actor"] == "http://mastodon.example.org/users/admin"
124 assert data["type"] == "Announce"
125 assert data["id"] == "http://mastodon.example.org/users/admin/statuses/99542391527669785/activity"
126 assert data["object"] == activity.data["object"]["id"]
127
128 assert Activity.get_create_activity_by_object_ap_id(data["object"]).id == activity.id
129 end
130
131 test "it works for incoming update activities" do
132 data = File.read!("test/fixtures/mastodon-post-activity.json") |> Poison.decode!
133
134 {:ok, %Activity{data: data, local: false}} = Transmogrifier.handle_incoming(data)
135 update_data = File.read!("test/fixtures/mastodon-update.json") |> Poison.decode!
136 object = update_data["object"]
137 |> Map.put("actor", data["actor"])
138 |> Map.put("id", data["actor"])
139
140 update_data = update_data
141 |> Map.put("actor", data["actor"])
142 |> Map.put("object", object)
143
144 {:ok, %Activity{data: data, local: false}} = Transmogrifier.handle_incoming(update_data)
145
146 user = User.get_cached_by_ap_id(data["actor"])
147 assert user.name == "gargle"
148 assert user.avatar["url"] == [%{"href" => "https://cd.niu.moe/accounts/avatars/000/033/323/original/fd7f8ae0b3ffedc9.jpeg"}]
149 assert user.info["banner"]["url"] == [%{"href" => "https://cd.niu.moe/accounts/headers/000/033/323/original/850b3448fa5fd477.png"}]
150 assert user.bio == "<p>Some bio</p>"
151 end
152
153 test "it works for incoming deletes" do
154 activity = insert(:note_activity)
155 data = File.read!("test/fixtures/mastodon-delete.json")
156 |> Poison.decode!
157
158 object = data["object"]
159 |> Map.put("id", activity.data["object"]["id"])
160
161 data = data
162 |> Map.put("object", object)
163 |> Map.put("actor", activity.data["actor"])
164
165 {:ok, %Activity{data: data, local: false}} = Transmogrifier.handle_incoming(data)
166
167 refute Repo.get(Activity, activity.id)
168 end
169 end
170
171 describe "prepare outgoing" do
172 test "it turns mentions into tags" do
173 user = insert(:user)
174 other_user = insert(:user)
175
176 {:ok, activity} = CommonAPI.post(user, %{"status" => "hey, @#{other_user.nickname}, how are ya? #2hu"})
177
178 {:ok, modified} = Transmogrifier.prepare_outgoing(activity.data)
179 object = modified["object"]
180
181 expected_mention = %{
182 "href" => other_user.ap_id,
183 "name" => "@#{other_user.nickname}",
184 "type" => "Mention"
185 }
186
187 expected_tag = %{
188 "href" => Pleroma.Web.Endpoint.url <> "/tags/2hu",
189 "type" => "Hashtag",
190 "name" => "#2hu"
191 }
192
193 assert Enum.member?(object["tag"], expected_tag)
194 assert Enum.member?(object["tag"], expected_mention)
195 end
196
197 test "it adds the sensitive property" do
198 user = insert(:user)
199
200 {:ok, activity} = CommonAPI.post(user, %{"status" => "#nsfw hey"})
201 {:ok, modified} = Transmogrifier.prepare_outgoing(activity.data)
202
203 assert modified["object"]["sensitive"]
204 end
205
206 test "it adds the json-ld context and the conversation property" do
207 user = insert(:user)
208
209 {:ok, activity} = CommonAPI.post(user, %{"status" => "hey"})
210 {:ok, modified} = Transmogrifier.prepare_outgoing(activity.data)
211
212 assert modified["@context"] == "https://www.w3.org/ns/activitystreams"
213 assert modified["object"]["conversation"] == modified["context"]
214 end
215
216 test "it sets the 'attributedTo' property to the actor of the object if it doesn't have one" do
217 user = insert(:user)
218
219 {:ok, activity} = CommonAPI.post(user, %{"status" => "hey"})
220 {:ok, modified} = Transmogrifier.prepare_outgoing(activity.data)
221
222 assert modified["object"]["actor"] == modified["object"]["attributedTo"]
223 end
224
225 test "it translates ostatus IDs to external URLs" do
226 incoming = File.read!("test/fixtures/incoming_note_activity.xml")
227 {:ok, [referent_activity]} = OStatus.handle_incoming(incoming)
228
229 user = insert(:user)
230
231 {:ok, activity, _} = CommonAPI.favorite(referent_activity.id, user)
232 {:ok, modified} = Transmogrifier.prepare_outgoing(activity.data)
233
234 assert modified["object"] == "http://gs.example.org:4040/index.php/notice/29"
235 end
236 end
237
238 describe "user upgrade" do
239 test "it upgrades a user to activitypub" do
240 user = insert(:user, %{nickname: "rye@niu.moe", local: false, ap_id: "https://niu.moe/users/rye", follower_address: User.ap_followers(%User{nickname: "rye@niu.moe"})})
241 user_two = insert(:user, %{following: [user.follower_address]})
242
243 {:ok, activity} = CommonAPI.post(user, %{"status" => "test"})
244 {:ok, unrelated_activity} = CommonAPI.post(user_two, %{"status" => "test"})
245 assert "http://localhost:4001/users/rye@niu.moe/followers" in activity.recipients
246
247 user = Repo.get(User, user.id)
248 assert user.info["note_count"] == 1
249
250 {:ok, user} = Transmogrifier.upgrade_user_from_ap_id("https://niu.moe/users/rye")
251 assert user.info["ap_enabled"]
252 assert user.info["note_count"] == 1
253 assert user.follower_address == "https://niu.moe/users/rye/followers"
254
255 # Wait for the background task
256 :timer.sleep(1000)
257
258 user = Repo.get(User, user.id)
259 assert user.info["note_count"] == 1
260
261 activity = Repo.get(Activity, activity.id)
262 assert user.follower_address in activity.recipients
263 assert %{"url" => [%{"href" => "https://cdn.niu.moe/accounts/avatars/000/033/323/original/fd7f8ae0b3ffedc9.jpeg"}]} = user.avatar
264 assert %{"url" => [%{"href" => "https://cdn.niu.moe/accounts/headers/000/033/323/original/850b3448fa5fd477.png"}]} = user.info["banner"]
265 refute "..." in activity.recipients
266
267 unrelated_activity = Repo.get(Activity, unrelated_activity.id)
268 refute user.follower_address in unrelated_activity.recipients
269
270 user_two = Repo.get(User, user_two.id)
271 assert user.follower_address in user_two.following
272 refute "..." in user_two.following
273 end
274 end
275
276 describe "maybe_retire_websub" do
277 test "it deletes all websub client subscripitions with the user as topic" do
278 subscription = %WebsubClientSubscription{topic: "https://niu.moe/users/rye.atom"}
279 {:ok, ws} = Repo.insert(subscription)
280
281 subscription = %WebsubClientSubscription{topic: "https://niu.moe/users/pasty.atom"}
282 {:ok, ws2} = Repo.insert(subscription)
283
284 Transmogrifier.maybe_retire_websub("https://niu.moe/users/rye")
285
286 refute Repo.get(WebsubClientSubscription, ws.id)
287 assert Repo.get(WebsubClientSubscription, ws2.id)
288 end
289 end
290 end