31c5962e2d0f4023e2922f0d58c34d302d523a44
[akkoma] / test / user_test.exs
1 defmodule Pleroma.UserTest do
2 alias Pleroma.Builders.UserBuilder
3 alias Pleroma.{User, Repo}
4 alias Pleroma.Web.OStatus
5 alias Pleroma.Web.Websub.WebsubClientSubscription
6 alias Pleroma.Web.CommonAPI
7 use Pleroma.DataCase
8
9 import Pleroma.Factory
10 import Ecto.Query
11
12 test "ap_id returns the activity pub id for the user" do
13 user = UserBuilder.build
14
15 expected_ap_id = "#{Pleroma.Web.base_url}/users/#{user.nickname}"
16
17 assert expected_ap_id == User.ap_id(user)
18 end
19
20 test "ap_followers returns the followers collection for the user" do
21 user = UserBuilder.build
22
23 expected_followers_collection = "#{User.ap_id(user)}/followers"
24
25 assert expected_followers_collection == User.ap_followers(user)
26 end
27
28 test "follow takes a user and another user" do
29 user = insert(:user)
30 followed = insert(:user)
31
32 {:ok, user} = User.follow(user, followed)
33
34 user = Repo.get(User, user.id)
35
36 followed = User.get_by_ap_id(followed.ap_id)
37 assert followed.info["follower_count"] == 1
38
39 assert User.ap_followers(followed) in user.following
40 end
41
42 test "following a remote user will ensure a websub subscription is present" do
43 user = insert(:user)
44 {:ok, followed} = OStatus.make_user("shp@social.heldscal.la")
45
46 assert followed.local == false
47
48 {:ok, user} = User.follow(user, followed)
49 assert User.ap_followers(followed) in user.following
50
51 query = from w in WebsubClientSubscription,
52 where: w.topic == ^followed.info["topic"]
53 websub = Repo.one(query)
54
55 assert websub
56 end
57
58 test "unfollow takes a user and another user" do
59 followed = insert(:user)
60 user = insert(:user, %{following: [User.ap_followers(followed)]})
61
62 {:ok, user, _activity } = User.unfollow(user, followed)
63
64 user = Repo.get(User, user.id)
65
66 assert user.following == []
67 end
68
69 test "unfollow doesn't unfollow yourself" do
70 user = insert(:user)
71
72 {:error, _} = User.unfollow(user, user)
73
74 user = Repo.get(User, user.id)
75 assert user.following == [user.ap_id]
76 end
77
78
79 test "test if a user is following another user" do
80 followed = insert(:user)
81 user = insert(:user, %{following: [User.ap_followers(followed)]})
82
83 assert User.following?(user, followed)
84 refute User.following?(followed, user)
85 end
86
87 describe "user registration" do
88 @full_user_data %{
89 bio: "A guy",
90 name: "my name",
91 nickname: "nick",
92 password: "test",
93 password_confirmation: "test",
94 email: "email@example.com"
95 }
96
97 test "it requires a bio, email, name, nickname and password" do
98 @full_user_data
99 |> Map.keys
100 |> Enum.each(fn (key) ->
101 params = Map.delete(@full_user_data, key)
102 changeset = User.register_changeset(%User{}, params)
103 assert changeset.valid? == false
104 end)
105 end
106
107 test "it sets the password_hash, ap_id and following fields" do
108 changeset = User.register_changeset(%User{}, @full_user_data)
109
110 assert changeset.valid?
111
112 assert is_binary(changeset.changes[:password_hash])
113 assert changeset.changes[:ap_id] == User.ap_id(%User{nickname: @full_user_data.nickname})
114 assert changeset.changes[:following] == [User.ap_followers(%User{nickname: @full_user_data.nickname})]
115 assert changeset.changes.follower_address == "#{changeset.changes.ap_id}/followers"
116 end
117 end
118
119 describe "fetching a user from nickname or trying to build one" do
120 test "gets an existing user" do
121 user = insert(:user)
122 fetched_user = User.get_or_fetch_by_nickname(user.nickname)
123
124 assert user == fetched_user
125 end
126
127 test "gets an existing user, case insensitive" do
128 user = insert(:user, nickname: "nick")
129 fetched_user = User.get_or_fetch_by_nickname("NICK")
130
131 assert user == fetched_user
132 end
133
134 test "fetches an external user via ostatus if no user exists" do
135 fetched_user = User.get_or_fetch_by_nickname("shp@social.heldscal.la")
136 assert fetched_user.nickname == "shp@social.heldscal.la"
137 end
138
139 test "returns nil if no user could be fetched" do
140 fetched_user = User.get_or_fetch_by_nickname("nonexistant@social.heldscal.la")
141 assert fetched_user == nil
142 end
143
144 test "returns nil for nonexistant local user" do
145 fetched_user = User.get_or_fetch_by_nickname("nonexistant")
146 assert fetched_user == nil
147 end
148 end
149
150 test "returns an ap_id for a user" do
151 user = insert(:user)
152 assert User.ap_id(user) == Pleroma.Web.Router.Helpers.o_status_url(Pleroma.Web.Endpoint, :feed_redirect, user.nickname)
153 end
154
155 test "returns an ap_followers link for a user" do
156 user = insert(:user)
157 assert User.ap_followers(user) == Pleroma.Web.Router.Helpers.o_status_url(Pleroma.Web.Endpoint, :feed_redirect, user.nickname) <> "/followers"
158 end
159
160 describe "remote user creation changeset" do
161 @valid_remote %{
162 bio: "hello",
163 name: "Someone",
164 nickname: "a@b.de",
165 ap_id: "http...",
166 info: %{some: "info"},
167 avatar: %{some: "avatar"}
168 }
169
170 test "it confirms validity" do
171 cs = User.remote_user_creation(@valid_remote)
172 assert cs.valid?
173 end
174
175 test "it sets the follower_adress" do
176 cs = User.remote_user_creation(@valid_remote)
177 # remote users get a fake local follower address
178 assert cs.changes.follower_address == User.ap_followers(%User{ nickname: @valid_remote[:nickname] })
179 end
180
181 test "it enforces the fqn format for nicknames" do
182 cs = User.remote_user_creation(%{@valid_remote | nickname: "bla"})
183 assert cs.changes.local == false
184 assert cs.changes.avatar
185 refute cs.valid?
186 end
187
188 test "it has required fields" do
189 [:name, :nickname, :ap_id]
190 |> Enum.each(fn (field) ->
191 cs = User.remote_user_creation(Map.delete(@valid_remote, field))
192 refute cs.valid?
193 end)
194 end
195
196 test "it restricts some sizes" do
197 [bio: 5000, name: 100]
198 |> Enum.each(fn ({field, size}) ->
199 string = String.pad_leading(".", size)
200 cs = User.remote_user_creation(Map.put(@valid_remote, field, string))
201 assert cs.valid?
202
203 string = String.pad_leading(".", size + 1)
204 cs = User.remote_user_creation(Map.put(@valid_remote, field, string))
205 refute cs.valid?
206 end)
207 end
208 end
209
210 describe "followers and friends" do
211 test "gets all followers for a given user" do
212 user = insert(:user)
213 follower_one = insert(:user)
214 follower_two = insert(:user)
215 not_follower = insert(:user)
216
217 {:ok, follower_one} = User.follow(follower_one, user)
218 {:ok, follower_two} = User.follow(follower_two, user)
219
220 {:ok, res} = User.get_followers(user)
221
222 assert Enum.member?(res, follower_one)
223 assert Enum.member?(res, follower_two)
224 refute Enum.member?(res, not_follower)
225 end
226
227 test "gets all friends (followed users) for a given user" do
228 user = insert(:user)
229 followed_one = insert(:user)
230 followed_two = insert(:user)
231 not_followed = insert(:user)
232
233 {:ok, user} = User.follow(user, followed_one)
234 {:ok, user} = User.follow(user, followed_two)
235
236 {:ok, res} = User.get_friends(user)
237
238 followed_one = User.get_by_ap_id(followed_one.ap_id)
239 followed_two = User.get_by_ap_id(followed_two.ap_id)
240 assert Enum.member?(res, followed_one)
241 assert Enum.member?(res, followed_two)
242 refute Enum.member?(res, not_followed)
243 end
244 end
245
246 describe "updating note and follower count" do
247 test "it sets the info->note_count property" do
248 note = insert(:note)
249
250 user = User.get_by_ap_id(note.data["actor"])
251
252 assert user.info["note_count"] == nil
253
254 {:ok, user} = User.update_note_count(user)
255
256 assert user.info["note_count"] == 1
257 end
258
259 test "it increases the info->note_count property" do
260 note = insert(:note)
261 user = User.get_by_ap_id(note.data["actor"])
262
263 assert user.info["note_count"] == nil
264
265 {:ok, user} = User.increase_note_count(user)
266
267 assert user.info["note_count"] == 1
268
269 {:ok, user} = User.increase_note_count(user)
270
271 assert user.info["note_count"] == 2
272 end
273
274 test "it sets the info->follower_count property" do
275 user = insert(:user)
276 follower = insert(:user)
277
278 User.follow(follower, user)
279
280 assert user.info["follower_count"] == nil
281
282 {:ok, user} = User.update_follower_count(user)
283
284 assert user.info["follower_count"] == 1
285 end
286 end
287
288 describe "blocks" do
289 test "it blocks people" do
290 user = insert(:user)
291 blocked_user = insert(:user)
292
293 refute User.blocks?(user, blocked_user)
294
295 {:ok, user} = User.block(user, blocked_user)
296
297 assert User.blocks?(user, blocked_user)
298 end
299
300 test "it unblocks users" do
301 user = insert(:user)
302 blocked_user = insert(:user)
303
304 {:ok, user} = User.block(user, blocked_user)
305 {:ok, user} = User.unblock(user, blocked_user)
306
307 refute User.blocks?(user, blocked_user)
308 end
309 end
310
311 test "get recipients from activity" do
312 actor = insert(:user)
313 user = insert(:user, local: true)
314 user_two = insert(:user, local: false)
315 addressed = insert(:user, local: true)
316 addressed_remote = insert(:user, local: false)
317 {:ok, activity} = CommonAPI.post(actor, %{"status" => "hey @#{addressed.nickname} @#{addressed_remote.nickname}"})
318
319 assert [addressed] == User.get_recipients_from_activity(activity)
320
321 {:ok, user} = User.follow(user, actor)
322 {:ok, user_two} = User.follow(user_two, actor)
323 recipients = User.get_recipients_from_activity(activity)
324 assert length(recipients) == 2
325 assert user in recipients
326 assert addressed in recipients
327 end
328 end
329