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,8 +4,9 @@ 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
 | 
					      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: nil, 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
 | 
				
			||||||
| 
						 | 
					@ -17,9 +18,51 @@ describe UniqueUsernameValidator do
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      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')
 | 
				
			||||||
      account = double(username: 'abcDEF', persisted?: false, errors: double(add: nil))
 | 
					        account = double(username: 'abcDEF', domain: nil, 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 '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
 | 
				
			||||||
 | 
					      account = double(username: nil, domain: 'example.com', 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', domain: 'example.com')
 | 
				
			||||||
 | 
					      expect(account).to be_valid
 | 
				
			||||||
 | 
					    end
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    it 'adds an error when the username 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 '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…
	
	Add table
		
		Reference in a new issue