2019-04-30 08:27:42 +08:00
|
|
|
|
# frozen_string_literal: true
|
|
|
|
|
|
2015-10-11 17:41:23 +08:00
|
|
|
|
require 'rails_helper'
|
2013-02-08 22:52:56 +08:00
|
|
|
|
|
|
|
|
|
describe UsernameValidator do
|
2019-04-23 18:22:47 +08:00
|
|
|
|
def expect_valid(*usernames)
|
|
|
|
|
usernames.each do |username|
|
|
|
|
|
validator = UsernameValidator.new(username)
|
|
|
|
|
|
|
|
|
|
aggregate_failures do
|
|
|
|
|
expect(validator.valid_format?).to eq(true), "expected '#{username}' to be valid"
|
|
|
|
|
expect(validator.errors).to be_empty
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
def expect_invalid(*usernames, error_message:)
|
|
|
|
|
usernames.each do |username|
|
|
|
|
|
validator = UsernameValidator.new(username)
|
|
|
|
|
|
|
|
|
|
aggregate_failures do
|
|
|
|
|
expect(validator.valid_format?).to eq(false), "expected '#{username}' to be invalid"
|
|
|
|
|
expect(validator.errors).to include(error_message)
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
shared_examples 'ASCII username' do
|
|
|
|
|
it 'is invalid when the username is blank' do
|
|
|
|
|
expect_invalid('', error_message: I18n.t(:'user.username.blank'))
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
it 'is invalid when the username is too short' do
|
|
|
|
|
SiteSetting.min_username_length = 4
|
|
|
|
|
|
|
|
|
|
expect_invalid('a', 'ab', 'abc',
|
|
|
|
|
error_message: I18n.t(:'user.username.short', min: 4))
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
it 'is valid when the username has the minimum lenght' do
|
|
|
|
|
SiteSetting.min_username_length = 4
|
|
|
|
|
|
|
|
|
|
expect_valid('abcd')
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
it 'is invalid when the username is too long' do
|
|
|
|
|
SiteSetting.max_username_length = 8
|
|
|
|
|
|
|
|
|
|
expect_invalid('abcdefghi',
|
|
|
|
|
error_message: I18n.t(:'user.username.long', max: 8))
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
it 'is valid when the username has the maximum lenght' do
|
|
|
|
|
SiteSetting.max_username_length = 8
|
|
|
|
|
|
|
|
|
|
expect_valid('abcdefgh')
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
it 'is valid when the username contains alphanumeric characters, dots, underscores and dashes' do
|
|
|
|
|
expect_valid('ab-cd.123_ABC-xYz')
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
it 'is invalid when the username contains non-alphanumeric characters other than dots, underscores and dashes' do
|
|
|
|
|
expect_invalid('abc|', 'a#bc', 'abc xyz',
|
|
|
|
|
error_message: I18n.t(:'user.username.characters'))
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
it 'is valid when the username starts with a alphanumeric character or underscore' do
|
|
|
|
|
expect_valid('abcd', '1abc', '_abc')
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
it 'is invalid when the username starts with a dot or dash' do
|
|
|
|
|
expect_invalid('.abc', '-abc',
|
|
|
|
|
error_message: I18n.t(:'user.username.must_begin_with_alphanumeric_or_underscore'))
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
it 'is valid when the username ends with a alphanumeric character' do
|
|
|
|
|
expect_valid('abcd', 'abc9')
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
it 'is invalid when the username ends with an underscore, a dot or dash' do
|
|
|
|
|
expect_invalid('abc_', 'abc.', 'abc-',
|
|
|
|
|
error_message: I18n.t(:'user.username.must_end_with_alphanumeric'))
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
it 'is invalid when the username contains consecutive underscores, dots or dashes' do
|
|
|
|
|
expect_invalid('a__bc', 'a..bc', 'a--bc',
|
|
|
|
|
error_message: I18n.t(:'user.username.must_not_contain_two_special_chars_in_seq'))
|
2013-02-08 22:52:56 +08:00
|
|
|
|
end
|
|
|
|
|
|
2019-04-23 18:22:47 +08:00
|
|
|
|
it 'is invalid when the username ends with certain file extensions' do
|
|
|
|
|
expect_invalid('abc.json', 'abc.png',
|
|
|
|
|
error_message: I18n.t(:'user.username.must_not_end_with_confusing_suffix'))
|
2013-02-08 22:52:56 +08:00
|
|
|
|
end
|
2019-04-23 18:22:47 +08:00
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
context 'when Unicode usernames are disabled' do
|
|
|
|
|
before { SiteSetting.unicode_usernames = false }
|
|
|
|
|
|
|
|
|
|
include_examples 'ASCII username'
|
|
|
|
|
|
|
|
|
|
it 'is invalid when the username contains non-ASCII characters except dots, underscores and dashes' do
|
|
|
|
|
expect_invalid('abcö', 'abc象',
|
|
|
|
|
error_message: I18n.t(:'user.username.characters'))
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
context 'when Unicode usernames are enabled' do
|
|
|
|
|
before { SiteSetting.unicode_usernames = true }
|
|
|
|
|
|
|
|
|
|
context "ASCII usernames" do
|
|
|
|
|
include_examples 'ASCII username'
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
context "Unicode usernames" do
|
|
|
|
|
before { SiteSetting.min_username_length = 1 }
|
|
|
|
|
|
|
|
|
|
it 'is invalid when the username is too short' do
|
|
|
|
|
SiteSetting.min_username_length = 3
|
|
|
|
|
|
|
|
|
|
expect_invalid('鳥', 'পাখি',
|
|
|
|
|
error_message: I18n.t(:'user.username.short', min: 3))
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
it 'is valid when the username has the minimum lenght' do
|
|
|
|
|
SiteSetting.min_username_length = 2
|
|
|
|
|
|
|
|
|
|
expect_valid('পাখি', 'طائر')
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
it 'is invalid when the username is too long' do
|
|
|
|
|
SiteSetting.max_username_length = 8
|
|
|
|
|
|
|
|
|
|
expect_invalid('חוטב_עצים', 'Holzfäller',
|
|
|
|
|
error_message: I18n.t(:'user.username.long', max: 8))
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
it 'is valid when the username has the maximum lenght' do
|
|
|
|
|
SiteSetting.max_username_length = 9
|
|
|
|
|
|
|
|
|
|
expect_valid('Дровосек', 'چوب-لباسی', 'தமிழ்-தமிழ்')
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
it 'is invalid when the username has too many Unicode codepoints' do
|
|
|
|
|
SiteSetting.max_username_length = 30
|
|
|
|
|
|
|
|
|
|
expect_invalid('য়ায়ায়ায়ায়ায়ায়ায়ায়ায়ায়ায়ায়ায়ায়ায়ায়ায়ায়ায়ায়ায়ায়ায়ায়া',
|
|
|
|
|
error_message: I18n.t(:'user.username.too_long'))
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
it 'is valid when the username contains Unicode letters' do
|
|
|
|
|
expect_valid('鳥', 'طائر', 'թռչուն', 'πουλί', 'পাখি', 'madár', '새',
|
|
|
|
|
'پرنده', 'птица', 'fågel', 'นก', 'پرندے', 'ציפור')
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
it 'is valid when the username contains numbers from the Nd or Nl Unicode category' do
|
|
|
|
|
expect_valid('arabic٠١٢٣٤٥٦٧٨٩', 'bengali০১২৩৪৫৬৭৮৯', 'romanⅥ', 'hangzhou〺')
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
it 'is invalid when the username contains numbers from the No Unicode category' do
|
|
|
|
|
expect_invalid('circled㊸', 'fraction¾',
|
|
|
|
|
error_message: I18n.t(:'user.username.characters'))
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
it 'is invalid when the username contains symbols or emojis' do
|
|
|
|
|
SiteSetting.min_username_length = 1
|
|
|
|
|
|
|
|
|
|
expect_invalid('©', '⇨', '“', '±', '‿', '😃', '🚗',
|
|
|
|
|
error_message: I18n.t(:'user.username.characters'))
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
it 'is invalid when the username contains zero width join characters' do
|
|
|
|
|
expect_invalid('ണ്', 'র্যাম',
|
|
|
|
|
error_message: I18n.t(:'user.username.characters'))
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
it 'is valid when the username ends with a Unicode Mark' do
|
|
|
|
|
expect_valid('தமிழ்')
|
|
|
|
|
end
|
|
|
|
|
|
2020-07-27 08:23:54 +08:00
|
|
|
|
it 'allows all Unicode letters when the allowlist is empty' do
|
2019-04-23 18:22:47 +08:00
|
|
|
|
expect_valid('鳥')
|
|
|
|
|
end
|
|
|
|
|
|
2020-07-27 08:23:54 +08:00
|
|
|
|
context "with Unicode allowlist" do
|
|
|
|
|
before { SiteSetting.allowed_unicode_username_characters = "[äöüÄÖÜß]" }
|
2019-04-23 18:22:47 +08:00
|
|
|
|
|
2020-07-27 08:23:54 +08:00
|
|
|
|
it 'is invalid when username contains non-allowlisted letters' do
|
2019-04-23 18:22:47 +08:00
|
|
|
|
expect_invalid('鳥', 'francès', error_message: I18n.t(:'user.username.characters'))
|
|
|
|
|
end
|
|
|
|
|
|
2020-07-27 08:23:54 +08:00
|
|
|
|
it 'is valid when username contains only allowlisted letters' do
|
2019-04-23 18:22:47 +08:00
|
|
|
|
expect_valid('Löwe', 'Ötzi')
|
|
|
|
|
end
|
|
|
|
|
|
2020-07-27 08:23:54 +08:00
|
|
|
|
it 'is valid when username contains only ASCII letters and numbers regardless of allowlist' do
|
2019-04-23 18:22:47 +08:00
|
|
|
|
expect_valid('a-z_A-Z.0-9')
|
|
|
|
|
end
|
2013-02-08 22:52:56 +08:00
|
|
|
|
|
2019-04-23 18:22:47 +08:00
|
|
|
|
it 'is valid after resetting the site setting' do
|
2020-07-27 08:23:54 +08:00
|
|
|
|
SiteSetting.allowed_unicode_username_characters = ""
|
2019-04-23 18:22:47 +08:00
|
|
|
|
expect_valid('鳥')
|
|
|
|
|
end
|
|
|
|
|
end
|
2013-02-08 22:52:56 +08:00
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
end
|