mirror of
https://github.com/mastodon/mastodon.git
synced 2025-01-18 15:11:12 +01:00
Search account domain in lowercase (#13016)
* Search account domain in lowercase * fix rubocop error * fix spec/models/account_spec.rb
This commit is contained in:
parent
37dc12dd53
commit
61a7390b66
5 changed files with 55 additions and 16 deletions
|
@ -70,14 +70,13 @@ class Account < ApplicationRecord
|
||||||
enum protocol: [:ostatus, :activitypub]
|
enum protocol: [:ostatus, :activitypub]
|
||||||
|
|
||||||
validates :username, presence: true
|
validates :username, presence: true
|
||||||
|
validates_with UniqueUsernameValidator, if: -> { will_save_change_to_username? }
|
||||||
|
|
||||||
# Remote user validations
|
# Remote user validations
|
||||||
validates :username, uniqueness: { scope: :domain, case_sensitive: true }, if: -> { !local? && will_save_change_to_username? }
|
|
||||||
validates :username, format: { with: /\A#{USERNAME_RE}\z/i }, if: -> { !local? && will_save_change_to_username? }
|
validates :username, format: { with: /\A#{USERNAME_RE}\z/i }, if: -> { !local? && will_save_change_to_username? }
|
||||||
|
|
||||||
# Local user validations
|
# Local user validations
|
||||||
validates :username, format: { with: /\A[a-z0-9_]+\z/i }, length: { maximum: 30 }, if: -> { local? && will_save_change_to_username? && actor_type != 'Application' }
|
validates :username, format: { with: /\A[a-z0-9_]+\z/i }, length: { maximum: 30 }, if: -> { local? && will_save_change_to_username? && actor_type != 'Application' }
|
||||||
validates_with UniqueUsernameValidator, if: -> { local? && will_save_change_to_username? }
|
|
||||||
validates_with UnreservedUsernameValidator, if: -> { local? && will_save_change_to_username? }
|
validates_with UnreservedUsernameValidator, if: -> { local? && will_save_change_to_username? }
|
||||||
validates :display_name, length: { maximum: 30 }, if: -> { local? && will_save_change_to_display_name? }
|
validates :display_name, length: { maximum: 30 }, if: -> { local? && will_save_change_to_display_name? }
|
||||||
validates :note, note_length: { maximum: 500 }, if: -> { local? && will_save_change_to_note? }
|
validates :note, note_length: { maximum: 500 }, if: -> { local? && will_save_change_to_note? }
|
||||||
|
|
|
@ -48,7 +48,7 @@ module AccountFinderConcern
|
||||||
end
|
end
|
||||||
|
|
||||||
def with_usernames
|
def with_usernames
|
||||||
Account.where.not(username: '')
|
Account.where.not(Account.arel_table[:username].lower.eq '')
|
||||||
end
|
end
|
||||||
|
|
||||||
def matching_username
|
def matching_username
|
||||||
|
@ -56,11 +56,7 @@ module AccountFinderConcern
|
||||||
end
|
end
|
||||||
|
|
||||||
def matching_domain
|
def matching_domain
|
||||||
if domain.nil?
|
Account.where(Account.arel_table[:domain].lower.eq(domain.nil? ? nil : domain.to_s.downcase))
|
||||||
Account.where(domain: nil)
|
|
||||||
else
|
|
||||||
Account.where(Account.arel_table[:domain].lower.eq domain.to_s.downcase)
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -7,8 +7,9 @@ class UniqueUsernameValidator < ActiveModel::Validator
|
||||||
return if account.username.nil?
|
return if account.username.nil?
|
||||||
|
|
||||||
normalized_username = account.username.downcase
|
normalized_username = account.username.downcase
|
||||||
|
normalized_domain = account.domain&.downcase
|
||||||
|
|
||||||
scope = Account.where(domain: nil).where('lower(username) = ?', normalized_username)
|
scope = Account.where(Account.arel_table[:username].lower.eq normalized_username).where(Account.arel_table[:domain].lower.eq normalized_domain)
|
||||||
scope = scope.where.not(id: account.id) if account.persisted?
|
scope = scope.where.not(id: account.id) if account.persisted?
|
||||||
|
|
||||||
account.errors.add(:username, :taken) if scope.exists?
|
account.errors.add(:username, :taken) if scope.exists?
|
||||||
|
|
|
@ -619,18 +619,18 @@ RSpec.describe Account, type: :model do
|
||||||
end
|
end
|
||||||
|
|
||||||
context 'when is remote' do
|
context 'when is remote' do
|
||||||
it 'is invalid if the username is not unique in case-sensitive comparison among accounts in the same normalized domain' do
|
it 'is invalid if the username is same among accounts in the same normalized domain' do
|
||||||
Fabricate(:account, domain: 'にゃん', username: 'username')
|
Fabricate(:account, domain: 'にゃん', username: 'username')
|
||||||
account = Fabricate.build(:account, domain: 'xn--r9j5b5b', username: 'username')
|
account = Fabricate.build(:account, domain: 'xn--r9j5b5b', username: 'username')
|
||||||
account.valid?
|
account.valid?
|
||||||
expect(account).to model_have_error_on_field(:username)
|
expect(account).to model_have_error_on_field(:username)
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'is valid even if the username is unique only in case-sensitive comparison among accounts in the same normalized domain' do
|
it 'is invalid if the username is not unique in case-insensitive comparison among accounts in the same normalized domain' do
|
||||||
Fabricate(:account, domain: 'にゃん', username: 'username')
|
Fabricate(:account, domain: 'にゃん', username: 'username')
|
||||||
account = Fabricate.build(:account, domain: 'xn--r9j5b5b', username: 'Username')
|
account = Fabricate.build(:account, domain: 'xn--r9j5b5b', username: 'Username')
|
||||||
account.valid?
|
account.valid?
|
||||||
expect(account).not_to model_have_error_on_field(:username)
|
expect(account).to model_have_error_on_field(:username)
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'is valid even if the username contains hyphens' do
|
it 'is valid even if the username contains hyphens' do
|
||||||
|
|
|
@ -4,22 +4,65 @@ require 'rails_helper'
|
||||||
|
|
||||||
describe UniqueUsernameValidator do
|
describe UniqueUsernameValidator do
|
||||||
describe '#validate' do
|
describe '#validate' do
|
||||||
|
context 'when local account' do
|
||||||
|
it 'does not add errors if username is nil' do
|
||||||
|
account = double(username: nil, domain: nil, persisted?: false, errors: double(add: nil))
|
||||||
|
subject.validate(account)
|
||||||
|
expect(account.errors).to_not have_received(:add)
|
||||||
|
end
|
||||||
|
|
||||||
|
it 'does not add errors when existing one is subject itself' do
|
||||||
|
account = Fabricate(:account, username: 'abcdef')
|
||||||
|
expect(account).to be_valid
|
||||||
|
end
|
||||||
|
|
||||||
|
it 'adds an error when the username is already used with ignoring cases' do
|
||||||
|
Fabricate(:account, username: 'ABCdef')
|
||||||
|
account = double(username: 'abcDEF', domain: nil, persisted?: false, errors: double(add: nil))
|
||||||
|
subject.validate(account)
|
||||||
|
expect(account.errors).to have_received(:add)
|
||||||
|
end
|
||||||
|
|
||||||
|
it 'does not add errors when same username remote account exists' do
|
||||||
|
Fabricate(:account, username: 'abcdef', domain: 'example.com')
|
||||||
|
account = double(username: 'abcdef', domain: nil, persisted?: false, errors: double(add: nil))
|
||||||
|
subject.validate(account)
|
||||||
|
expect(account.errors).to_not have_received(:add)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
context 'when remote account' do
|
||||||
it 'does not add errors if username is nil' do
|
it 'does not add errors if username is nil' do
|
||||||
account = double(username: nil, persisted?: false, errors: double(add: nil))
|
account = double(username: nil, domain: 'example.com', persisted?: false, errors: double(add: nil))
|
||||||
subject.validate(account)
|
subject.validate(account)
|
||||||
expect(account.errors).to_not have_received(:add)
|
expect(account.errors).to_not have_received(:add)
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'does not add errors when existing one is subject itself' do
|
it 'does not add errors when existing one is subject itself' do
|
||||||
account = Fabricate(:account, username: 'abcdef')
|
account = Fabricate(:account, username: 'abcdef', domain: 'example.com')
|
||||||
expect(account).to be_valid
|
expect(account).to be_valid
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'adds an error when the username is already used with ignoring cases' do
|
it 'adds an error when the username is already used with ignoring cases' do
|
||||||
Fabricate(:account, username: 'ABCdef')
|
Fabricate(:account, username: 'ABCdef', domain: 'example.com')
|
||||||
account = double(username: 'abcDEF', persisted?: false, errors: double(add: nil))
|
account = double(username: 'abcDEF', domain: 'example.com', persisted?: false, errors: double(add: nil))
|
||||||
subject.validate(account)
|
subject.validate(account)
|
||||||
expect(account.errors).to have_received(:add)
|
expect(account.errors).to have_received(:add)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
it 'adds an error when the domain is already used with ignoring cases' do
|
||||||
|
Fabricate(:account, username: 'ABCdef', domain: 'example.com')
|
||||||
|
account = double(username: 'ABCdef', domain: 'EXAMPLE.COM', persisted?: false, errors: double(add: nil))
|
||||||
|
subject.validate(account)
|
||||||
|
expect(account.errors).to have_received(:add)
|
||||||
|
end
|
||||||
|
|
||||||
|
it 'does not add errors when account with the same username and another domain exists' do
|
||||||
|
Fabricate(:account, username: 'abcdef', domain: 'example.com')
|
||||||
|
account = double(username: 'abcdef', domain: 'example2.com', persisted?: false, errors: double(add: nil))
|
||||||
|
subject.validate(account)
|
||||||
|
expect(account.errors).to_not have_received(:add)
|
||||||
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
Loading…
Reference in a new issue