Merge branch 'bugfix/activitypub-hashtags' into 'develop'
[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 notices with hashtags" do
73 data = File.read!("test/fixtures/mastodon-post-activity-hashtag.json") |> Poison.decode!
74
75 {:ok, %Activity{data: data, local: false}} = Transmogrifier.handle_incoming(data)
76 assert Enum.at(data["object"]["tag"], 2) == "moo"
77 end
78
79 test "it works for incoming follow requests" do
80 user = insert(:user)
81 data = File.read!("test/fixtures/mastodon-follow-activity.json") |> Poison.decode!
82 |> Map.put("object", user.ap_id)
83
84 {:ok, %Activity{data: data, local: false}} = Transmogrifier.handle_incoming(data)
85
86 assert data["actor"] == "http://mastodon.example.org/users/admin"
87 assert data["type"] == "Follow"
88 assert data["id"] == "http://mastodon.example.org/users/admin#follows/2"
89 assert User.following?(User.get_by_ap_id(data["actor"]), user)
90 end
91
92 test "it works for incoming likes" do
93 user = insert(:user)
94 {:ok, activity} = CommonAPI.post(user, %{"status" => "hello"})
95
96 data = File.read!("test/fixtures/mastodon-like.json") |> Poison.decode!
97 |> Map.put("object", activity.data["object"]["id"])
98
99 {:ok, %Activity{data: data, local: false}} = Transmogrifier.handle_incoming(data)
100
101 assert data["actor"] == "http://mastodon.example.org/users/admin"
102 assert data["type"] == "Like"
103 assert data["id"] == "http://mastodon.example.org/users/admin#likes/2"
104 assert data["object"] == activity.data["object"]["id"]
105 end
106
107 test "it works for incoming announces" do
108 data = File.read!("test/fixtures/mastodon-announce.json") |> Poison.decode!
109
110 {:ok, %Activity{data: data, local: false}} = Transmogrifier.handle_incoming(data)
111
112 assert data["actor"] == "http://mastodon.example.org/users/admin"
113 assert data["type"] == "Announce"
114 assert data["id"] == "http://mastodon.example.org/users/admin/statuses/99542391527669785/activity"
115 assert data["object"] == "http://mastodon.example.org/users/admin/statuses/99541947525187367"
116
117 assert Activity.get_create_activity_by_object_ap_id(data["object"])
118 end
119
120 test "it works for incoming announces with an existing activity" do
121 user = insert(:user)
122 {:ok, activity} = CommonAPI.post(user, %{"status" => "hey"})
123
124 data = File.read!("test/fixtures/mastodon-announce.json")
125 |> Poison.decode!
126 |> Map.put("object", activity.data["object"]["id"])
127
128 {:ok, %Activity{data: data, local: false}} = Transmogrifier.handle_incoming(data)
129
130 assert data["actor"] == "http://mastodon.example.org/users/admin"
131 assert data["type"] == "Announce"
132 assert data["id"] == "http://mastodon.example.org/users/admin/statuses/99542391527669785/activity"
133 assert data["object"] == activity.data["object"]["id"]
134
135 assert Activity.get_create_activity_by_object_ap_id(data["object"]).id == activity.id
136 end
137
138 test "it works for incoming update activities" do
139 data = File.read!("test/fixtures/mastodon-post-activity.json") |> Poison.decode!
140
141 {:ok, %Activity{data: data, local: false}} = Transmogrifier.handle_incoming(data)
142 update_data = File.read!("test/fixtures/mastodon-update.json") |> Poison.decode!
143 object = update_data["object"]
144 |> Map.put("actor", data["actor"])
145 |> Map.put("id", data["actor"])
146
147 update_data = update_data
148 |> Map.put("actor", data["actor"])
149 |> Map.put("object", object)
150
151 {:ok, %Activity{data: data, local: false}} = Transmogrifier.handle_incoming(update_data)
152
153 user = User.get_cached_by_ap_id(data["actor"])
154 assert user.name == "gargle"
155 assert user.avatar["url"] == [%{"href" => "https://cd.niu.moe/accounts/avatars/000/033/323/original/fd7f8ae0b3ffedc9.jpeg"}]
156 assert user.info["banner"]["url"] == [%{"href" => "https://cd.niu.moe/accounts/headers/000/033/323/original/850b3448fa5fd477.png"}]
157 assert user.bio == "<p>Some bio</p>"
158 end
159
160 test "it works for incoming deletes" do
161 activity = insert(:note_activity)
162 data = File.read!("test/fixtures/mastodon-delete.json")
163 |> Poison.decode!
164
165 object = data["object"]
166 |> Map.put("id", activity.data["object"]["id"])
167
168 data = data
169 |> Map.put("object", object)
170 |> Map.put("actor", activity.data["actor"])
171
172 {:ok, %Activity{data: data, local: false}} = Transmogrifier.handle_incoming(data)
173
174 refute Repo.get(Activity, activity.id)
175 end
176 end
177
178 describe "prepare outgoing" do
179 test "it turns mentions into tags" do
180 user = insert(:user)
181 other_user = insert(:user)
182
183 {:ok, activity} = CommonAPI.post(user, %{"status" => "hey, @#{other_user.nickname}, how are ya? #2hu"})
184
185 {:ok, modified} = Transmogrifier.prepare_outgoing(activity.data)
186 object = modified["object"]
187
188 expected_mention = %{
189 "href" => other_user.ap_id,
190 "name" => "@#{other_user.nickname}",
191 "type" => "Mention"
192 }
193
194 expected_tag = %{
195 "href" => Pleroma.Web.Endpoint.url <> "/tags/2hu",
196 "type" => "Hashtag",
197 "name" => "#2hu"
198 }
199
200 assert Enum.member?(object["tag"], expected_tag)
201 assert Enum.member?(object["tag"], expected_mention)
202 end
203
204 test "it adds the sensitive property" do
205 user = insert(:user)
206
207 {:ok, activity} = CommonAPI.post(user, %{"status" => "#nsfw hey"})
208 {:ok, modified} = Transmogrifier.prepare_outgoing(activity.data)
209
210 assert modified["object"]["sensitive"]
211 end
212
213 test "it adds the json-ld context and the conversation property" do
214 user = insert(:user)
215
216 {:ok, activity} = CommonAPI.post(user, %{"status" => "hey"})
217 {:ok, modified} = Transmogrifier.prepare_outgoing(activity.data)
218
219 assert modified["@context"] == "https://www.w3.org/ns/activitystreams"
220 assert modified["object"]["conversation"] == modified["context"]
221 end
222
223 test "it sets the 'attributedTo' property to the actor of the object if it doesn't have one" do
224 user = insert(:user)
225
226 {:ok, activity} = CommonAPI.post(user, %{"status" => "hey"})
227 {:ok, modified} = Transmogrifier.prepare_outgoing(activity.data)
228
229 assert modified["object"]["actor"] == modified["object"]["attributedTo"]
230 end
231
232 test "it translates ostatus IDs to external URLs" do
233 incoming = File.read!("test/fixtures/incoming_note_activity.xml")
234 {:ok, [referent_activity]} = OStatus.handle_incoming(incoming)
235
236 user = insert(:user)
237
238 {:ok, activity, _} = CommonAPI.favorite(referent_activity.id, user)
239 {:ok, modified} = Transmogrifier.prepare_outgoing(activity.data)
240
241 assert modified["object"] == "http://gs.example.org:4040/index.php/notice/29"
242 end
243
244 test "it translates ostatus reply_to IDs to external URLs" do
245 incoming = File.read!("test/fixtures/incoming_note_activity.xml")
246 {:ok, [referred_activity]} = OStatus.handle_incoming(incoming)
247
248 user = insert(:user)
249
250 {:ok, activity} = CommonAPI.post(user, %{"status" => "HI!", "in_reply_to_status_id" => referred_activity.id})
251 {:ok, modified} = Transmogrifier.prepare_outgoing(activity.data)
252
253 assert modified["object"]["inReplyTo"] == "http://gs.example.org:4040/index.php/notice/29"
254 end
255 end
256
257 describe "user upgrade" do
258 test "it upgrades a user to activitypub" do
259 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"})})
260 user_two = insert(:user, %{following: [user.follower_address]})
261
262 {:ok, activity} = CommonAPI.post(user, %{"status" => "test"})
263 {:ok, unrelated_activity} = CommonAPI.post(user_two, %{"status" => "test"})
264 assert "http://localhost:4001/users/rye@niu.moe/followers" in activity.recipients
265
266 user = Repo.get(User, user.id)
267 assert user.info["note_count"] == 1
268
269 {:ok, user} = Transmogrifier.upgrade_user_from_ap_id("https://niu.moe/users/rye")
270 assert user.info["ap_enabled"]
271 assert user.info["note_count"] == 1
272 assert user.follower_address == "https://niu.moe/users/rye/followers"
273
274 # Wait for the background task
275 :timer.sleep(1000)
276
277 user = Repo.get(User, user.id)
278 assert user.info["note_count"] == 1
279
280 activity = Repo.get(Activity, activity.id)
281 assert user.follower_address in activity.recipients
282 assert %{"url" => [%{"href" => "https://cdn.niu.moe/accounts/avatars/000/033/323/original/fd7f8ae0b3ffedc9.jpeg"}]} = user.avatar
283 assert %{"url" => [%{"href" => "https://cdn.niu.moe/accounts/headers/000/033/323/original/850b3448fa5fd477.png"}]} = user.info["banner"]
284 refute "..." in activity.recipients
285
286 unrelated_activity = Repo.get(Activity, unrelated_activity.id)
287 refute user.follower_address in unrelated_activity.recipients
288
289 user_two = Repo.get(User, user_two.id)
290 assert user.follower_address in user_two.following
291 refute "..." in user_two.following
292 end
293 end
294
295 describe "maybe_retire_websub" do
296 test "it deletes all websub client subscripitions with the user as topic" do
297 subscription = %WebsubClientSubscription{topic: "https://niu.moe/users/rye.atom"}
298 {:ok, ws} = Repo.insert(subscription)
299
300 subscription = %WebsubClientSubscription{topic: "https://niu.moe/users/pasty.atom"}
301 {:ok, ws2} = Repo.insert(subscription)
302
303 Transmogrifier.maybe_retire_websub("https://niu.moe/users/rye")
304
305 refute Repo.get(WebsubClientSubscription, ws.id)
306 assert Repo.get(WebsubClientSubscription, ws2.id)
307 end
308 end
309 end