]> git.openstreetmap.org Git - rails.git/blob - test/models/user_test.rb
Avoid highlighting multiple changesets in history sidebar
[rails.git] / test / models / user_test.rb
1 require "test_helper"
2
3 class UserTest < ActiveSupport::TestCase
4   include Rails::Dom::Testing::Assertions::SelectorAssertions
5
6   def test_invalid_with_empty_attributes
7     user = build(:user, :email => nil,
8                         :pass_crypt => nil,
9                         :display_name => nil,
10                         :home_lat => nil,
11                         :home_lon => nil,
12                         :home_zoom => nil)
13     assert_not_predicate user, :valid?
14     assert_predicate user.errors[:email], :any?
15     assert_predicate user.errors[:pass_crypt], :any?
16     assert_predicate user.errors[:display_name], :any?
17     assert_predicate user.errors[:home_lat], :none?
18     assert_predicate user.errors[:home_lon], :none?
19     assert_predicate user.errors[:home_zoom], :none?
20   end
21
22   def test_unique_email
23     existing_user = create(:user)
24     new_user = build(:user, :email => existing_user.email)
25     assert_not new_user.save
26     assert_includes new_user.errors[:email], "has already been taken"
27   end
28
29   def test_unique_display_name
30     create(:user, :display_name => "H\u{e9}nryIV")
31
32     %W[H\u{e9}nryIV he\u{301}nryiv H\u{c9}nry\u2163 he\u{301}nry\u2173].each do |name|
33       new_user = build(:user, :display_name => name)
34       assert_not new_user.save
35       assert_includes new_user.errors[:display_name], "has already been taken"
36     end
37   end
38
39   def test_email_valid
40     ok = %w[a@s.com test@shaunmcdonald.me.uk hello_local@ping-d.ng
41             test_local@openstreetmap.org test-local@example.com]
42     bad = %w[hi ht@ n@ @.com help@.me.uk help"hi.me.uk も対@応します
43              輕觸搖晃的遊戲@ah.com も対応します@s.name]
44
45     ok.each do |name|
46       user = build(:user)
47       user.email = name
48       assert user.valid?(:save), "#{name} isn't valid when it should be"
49     end
50
51     bad.each do |name|
52       user = build(:user)
53       user.email = name
54       assert user.invalid?(:save), "#{name} is valid when it shouldn't be"
55     end
56   end
57
58   def test_display_name_length
59     user = build(:user)
60     user.display_name = "123"
61     assert_predicate user, :valid?, "should allow 3 char name name"
62     user.display_name = "12"
63     assert_not_predicate user, :valid?, "should not allow 2 char name"
64     user.display_name = ""
65     assert_not_predicate user, :valid?, "should not allow blank/0 char name"
66     user.display_name = nil
67     assert_not_predicate user, :valid?, "should not allow nil value"
68   end
69
70   def test_display_name_width
71     user = build(:user)
72     user.display_name = "123"
73     assert_predicate user, :valid?, "should allow 3 column name name"
74     user.display_name = "12"
75     assert_not_predicate user, :valid?, "should not allow 2 column name"
76     user.display_name = "1\u{200B}2"
77     assert_not_predicate user, :valid?, "should not allow 2 column name"
78     user.display_name = "\u{200B}\u{200B}\u{200B}"
79     assert_not_predicate user, :valid?, "should not allow 0 column name"
80   end
81
82   def test_display_name_valid
83     # Due to sanitisation in the view some of these that you might not
84     # expect are allowed
85     # However, would they affect the xml planet dumps?
86     ok = ["Name", "'me", "he\"", "<hr>", "*ho", "\"help\"@",
87           "vergrößern", "ルシステムにも対応します", "輕觸搖晃的遊戲", "space space"]
88     # These need to be 3 chars in length, otherwise the length test above
89     # should be used.
90     bad = ["<hr/>", "test@example.com", "s/f", "aa/", "aa;", "aa.",
91            "aa,", "aa?", "/;.,?", "も対応します/", "#ping",
92            "foo\x1fbar", "foo\x7fbar", "foo\ufffebar", "foo\uffffbar",
93            "new", "terms", "save", "confirm", "confirm-email",
94            "go_public", "reset-password", "forgot-password", "suspended",
95            "trailing whitespace ", " leading whitespace"]
96     ok.each do |display_name|
97       user = build(:user)
98       user.display_name = display_name
99       assert_predicate user, :valid?, "#{display_name} is invalid, when it should be"
100     end
101
102     bad.each do |display_name|
103       user = build(:user)
104       user.display_name = display_name
105       assert_not_predicate user, :valid?, "#{display_name} is valid when it shouldn't be"
106     end
107   end
108
109   def test_display_name_user_id_new
110     existing_user = create(:user)
111     user = build(:user)
112
113     user.display_name = "user_#{existing_user.id}"
114     assert_not_predicate user, :valid?, "user_<id> name is valid for existing user id when it shouldn't be"
115
116     user.display_name = "user_#{existing_user.id + 1}"
117     assert_not_predicate user, :valid?, "user_<id> name is valid for new user id when it shouldn't be"
118   end
119
120   def test_display_name_user_id_rename
121     existing_user = create(:user)
122     user = create(:user)
123
124     user.display_name = "user_#{existing_user.id}"
125     assert_not_predicate user, :valid?, "user_<id> name is valid for existing user id when it shouldn't be"
126
127     user.display_name = "user_#{user.id}"
128     assert_predicate user, :valid?, "user_<id> name is invalid for own id, when it should be"
129   end
130
131   def test_display_name_user_id_unchanged_is_valid
132     user = build(:user, :display_name => "user_0")
133     user.save(:validate => false)
134     user.reload
135
136     assert_predicate user, :valid?, "user_0 display_name is invalid but it hasn't been changed"
137   end
138
139   def test_description_length
140     user = build(:user)
141     user.description = "x" * 65536
142     assert_predicate user, :valid?, "should allow 65536 char description"
143     user.description = "x" * 65537
144     assert_not_predicate user, :valid?, "should not allow 65537 char description"
145     user.description = ""
146     assert_predicate user, :valid?, "should allow blank/0 char description"
147     user.description = nil
148     assert_predicate user, :valid?, "should allow nil value"
149   end
150
151   def test_follows
152     alice = create(:user, :active)
153     bob = create(:user, :active)
154     charlie = create(:user, :active)
155     create(:follow, :follower => alice, :following => bob)
156
157     assert alice.follows?(bob)
158     assert_not alice.follows?(charlie)
159     assert_not bob.follows?(alice)
160     assert_not bob.follows?(charlie)
161     assert_not charlie.follows?(bob)
162     assert_not charlie.follows?(alice)
163   end
164
165   def test_users_nearby
166     alice = create(:user, :active, :home_lat => 51.0, :home_lon => 1.0, :data_public => false)
167     bob = create(:user, :active, :home_lat => 51.1, :home_lon => 1.0, :data_public => true)
168     charlie = create(:user, :active, :home_lat => 51.1, :home_lon => 1.1, :data_public => true)
169     david = create(:user, :active, :home_lat => 10.0, :home_lon => -123.0, :data_public => true)
170     _edward = create(:user, :suspended, :home_lat => 10.0, :home_lon => -123.0, :data_public => true)
171     south_pole_user = create(:user, :active, :home_lat => -90.0, :home_lon => 0.0, :data_public => true)
172     vagrant_user = create(:user, :active, :home_lat => nil, :home_lon => nil, :data_public => true)
173
174     # bob and charlie are both near alice
175     assert_equal [bob, charlie], alice.nearby
176     # charlie and alice are both near bob, but alice has their data private
177     assert_equal [charlie], bob.nearby
178     # david has no user nearby, since edward is not active
179     assert_empty david.nearby
180     # south_pole_user has no user nearby, and doesn't throw exception
181     assert_empty south_pole_user.nearby
182     # vagrant_user has no home location
183     assert_empty vagrant_user.nearby
184   end
185
186   def test_friends
187     norm = create(:user, :active)
188     sec = create(:user, :active)
189     create(:follow, :follower => norm, :following => sec)
190
191     assert_equal [sec], norm.followings
192     assert_equal 1, norm.followings.size
193
194     assert_empty sec.followings
195     assert_equal 0, sec.followings.size
196   end
197
198   def test_user_preferred_editor
199     user = create(:user)
200     assert_nil user.preferred_editor
201     user.preferred_editor = "id"
202     assert_equal "id", user.preferred_editor
203     user.save!
204
205     user.preferred_editor = "invalid_editor"
206     assert_raise(ActiveRecord::RecordInvalid) { user.save! }
207   end
208
209   def test_visible
210     pending = create(:user, :pending)
211     active = create(:user, :active)
212     confirmed = create(:user, :confirmed)
213     suspended = create(:user, :suspended)
214     deleted = create(:user, :deleted)
215
216     assert User.visible.find(pending.id)
217     assert User.visible.find(active.id)
218     assert User.visible.find(confirmed.id)
219     assert_raise ActiveRecord::RecordNotFound do
220       User.visible.find(suspended.id)
221     end
222     assert_raise ActiveRecord::RecordNotFound do
223       User.visible.find(deleted.id)
224     end
225   end
226
227   def test_active
228     pending = create(:user, :pending)
229     active = create(:user, :active)
230     confirmed = create(:user, :confirmed)
231     suspended = create(:user, :suspended)
232     deleted = create(:user, :deleted)
233
234     assert User.active.find(active.id)
235     assert User.active.find(confirmed.id)
236     assert_raise ActiveRecord::RecordNotFound do
237       User.active.find(pending.id)
238     end
239     assert_raise ActiveRecord::RecordNotFound do
240       User.active.find(suspended.id)
241     end
242     assert_raise ActiveRecord::RecordNotFound do
243       User.active.find(deleted.id)
244     end
245   end
246
247   def test_identifiable
248     public_user = create(:user, :data_public => true)
249     private_user = create(:user, :data_public => false)
250
251     assert User.identifiable.find(public_user.id)
252     assert_raise ActiveRecord::RecordNotFound do
253       User.identifiable.find(private_user.id)
254     end
255   end
256
257   def test_languages
258     user = create(:user, :languages => ["en"])
259     assert_equal ["en"], user.languages
260     user.languages = %w[de fr en]
261     assert_equal %w[de fr en], user.languages
262     user.languages = %w[fr de sl]
263     assert_equal %w[fr de sl], user.preferred_languages.map(&:to_s)
264     user = create(:user, :languages => %w[en de])
265     assert_equal %w[en de], user.languages
266   end
267
268   def test_default_diary_language_undefined
269     create(:language, :code => "en")
270     user = create(:user, :languages => [])
271     assert_nil user.default_diary_language
272   end
273
274   def test_default_diary_language_known
275     create(:language, :code => "en")
276     user = create(:user, :languages => ["en"])
277     assert_equal "en", user.default_diary_language
278   end
279
280   def test_default_diary_language_known_with_fallback
281     create(:language, :code => "en")
282     create(:language, :code => "fr")
283     user = create(:user, :languages => ["fr en"])
284     assert_equal "fr", user.default_diary_language
285   end
286
287   def test_default_diary_language_unknown
288     create(:language, :code => "en")
289     user = create(:user, :languages => ["unknown"])
290     assert_nil user.default_diary_language
291   end
292
293   def test_default_diary_language_unknown_with_known_fallback
294     create(:language, :code => "en")
295     user = create(:user, :languages => ["unknown en"])
296     assert_equal "en", user.default_diary_language
297   end
298
299   def test_default_diary_language_set
300     create(:language, :code => "en")
301     user = create(:user, :languages => [])
302
303     assert_difference "user.preferences.count", 1 do
304       assert_equal "en", (user.default_diary_language = "en")
305     end
306
307     user.reload
308     assert_equal "en", user.default_diary_language
309     preference = user.preferences.find_by(:k => "diary.default_language")
310     assert_equal "en", preference.v
311   end
312
313   def test_default_diary_language_set_twice
314     create(:language, :code => "en")
315     create(:language, :code => "fr")
316     user = create(:user, :languages => [])
317
318     assert_difference "user.preferences.count", 1 do
319       assert_equal "en", (user.default_diary_language = "en")
320     end
321     assert_difference "user.preferences.count", 0 do
322       assert_equal "fr", (user.default_diary_language = "fr")
323     end
324
325     user.reload
326     assert_equal "fr", user.default_diary_language
327     preference = user.preferences.find_by(:k => "diary.default_language")
328     assert_equal "fr", preference.v
329   end
330
331   def test_visible?
332     assert_predicate build(:user, :pending), :visible?
333     assert_predicate build(:user, :active), :visible?
334     assert_predicate build(:user, :confirmed), :visible?
335     assert_not_predicate build(:user, :suspended), :visible?
336     assert_not_predicate build(:user, :deleted), :visible?
337   end
338
339   def test_active?
340     assert_not_predicate build(:user, :pending), :active?
341     assert_predicate build(:user, :active), :active?
342     assert_predicate build(:user, :confirmed), :active?
343     assert_not_predicate build(:user, :suspended), :active?
344     assert_not_predicate build(:user, :deleted), :active?
345   end
346
347   def test_moderator?
348     assert_not_predicate create(:user), :moderator?
349     assert_predicate create(:moderator_user), :moderator?
350   end
351
352   def test_administrator?
353     assert_not_predicate create(:user), :administrator?
354     assert_predicate create(:administrator_user), :administrator?
355   end
356
357   def test_role?
358     assert_not create(:user).role?("administrator")
359     assert_not create(:user).role?("moderator")
360     assert create(:administrator_user).role?("administrator")
361     assert create(:moderator_user).role?("moderator")
362   end
363
364   def test_soft_destroy
365     user = create(:user, :with_home_location, :description => "foo")
366     user.soft_destroy
367     assert_equal "user_#{user.id}", user.display_name
368     assert_predicate user.description, :blank?
369     assert_nil user.home_lat
370     assert_nil user.home_lon
371     assert_not_predicate user.avatar, :attached?
372     assert_equal "deleted", user.status
373     assert_not_predicate user, :visible?
374     assert_not_predicate user, :active?
375   end
376
377   def test_soft_destroy_revokes_oauth2_tokens
378     user = create(:user)
379     oauth_access_token = create(:oauth_access_token, :user => user)
380     assert_equal 1, user.access_tokens.not_expired.count
381
382     user.soft_destroy
383
384     assert_equal 0, user.access_tokens.not_expired.count
385     oauth_access_token.reload
386     assert_predicate oauth_access_token, :revoked?
387   end
388
389   def test_deletion_allowed_when_no_changesets
390     with_user_account_deletion_delay(10000) do
391       user = create(:user)
392       assert_predicate user, :deletion_allowed?
393     end
394   end
395
396   def test_deletion_allowed_without_delay
397     with_user_account_deletion_delay(nil) do
398       user = create(:user)
399       create(:changeset, :user => user)
400       user.reload
401       assert_predicate user, :deletion_allowed?
402     end
403   end
404
405   def test_deletion_allowed_past_delay
406     with_user_account_deletion_delay(10) do
407       user = create(:user)
408       create(:changeset, :user => user, :created_at => Time.now.utc - 12.hours, :closed_at => Time.now.utc - 10.hours)
409       user.reload
410       assert_predicate user, :deletion_allowed?
411     end
412   end
413
414   def test_deletion_allowed_during_delay
415     with_user_account_deletion_delay(10) do
416       user = create(:user)
417       create(:changeset, :user => user, :created_at => Time.now.utc - 11.hours, :closed_at => Time.now.utc - 9.hours)
418       user.reload
419       assert_not_predicate user, :deletion_allowed?
420       assert_equal Time.now.utc + 1.hour, user.deletion_allowed_at
421     end
422   end
423
424   def test_deletion_allowed_past_zero_delay
425     with_user_account_deletion_delay(0) do
426       user = create(:user)
427       create(:changeset, :user => user, :created_at => Time.now.utc, :closed_at => Time.now.utc + 1.hour)
428       travel 90.minutes do
429         user.reload
430         assert_predicate user, :deletion_allowed?
431       end
432     end
433   end
434
435   def test_deletion_allowed_during_zero_delay
436     with_user_account_deletion_delay(0) do
437       user = create(:user)
438       create(:changeset, :user => user, :created_at => Time.now.utc, :closed_at => Time.now.utc + 1.hour)
439       travel 30.minutes do
440         user.reload
441         assert_not_predicate user, :deletion_allowed?
442         assert_equal Time.now.utc + 30.minutes, user.deletion_allowed_at
443       end
444     end
445   end
446 end