Merge branch 'develop' into feature/compat/push-subscriptions
[akkoma] / test / web / ostatus / ostatus_test.exs
1 defmodule Pleroma.Web.OStatusTest do
2 use Pleroma.DataCase
3 alias Pleroma.Web.OStatus
4 alias Pleroma.Web.XML
5 alias Pleroma.{Object, Repo, User, Activity}
6 import Pleroma.Factory
7 import ExUnit.CaptureLog
8
9 setup_all do
10 Tesla.Mock.mock_global(fn env -> apply(HttpRequestMock, :request, [env]) end)
11 :ok
12 end
13
14 test "don't insert create notes twice" do
15 incoming = File.read!("test/fixtures/incoming_note_activity.xml")
16 {:ok, [activity]} = OStatus.handle_incoming(incoming)
17 assert {:ok, [activity]} == OStatus.handle_incoming(incoming)
18 end
19
20 test "handle incoming note - GS, Salmon" do
21 incoming = File.read!("test/fixtures/incoming_note_activity.xml")
22 {:ok, [activity]} = OStatus.handle_incoming(incoming)
23
24 user = User.get_by_ap_id(activity.data["actor"])
25 assert user.info.note_count == 1
26 assert activity.data["type"] == "Create"
27 assert activity.data["object"]["type"] == "Note"
28
29 assert activity.data["object"]["id"] ==
30 "tag:gs.example.org:4040,2017-04-23:noticeId=29:objectType=note"
31
32 assert activity.data["published"] == "2017-04-23T14:51:03+00:00"
33 assert activity.data["object"]["published"] == "2017-04-23T14:51:03+00:00"
34
35 assert activity.data["context"] ==
36 "tag:gs.example.org:4040,2017-04-23:objectType=thread:nonce=f09e22f58abd5c7b"
37
38 assert "http://pleroma.example.org:4000/users/lain3" in activity.data["to"]
39 assert activity.data["object"]["emoji"] == %{"marko" => "marko.png", "reimu" => "reimu.png"}
40 assert activity.local == false
41 end
42
43 test "handle incoming notes - GS, subscription" do
44 incoming = File.read!("test/fixtures/ostatus_incoming_post.xml")
45 {:ok, [activity]} = OStatus.handle_incoming(incoming)
46
47 assert activity.data["type"] == "Create"
48 assert activity.data["object"]["type"] == "Note"
49 assert activity.data["object"]["actor"] == "https://social.heldscal.la/user/23211"
50 assert activity.data["object"]["content"] == "Will it blend?"
51 user = User.get_cached_by_ap_id(activity.data["actor"])
52 assert User.ap_followers(user) in activity.data["to"]
53 assert "https://www.w3.org/ns/activitystreams#Public" in activity.data["to"]
54 end
55
56 test "handle incoming notes with attachments - GS, subscription" do
57 incoming = File.read!("test/fixtures/incoming_websub_gnusocial_attachments.xml")
58 {:ok, [activity]} = OStatus.handle_incoming(incoming)
59
60 assert activity.data["type"] == "Create"
61 assert activity.data["object"]["type"] == "Note"
62 assert activity.data["object"]["actor"] == "https://social.heldscal.la/user/23211"
63 assert activity.data["object"]["attachment"] |> length == 2
64 assert activity.data["object"]["external_url"] == "https://social.heldscal.la/notice/2020923"
65 assert "https://www.w3.org/ns/activitystreams#Public" in activity.data["to"]
66 end
67
68 test "handle incoming notes with tags" do
69 incoming = File.read!("test/fixtures/ostatus_incoming_post_tag.xml")
70 {:ok, [activity]} = OStatus.handle_incoming(incoming)
71
72 assert activity.data["object"]["tag"] == ["nsfw"]
73 assert "https://www.w3.org/ns/activitystreams#Public" in activity.data["to"]
74 end
75
76 test "handle incoming notes - Mastodon, salmon, reply" do
77 # It uses the context of the replied to object
78 Repo.insert!(%Object{
79 data: %{
80 "id" => "https://pleroma.soykaf.com/objects/c237d966-ac75-4fe3-a87a-d89d71a3a7a4",
81 "context" => "2hu"
82 }
83 })
84
85 incoming = File.read!("test/fixtures/incoming_reply_mastodon.xml")
86 {:ok, [activity]} = OStatus.handle_incoming(incoming)
87
88 assert activity.data["type"] == "Create"
89 assert activity.data["object"]["type"] == "Note"
90 assert activity.data["object"]["actor"] == "https://mastodon.social/users/lambadalambda"
91 assert activity.data["context"] == "2hu"
92 assert "https://www.w3.org/ns/activitystreams#Public" in activity.data["to"]
93 end
94
95 test "handle incoming notes - Mastodon, with CW" do
96 incoming = File.read!("test/fixtures/mastodon-note-cw.xml")
97 {:ok, [activity]} = OStatus.handle_incoming(incoming)
98
99 assert activity.data["type"] == "Create"
100 assert activity.data["object"]["type"] == "Note"
101 assert activity.data["object"]["actor"] == "https://mastodon.social/users/lambadalambda"
102 assert activity.data["object"]["summary"] == "technologic"
103 assert "https://www.w3.org/ns/activitystreams#Public" in activity.data["to"]
104 end
105
106 test "handle incoming unlisted messages, put public into cc" do
107 incoming = File.read!("test/fixtures/mastodon-note-unlisted.xml")
108 {:ok, [activity]} = OStatus.handle_incoming(incoming)
109 refute "https://www.w3.org/ns/activitystreams#Public" in activity.data["to"]
110 assert "https://www.w3.org/ns/activitystreams#Public" in activity.data["cc"]
111 refute "https://www.w3.org/ns/activitystreams#Public" in activity.data["object"]["to"]
112 assert "https://www.w3.org/ns/activitystreams#Public" in activity.data["object"]["cc"]
113 end
114
115 test "handle incoming retweets - Mastodon, with CW" do
116 incoming = File.read!("test/fixtures/cw_retweet.xml")
117 {:ok, [[_activity, retweeted_activity]]} = OStatus.handle_incoming(incoming)
118
119 assert retweeted_activity.data["object"]["summary"] == "Hey."
120 end
121
122 test "handle incoming notes - GS, subscription, reply" do
123 incoming = File.read!("test/fixtures/ostatus_incoming_reply.xml")
124 {:ok, [activity]} = OStatus.handle_incoming(incoming)
125
126 assert activity.data["type"] == "Create"
127 assert activity.data["object"]["type"] == "Note"
128 assert activity.data["object"]["actor"] == "https://social.heldscal.la/user/23211"
129
130 assert activity.data["object"]["content"] ==
131 "@<a href=\"https://gs.archae.me/user/4687\" class=\"h-card u-url p-nickname mention\" title=\"shpbot\">shpbot</a> why not indeed."
132
133 assert activity.data["object"]["inReplyTo"] ==
134 "tag:gs.archae.me,2017-04-30:noticeId=778260:objectType=note"
135
136 assert "https://www.w3.org/ns/activitystreams#Public" in activity.data["to"]
137 end
138
139 test "handle incoming retweets - GS, subscription" do
140 incoming = File.read!("test/fixtures/share-gs.xml")
141 {:ok, [[activity, retweeted_activity]]} = OStatus.handle_incoming(incoming)
142
143 assert activity.data["type"] == "Announce"
144 assert activity.data["actor"] == "https://social.heldscal.la/user/23211"
145 assert activity.data["object"] == retweeted_activity.data["object"]["id"]
146 assert "https://pleroma.soykaf.com/users/lain" in activity.data["to"]
147 refute activity.local
148
149 retweeted_activity = Repo.get(Activity, retweeted_activity.id)
150 assert retweeted_activity.data["type"] == "Create"
151 assert retweeted_activity.data["actor"] == "https://pleroma.soykaf.com/users/lain"
152 refute retweeted_activity.local
153 assert retweeted_activity.data["object"]["announcement_count"] == 1
154 assert String.contains?(retweeted_activity.data["object"]["content"], "mastodon")
155 refute String.contains?(retweeted_activity.data["object"]["content"], "Test account")
156 end
157
158 test "handle incoming retweets - GS, subscription - local message" do
159 incoming = File.read!("test/fixtures/share-gs-local.xml")
160 note_activity = insert(:note_activity)
161 user = User.get_cached_by_ap_id(note_activity.data["actor"])
162
163 incoming =
164 incoming
165 |> String.replace("LOCAL_ID", note_activity.data["object"]["id"])
166 |> String.replace("LOCAL_USER", user.ap_id)
167
168 {:ok, [[activity, retweeted_activity]]} = OStatus.handle_incoming(incoming)
169
170 assert activity.data["type"] == "Announce"
171 assert activity.data["actor"] == "https://social.heldscal.la/user/23211"
172 assert activity.data["object"] == retweeted_activity.data["object"]["id"]
173 assert user.ap_id in activity.data["to"]
174 refute activity.local
175
176 retweeted_activity = Repo.get(Activity, retweeted_activity.id)
177 assert note_activity.id == retweeted_activity.id
178 assert retweeted_activity.data["type"] == "Create"
179 assert retweeted_activity.data["actor"] == user.ap_id
180 assert retweeted_activity.local
181 assert retweeted_activity.data["object"]["announcement_count"] == 1
182 end
183
184 test "handle incoming retweets - Mastodon, salmon" do
185 incoming = File.read!("test/fixtures/share.xml")
186 {:ok, [[activity, retweeted_activity]]} = OStatus.handle_incoming(incoming)
187
188 assert activity.data["type"] == "Announce"
189 assert activity.data["actor"] == "https://mastodon.social/users/lambadalambda"
190 assert activity.data["object"] == retweeted_activity.data["object"]["id"]
191
192 assert activity.data["id"] ==
193 "tag:mastodon.social,2017-05-03:objectId=4934452:objectType=Status"
194
195 refute activity.local
196 assert retweeted_activity.data["type"] == "Create"
197 assert retweeted_activity.data["actor"] == "https://pleroma.soykaf.com/users/lain"
198 refute retweeted_activity.local
199 refute String.contains?(retweeted_activity.data["object"]["content"], "Test account")
200 end
201
202 test "handle incoming favorites - GS, websub" do
203 capture_log(fn ->
204 incoming = File.read!("test/fixtures/favorite.xml")
205 {:ok, [[activity, favorited_activity]]} = OStatus.handle_incoming(incoming)
206
207 assert activity.data["type"] == "Like"
208 assert activity.data["actor"] == "https://social.heldscal.la/user/23211"
209 assert activity.data["object"] == favorited_activity.data["object"]["id"]
210
211 assert activity.data["id"] ==
212 "tag:social.heldscal.la,2017-05-05:fave:23211:comment:2061643:2017-05-05T09:12:50+00:00"
213
214 refute activity.local
215 assert favorited_activity.data["type"] == "Create"
216 assert favorited_activity.data["actor"] == "https://shitposter.club/user/1"
217
218 assert favorited_activity.data["object"]["id"] ==
219 "tag:shitposter.club,2017-05-05:noticeId=2827873:objectType=comment"
220
221 refute favorited_activity.local
222 end)
223 end
224
225 test "handle conversation references" do
226 incoming = File.read!("test/fixtures/mastodon_conversation.xml")
227 {:ok, [activity]} = OStatus.handle_incoming(incoming)
228
229 assert activity.data["context"] ==
230 "tag:mastodon.social,2017-08-28:objectId=7876885:objectType=Conversation"
231 end
232
233 test "handle incoming favorites with locally available object - GS, websub" do
234 note_activity = insert(:note_activity)
235
236 incoming =
237 File.read!("test/fixtures/favorite_with_local_note.xml")
238 |> String.replace("localid", note_activity.data["object"]["id"])
239
240 {:ok, [[activity, favorited_activity]]} = OStatus.handle_incoming(incoming)
241
242 assert activity.data["type"] == "Like"
243 assert activity.data["actor"] == "https://social.heldscal.la/user/23211"
244 assert activity.data["object"] == favorited_activity.data["object"]["id"]
245 refute activity.local
246 assert note_activity.id == favorited_activity.id
247 assert favorited_activity.local
248 end
249
250 test "handle incoming replies" do
251 incoming = File.read!("test/fixtures/incoming_note_activity_answer.xml")
252 {:ok, [activity]} = OStatus.handle_incoming(incoming)
253
254 assert activity.data["type"] == "Create"
255 assert activity.data["object"]["type"] == "Note"
256
257 assert activity.data["object"]["inReplyTo"] ==
258 "http://pleroma.example.org:4000/objects/55bce8fc-b423-46b1-af71-3759ab4670bc"
259
260 assert "http://pleroma.example.org:4000/users/lain5" in activity.data["to"]
261
262 assert activity.data["object"]["id"] ==
263 "tag:gs.example.org:4040,2017-04-25:noticeId=55:objectType=note"
264
265 assert "https://www.w3.org/ns/activitystreams#Public" in activity.data["to"]
266 end
267
268 test "handle incoming follows" do
269 incoming = File.read!("test/fixtures/follow.xml")
270 {:ok, [activity]} = OStatus.handle_incoming(incoming)
271 assert activity.data["type"] == "Follow"
272
273 assert activity.data["id"] ==
274 "tag:social.heldscal.la,2017-05-07:subscription:23211:person:44803:2017-05-07T09:54:48+00:00"
275
276 assert activity.data["actor"] == "https://social.heldscal.la/user/23211"
277 assert activity.data["object"] == "https://pawoo.net/users/pekorino"
278 refute activity.local
279
280 follower = User.get_by_ap_id(activity.data["actor"])
281 followed = User.get_by_ap_id(activity.data["object"])
282
283 assert User.following?(follower, followed)
284 end
285
286 test "handle incoming unfollows with existing follow" do
287 incoming_follow = File.read!("test/fixtures/follow.xml")
288 {:ok, [_activity]} = OStatus.handle_incoming(incoming_follow)
289
290 incoming = File.read!("test/fixtures/unfollow.xml")
291 {:ok, [activity]} = OStatus.handle_incoming(incoming)
292
293 assert activity.data["type"] == "Undo"
294
295 assert activity.data["id"] ==
296 "undo:tag:social.heldscal.la,2017-05-07:subscription:23211:person:44803:2017-05-07T09:54:48+00:00"
297
298 assert activity.data["actor"] == "https://social.heldscal.la/user/23211"
299 assert is_map(activity.data["object"])
300 assert activity.data["object"]["type"] == "Follow"
301 assert activity.data["object"]["object"] == "https://pawoo.net/users/pekorino"
302 refute activity.local
303
304 follower = User.get_by_ap_id(activity.data["actor"])
305 followed = User.get_by_ap_id(activity.data["object"]["object"])
306
307 refute User.following?(follower, followed)
308 end
309
310 describe "new remote user creation" do
311 test "returns local users" do
312 local_user = insert(:user)
313 {:ok, user} = OStatus.find_or_make_user(local_user.ap_id)
314
315 assert user == local_user
316 end
317
318 test "tries to use the information in poco fields" do
319 uri = "https://social.heldscal.la/user/23211"
320
321 {:ok, user} = OStatus.find_or_make_user(uri)
322
323 user = Repo.get(Pleroma.User, user.id)
324 assert user.name == "Constance Variable"
325 assert user.nickname == "lambadalambda@social.heldscal.la"
326 assert user.local == false
327 assert user.info.uri == uri
328 assert user.ap_id == uri
329 assert user.bio == "Call me Deacon Blues."
330 assert user.avatar["type"] == "Image"
331
332 {:ok, user_again} = OStatus.find_or_make_user(uri)
333
334 assert user == user_again
335 end
336
337 test "find_or_make_user sets all the nessary input fields" do
338 uri = "https://social.heldscal.la/user/23211"
339 {:ok, user} = OStatus.find_or_make_user(uri)
340
341 assert user.info ==
342 %Pleroma.User.Info{
343 id: user.info.id,
344 ap_enabled: false,
345 background: %{},
346 banner: %{},
347 blocks: [],
348 deactivated: false,
349 default_scope: "public",
350 domain_blocks: [],
351 follower_count: 0,
352 is_admin: false,
353 is_moderator: false,
354 keys: nil,
355 locked: false,
356 no_rich_text: false,
357 note_count: 0,
358 settings: nil,
359 source_data: %{},
360 hub: "https://social.heldscal.la/main/push/hub",
361 magic_key:
362 "RSA.uzg6r1peZU0vXGADWxGJ0PE34WvmhjUmydbX5YYdOiXfODVLwCMi1umGoqUDm-mRu4vNEdFBVJU1CpFA7dKzWgIsqsa501i2XqElmEveXRLvNRWFB6nG03Q5OUY2as8eE54BJm0p20GkMfIJGwP6TSFb-ICp3QjzbatuSPJ6xCE=.AQAB",
363 salmon: "https://social.heldscal.la/main/salmon/user/23211",
364 topic: "https://social.heldscal.la/api/statuses/user_timeline/23211.atom",
365 uri: "https://social.heldscal.la/user/23211"
366 }
367 end
368
369 test "find_make_or_update_user takes an author element and returns an updated user" do
370 uri = "https://social.heldscal.la/user/23211"
371
372 {:ok, user} = OStatus.find_or_make_user(uri)
373 old_name = user.name
374 old_bio = user.bio
375 change = Ecto.Changeset.change(user, %{avatar: nil, bio: nil, old_name: nil})
376
377 {:ok, user} = Repo.update(change)
378 refute user.avatar
379
380 doc = XML.parse_document(File.read!("test/fixtures/23211.atom"))
381 [author] = :xmerl_xpath.string('//author[1]', doc)
382 {:ok, user} = OStatus.find_make_or_update_user(author)
383 assert user.avatar["type"] == "Image"
384 assert user.name == old_name
385 assert user.bio == old_bio
386
387 {:ok, user_again} = OStatus.find_make_or_update_user(author)
388 assert user_again == user
389 end
390 end
391
392 describe "gathering user info from a user id" do
393 test "it returns user info in a hash" do
394 user = "shp@social.heldscal.la"
395
396 # TODO: make test local
397 {:ok, data} = OStatus.gather_user_info(user)
398
399 expected = %{
400 "hub" => "https://social.heldscal.la/main/push/hub",
401 "magic_key" =>
402 "RSA.wQ3i9UA0qmAxZ0WTIp4a-waZn_17Ez1pEEmqmqoooRsG1_BvpmOvLN0G2tEcWWxl2KOtdQMCiPptmQObeZeuj48mdsDZ4ArQinexY2hCCTcbV8Xpswpkb8K05RcKipdg07pnI7tAgQ0VWSZDImncL6YUGlG5YN8b5TjGOwk2VG8=.AQAB",
403 "name" => "shp",
404 "nickname" => "shp",
405 "salmon" => "https://social.heldscal.la/main/salmon/user/29191",
406 "subject" => "acct:shp@social.heldscal.la",
407 "topic" => "https://social.heldscal.la/api/statuses/user_timeline/29191.atom",
408 "uri" => "https://social.heldscal.la/user/29191",
409 "host" => "social.heldscal.la",
410 "fqn" => user,
411 "bio" => "cofe",
412 "avatar" => %{
413 "type" => "Image",
414 "url" => [
415 %{
416 "href" => "https://social.heldscal.la/avatar/29191-original-20170421154949.jpeg",
417 "mediaType" => "image/jpeg",
418 "type" => "Link"
419 }
420 ]
421 },
422 "subscribe_address" => "https://social.heldscal.la/main/ostatussub?profile={uri}",
423 "ap_id" => nil
424 }
425
426 assert data == expected
427 end
428
429 test "it works with the uri" do
430 user = "https://social.heldscal.la/user/29191"
431
432 # TODO: make test local
433 {:ok, data} = OStatus.gather_user_info(user)
434
435 expected = %{
436 "hub" => "https://social.heldscal.la/main/push/hub",
437 "magic_key" =>
438 "RSA.wQ3i9UA0qmAxZ0WTIp4a-waZn_17Ez1pEEmqmqoooRsG1_BvpmOvLN0G2tEcWWxl2KOtdQMCiPptmQObeZeuj48mdsDZ4ArQinexY2hCCTcbV8Xpswpkb8K05RcKipdg07pnI7tAgQ0VWSZDImncL6YUGlG5YN8b5TjGOwk2VG8=.AQAB",
439 "name" => "shp",
440 "nickname" => "shp",
441 "salmon" => "https://social.heldscal.la/main/salmon/user/29191",
442 "subject" => "https://social.heldscal.la/user/29191",
443 "topic" => "https://social.heldscal.la/api/statuses/user_timeline/29191.atom",
444 "uri" => "https://social.heldscal.la/user/29191",
445 "host" => "social.heldscal.la",
446 "fqn" => user,
447 "bio" => "cofe",
448 "avatar" => %{
449 "type" => "Image",
450 "url" => [
451 %{
452 "href" => "https://social.heldscal.la/avatar/29191-original-20170421154949.jpeg",
453 "mediaType" => "image/jpeg",
454 "type" => "Link"
455 }
456 ]
457 },
458 "subscribe_address" => "https://social.heldscal.la/main/ostatussub?profile={uri}",
459 "ap_id" => nil
460 }
461
462 assert data == expected
463 end
464 end
465
466 describe "fetching a status by it's HTML url" do
467 test "it builds a missing status from an html url" do
468 capture_log(fn ->
469 url = "https://shitposter.club/notice/2827873"
470 {:ok, [activity]} = OStatus.fetch_activity_from_url(url)
471
472 assert activity.data["actor"] == "https://shitposter.club/user/1"
473
474 assert activity.data["object"]["id"] ==
475 "tag:shitposter.club,2017-05-05:noticeId=2827873:objectType=comment"
476 end)
477 end
478
479 test "it works for atom notes, too" do
480 url = "https://social.sakamoto.gq/objects/0ccc1a2c-66b0-4305-b23a-7f7f2b040056"
481 {:ok, [activity]} = OStatus.fetch_activity_from_url(url)
482 assert activity.data["actor"] == "https://social.sakamoto.gq/users/eal"
483 assert activity.data["object"]["id"] == url
484 end
485 end
486
487 test "it doesn't add nil in the to field" do
488 incoming = File.read!("test/fixtures/nil_mention_entry.xml")
489 {:ok, [activity]} = OStatus.handle_incoming(incoming)
490
491 assert activity.data["to"] == [
492 "http://localhost:4001/users/atarifrosch@social.stopwatchingus-heidelberg.de/followers",
493 "https://www.w3.org/ns/activitystreams#Public"
494 ]
495 end
496
497 describe "is_representable?" do
498 test "Note objects are representable" do
499 note_activity = insert(:note_activity)
500
501 assert OStatus.is_representable?(note_activity)
502 end
503
504 test "Article objects are not representable" do
505 note_activity = insert(:note_activity)
506
507 note_object = Object.normalize(note_activity.data["object"])
508
509 note_data =
510 note_object.data
511 |> Map.put("type", "Article")
512
513 cs = Object.change(note_object, %{data: note_data})
514 {:ok, article_object} = Repo.update(cs)
515
516 # the underlying object is now an Article instead of a note, so this should fail
517 refute OStatus.is_representable?(note_activity)
518 end
519 end
520 end