Ignore duplicate create activities.
[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 test "don't insert create notes twice" do
10 incoming = File.read!("test/fixtures/incoming_note_activity.xml")
11 {:ok, [activity]} = OStatus.handle_incoming(incoming)
12 assert {:ok, [activity]} == OStatus.handle_incoming(incoming)
13 end
14
15 test "handle incoming note - GS, Salmon" do
16 incoming = File.read!("test/fixtures/incoming_note_activity.xml")
17 {:ok, [activity]} = OStatus.handle_incoming(incoming)
18
19 user = User.get_by_ap_id(activity.data["actor"])
20 assert user.info["note_count"] == 1
21 assert activity.data["type"] == "Create"
22 assert activity.data["object"]["type"] == "Note"
23 assert activity.data["object"]["id"] == "tag:gs.example.org:4040,2017-04-23:noticeId=29:objectType=note"
24 assert activity.data["published"] == "2017-04-23T14:51:03+00:00"
25 assert activity.data["object"]["published"] == "2017-04-23T14:51:03+00:00"
26 assert activity.data["context"] == "tag:gs.example.org:4040,2017-04-23:objectType=thread:nonce=f09e22f58abd5c7b"
27 assert "http://pleroma.example.org:4000/users/lain3" in activity.data["to"]
28 assert activity.data["object"]["emoji"] == %{ "marko" => "marko.png", "reimu" => "reimu.png" }
29 assert activity.local == false
30 end
31
32 test "handle incoming notes - GS, subscription" do
33 incoming = File.read!("test/fixtures/ostatus_incoming_post.xml")
34 {:ok, [activity]} = OStatus.handle_incoming(incoming)
35
36 assert activity.data["type"] == "Create"
37 assert activity.data["object"]["type"] == "Note"
38 assert activity.data["object"]["actor"] == "https://social.heldscal.la/user/23211"
39 assert activity.data["object"]["content"] == "Will it blend?"
40 user = User.get_cached_by_ap_id(activity.data["actor"])
41 assert User.ap_followers(user) in activity.data["to"]
42 assert "https://www.w3.org/ns/activitystreams#Public" in activity.data["to"]
43 end
44
45 test "handle incoming notes with attachments - GS, subscription" do
46 incoming = File.read!("test/fixtures/incoming_websub_gnusocial_attachments.xml")
47 {:ok, [activity]} = OStatus.handle_incoming(incoming)
48
49 assert activity.data["type"] == "Create"
50 assert activity.data["object"]["type"] == "Note"
51 assert activity.data["object"]["actor"] == "https://social.heldscal.la/user/23211"
52 assert activity.data["object"]["attachment"] |> length == 2
53 assert activity.data["object"]["external_url"] == "https://social.heldscal.la/notice/2020923"
54 assert "https://www.w3.org/ns/activitystreams#Public" in activity.data["to"]
55 end
56
57 test "handle incoming notes with tags" do
58 incoming = File.read!("test/fixtures/ostatus_incoming_post_tag.xml")
59 {:ok, [activity]} = OStatus.handle_incoming(incoming)
60
61 assert activity.data["object"]["tag"] == ["nsfw"]
62 assert "https://www.w3.org/ns/activitystreams#Public" in activity.data["to"]
63 end
64
65 test "handle incoming notes - Mastodon, salmon, reply" do
66 # It uses the context of the replied to object
67 Repo.insert!(%Object{
68 data: %{
69 "id" => "https://pleroma.soykaf.com/objects/c237d966-ac75-4fe3-a87a-d89d71a3a7a4",
70 "context" => "2hu"
71 }})
72 incoming = File.read!("test/fixtures/incoming_reply_mastodon.xml")
73 {:ok, [activity]} = OStatus.handle_incoming(incoming)
74
75 assert activity.data["type"] == "Create"
76 assert activity.data["object"]["type"] == "Note"
77 assert activity.data["object"]["actor"] == "https://mastodon.social/users/lambadalambda"
78 assert activity.data["context"] == "2hu"
79 assert "https://www.w3.org/ns/activitystreams#Public" in activity.data["to"]
80 end
81
82 test "handle incoming notes - Mastodon, with CW" do
83 incoming = File.read!("test/fixtures/mastodon-note-cw.xml")
84 {:ok, [activity]} = OStatus.handle_incoming(incoming)
85
86 assert activity.data["type"] == "Create"
87 assert activity.data["object"]["type"] == "Note"
88 assert activity.data["object"]["actor"] == "https://mastodon.social/users/lambadalambda"
89 assert activity.data["object"]["summary"] == "technologic"
90 assert "https://www.w3.org/ns/activitystreams#Public" in activity.data["to"]
91 end
92
93 test "handle incoming retweets - Mastodon, with CW" do
94 incoming = File.read!("test/fixtures/cw_retweet.xml")
95 {:ok, [[_activity, retweeted_activity]]} = OStatus.handle_incoming(incoming)
96
97 assert retweeted_activity.data["object"]["summary"] == "Hey."
98 end
99
100 test "handle incoming notes - GS, subscription, reply" do
101 incoming = File.read!("test/fixtures/ostatus_incoming_reply.xml")
102 {:ok, [activity]} = OStatus.handle_incoming(incoming)
103
104 assert activity.data["type"] == "Create"
105 assert activity.data["object"]["type"] == "Note"
106 assert activity.data["object"]["actor"] == "https://social.heldscal.la/user/23211"
107 assert activity.data["object"]["content"] == "@<a href=\"https://gs.archae.me/user/4687\" class=\"h-card u-url p-nickname mention\" title=\"shpbot\">shpbot</a> why not indeed."
108 assert activity.data["object"]["inReplyTo"] == "tag:gs.archae.me,2017-04-30:noticeId=778260:objectType=note"
109 assert "https://www.w3.org/ns/activitystreams#Public" in activity.data["to"]
110 end
111
112 test "handle incoming retweets - GS, subscription" do
113 incoming = File.read!("test/fixtures/share-gs.xml")
114 {:ok, [[activity, retweeted_activity]]} = OStatus.handle_incoming(incoming)
115
116 assert activity.data["type"] == "Announce"
117 assert activity.data["actor"] == "https://social.heldscal.la/user/23211"
118 assert activity.data["object"] == retweeted_activity.data["object"]["id"]
119 assert "https://pleroma.soykaf.com/users/lain" in activity.data["to"]
120 refute activity.local
121
122 retweeted_activity = Repo.get(Activity, retweeted_activity.id)
123 assert retweeted_activity.data["type"] == "Create"
124 assert retweeted_activity.data["actor"] == "https://pleroma.soykaf.com/users/lain"
125 refute retweeted_activity.local
126 assert retweeted_activity.data["object"]["announcement_count"] == 1
127 assert String.contains?(retweeted_activity.data["object"]["content"], "mastodon")
128 refute String.contains?(retweeted_activity.data["object"]["content"], "Test account")
129 end
130
131 test "handle incoming retweets - GS, subscription - local message" do
132 incoming = File.read!("test/fixtures/share-gs-local.xml")
133 note_activity = insert(:note_activity)
134 user = User.get_cached_by_ap_id(note_activity.data["actor"])
135 incoming = incoming
136 |> String.replace("LOCAL_ID", note_activity.data["object"]["id"])
137 |> String.replace("LOCAL_USER", user.ap_id)
138
139 {:ok, [[activity, retweeted_activity]]} = OStatus.handle_incoming(incoming)
140
141 assert activity.data["type"] == "Announce"
142 assert activity.data["actor"] == "https://social.heldscal.la/user/23211"
143 assert activity.data["object"] == retweeted_activity.data["object"]["id"]
144 assert user.ap_id in activity.data["to"]
145 refute activity.local
146
147 retweeted_activity = Repo.get(Activity, retweeted_activity.id)
148 assert note_activity.id == retweeted_activity.id
149 assert retweeted_activity.data["type"] == "Create"
150 assert retweeted_activity.data["actor"] == user.ap_id
151 assert retweeted_activity.local
152 assert retweeted_activity.data["object"]["announcement_count"] == 1
153 end
154
155 test "handle incoming retweets - Mastodon, salmon" do
156 incoming = File.read!("test/fixtures/share.xml")
157 {:ok, [[activity, retweeted_activity]]} = OStatus.handle_incoming(incoming)
158
159 assert activity.data["type"] == "Announce"
160 assert activity.data["actor"] == "https://mastodon.social/users/lambadalambda"
161 assert activity.data["object"] == retweeted_activity.data["object"]["id"]
162 assert activity.data["id"] == "tag:mastodon.social,2017-05-03:objectId=4934452:objectType=Status"
163
164 refute activity.local
165 assert retweeted_activity.data["type"] == "Create"
166 assert retweeted_activity.data["actor"] == "https://pleroma.soykaf.com/users/lain"
167 refute retweeted_activity.local
168 refute String.contains?(retweeted_activity.data["object"]["content"], "Test account")
169 end
170
171 test "handle incoming favorites - GS, websub" do
172 capture_log fn ->
173 incoming = File.read!("test/fixtures/favorite.xml")
174 {:ok, [[activity, favorited_activity]]} = OStatus.handle_incoming(incoming)
175
176 assert activity.data["type"] == "Like"
177 assert activity.data["actor"] == "https://social.heldscal.la/user/23211"
178 assert activity.data["object"] == favorited_activity.data["object"]["id"]
179 assert activity.data["id"] == "tag:social.heldscal.la,2017-05-05:fave:23211:comment:2061643:2017-05-05T09:12:50+00:00"
180
181 refute activity.local
182 assert favorited_activity.data["type"] == "Create"
183 assert favorited_activity.data["actor"] == "https://shitposter.club/user/1"
184 assert favorited_activity.data["object"]["id"] == "tag:shitposter.club,2017-05-05:noticeId=2827873:objectType=comment"
185 refute favorited_activity.local
186 end
187 end
188
189 test "handle conversation references" do
190 incoming = File.read!("test/fixtures/mastodon_conversation.xml")
191 {:ok, [activity]} = OStatus.handle_incoming(incoming)
192
193 assert activity.data["context"] == "tag:mastodon.social,2017-08-28:objectId=7876885:objectType=Conversation"
194 end
195
196 test "handle incoming favorites with locally available object - GS, websub" do
197 note_activity = insert(:note_activity)
198
199 incoming = File.read!("test/fixtures/favorite_with_local_note.xml")
200 |> String.replace("localid", note_activity.data["object"]["id"])
201
202 {:ok, [[activity, favorited_activity]]} = OStatus.handle_incoming(incoming)
203
204 assert activity.data["type"] == "Like"
205 assert activity.data["actor"] == "https://social.heldscal.la/user/23211"
206 assert activity.data["object"] == favorited_activity.data["object"]["id"]
207 refute activity.local
208 assert note_activity.id == favorited_activity.id
209 assert favorited_activity.local
210 end
211
212 test "handle incoming replies" do
213 incoming = File.read!("test/fixtures/incoming_note_activity_answer.xml")
214 {:ok, [activity]} = OStatus.handle_incoming(incoming)
215
216 assert activity.data["type"] == "Create"
217 assert activity.data["object"]["type"] == "Note"
218 assert activity.data["object"]["inReplyTo"] == "http://pleroma.example.org:4000/objects/55bce8fc-b423-46b1-af71-3759ab4670bc"
219 assert "http://pleroma.example.org:4000/users/lain5" in activity.data["to"]
220 assert activity.data["object"]["id"] == "tag:gs.example.org:4040,2017-04-25:noticeId=55:objectType=note"
221 assert "https://www.w3.org/ns/activitystreams#Public" in activity.data["to"]
222 end
223
224 test "handle incoming follows" do
225 incoming = File.read!("test/fixtures/follow.xml")
226 {:ok, [activity]} = OStatus.handle_incoming(incoming)
227 assert activity.data["type"] == "Follow"
228 assert activity.data["id"] == "tag:social.heldscal.la,2017-05-07:subscription:23211:person:44803:2017-05-07T09:54:48+00:00"
229 assert activity.data["actor"] == "https://social.heldscal.la/user/23211"
230 assert activity.data["object"] == "https://pawoo.net/users/pekorino"
231 refute activity.local
232
233 follower = User.get_by_ap_id(activity.data["actor"])
234 followed = User.get_by_ap_id(activity.data["object"])
235
236 assert User.following?(follower, followed)
237 end
238
239 describe "new remote user creation" do
240 test "returns local users" do
241 local_user = insert(:user)
242 {:ok, user} = OStatus.find_or_make_user(local_user.ap_id)
243
244 assert user == local_user
245 end
246
247 test "tries to use the information in poco fields" do
248 uri = "https://social.heldscal.la/user/23211"
249
250 {:ok, user} = OStatus.find_or_make_user(uri)
251
252 user = Repo.get(Pleroma.User, user.id)
253 assert user.name == "Constance Variable"
254 assert user.nickname == "lambadalambda@social.heldscal.la"
255 assert user.local == false
256 assert user.info["uri"] == uri
257 assert user.ap_id == uri
258 assert user.bio == "Call me Deacon Blues."
259 assert user.avatar["type"] == "Image"
260
261 {:ok, user_again} = OStatus.find_or_make_user(uri)
262
263 assert user == user_again
264 end
265
266 test "find_make_or_update_user takes an author element and returns an updated user" do
267 uri = "https://social.heldscal.la/user/23211"
268
269 {:ok, user} = OStatus.find_or_make_user(uri)
270 old_name = user.name
271 old_bio = user.bio
272 change = Ecto.Changeset.change(user, %{avatar: nil, bio: nil, old_name: nil})
273
274 {:ok, user} = Repo.update(change)
275 refute user.avatar
276
277 doc = XML.parse_document(File.read!("test/fixtures/23211.atom"))
278 [author] = :xmerl_xpath.string('//author[1]', doc)
279 {:ok, user} = OStatus.find_make_or_update_user(author)
280 assert user.avatar["type"] == "Image"
281 assert user.name == old_name
282 assert user.bio == old_bio
283
284 {:ok, user_again} = OStatus.find_make_or_update_user(author)
285 assert user_again == user
286 end
287 end
288
289 describe "gathering user info from a user id" do
290 test "it returns user info in a hash" do
291 user = "shp@social.heldscal.la"
292
293 # TODO: make test local
294 {:ok, data} = OStatus.gather_user_info(user)
295
296 expected = %{
297 "hub" => "https://social.heldscal.la/main/push/hub",
298 "magic_key" => "RSA.wQ3i9UA0qmAxZ0WTIp4a-waZn_17Ez1pEEmqmqoooRsG1_BvpmOvLN0G2tEcWWxl2KOtdQMCiPptmQObeZeuj48mdsDZ4ArQinexY2hCCTcbV8Xpswpkb8K05RcKipdg07pnI7tAgQ0VWSZDImncL6YUGlG5YN8b5TjGOwk2VG8=.AQAB",
299 "name" => "shp",
300 "nickname" => "shp",
301 "salmon" => "https://social.heldscal.la/main/salmon/user/29191",
302 "subject" => "acct:shp@social.heldscal.la",
303 "topic" => "https://social.heldscal.la/api/statuses/user_timeline/29191.atom",
304 "uri" => "https://social.heldscal.la/user/29191",
305 "host" => "social.heldscal.la",
306 "fqn" => user,
307 "bio" => "cofe",
308 "avatar" => %{"type" => "Image", "url" => [%{"href" => "https://social.heldscal.la/avatar/29191-original-20170421154949.jpeg", "mediaType" => "image/jpeg", "type" => "Link"}]},
309 "subscribe_address" => "https://social.heldscal.la/main/ostatussub?profile={uri}"
310 }
311 assert data == expected
312 end
313
314 test "it works with the uri" do
315 user = "https://social.heldscal.la/user/29191"
316
317 # TODO: make test local
318 {:ok, data} = OStatus.gather_user_info(user)
319
320 expected = %{
321 "hub" => "https://social.heldscal.la/main/push/hub",
322 "magic_key" => "RSA.wQ3i9UA0qmAxZ0WTIp4a-waZn_17Ez1pEEmqmqoooRsG1_BvpmOvLN0G2tEcWWxl2KOtdQMCiPptmQObeZeuj48mdsDZ4ArQinexY2hCCTcbV8Xpswpkb8K05RcKipdg07pnI7tAgQ0VWSZDImncL6YUGlG5YN8b5TjGOwk2VG8=.AQAB",
323 "name" => "shp",
324 "nickname" => "shp",
325 "salmon" => "https://social.heldscal.la/main/salmon/user/29191",
326 "subject" => "https://social.heldscal.la/user/29191",
327 "topic" => "https://social.heldscal.la/api/statuses/user_timeline/29191.atom",
328 "uri" => "https://social.heldscal.la/user/29191",
329 "host" => "social.heldscal.la",
330 "fqn" => user,
331 "bio" => "cofe",
332 "avatar" => %{"type" => "Image", "url" => [%{"href" => "https://social.heldscal.la/avatar/29191-original-20170421154949.jpeg", "mediaType" => "image/jpeg", "type" => "Link"}]},
333 "subscribe_address" => "https://social.heldscal.la/main/ostatussub?profile={uri}"
334 }
335 assert data == expected
336 end
337 end
338
339 describe "fetching a status by it's HTML url" do
340 test "it builds a missing status from an html url" do
341 capture_log fn ->
342 url = "https://shitposter.club/notice/2827873"
343 {:ok, [activity] } = OStatus.fetch_activity_from_url(url)
344
345 assert activity.data["actor"] == "https://shitposter.club/user/1"
346 assert activity.data["object"]["id"] == "tag:shitposter.club,2017-05-05:noticeId=2827873:objectType=comment"
347 end
348 end
349
350 test "it works for atom notes, too" do
351 url = "https://social.sakamoto.gq/objects/0ccc1a2c-66b0-4305-b23a-7f7f2b040056"
352 {:ok, [activity] } = OStatus.fetch_activity_from_url(url)
353 assert activity.data["actor"] == "https://social.sakamoto.gq/users/eal"
354 assert activity.data["object"]["id"] == url
355 end
356 end
357
358 test "it doesn't add nil in the do field" do
359 incoming = File.read!("test/fixtures/nil_mention_entry.xml")
360 {:ok, [activity]} = OStatus.handle_incoming(incoming)
361
362 assert activity.data["to"] == ["http://localhost:4001/users/atarifrosch@social.stopwatchingus-heidelberg.de/followers", "https://www.w3.org/ns/activitystreams#Public"]
363 end
364 end