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