Expand spec coverage and refactor the `Account.find_` methods (#3485)
* Move specs for account finder methods to concern spec * Move account finder methods to concern * Improve spec wording * Use more explicit comparison to ensure correct return value * Add coverage for .find_local! and .find_remote! * Add some methods to the finder * Use arel on matching_username method * Avoid ternary in matching domain method * Simplify finder methods * Use an AccountFinder class to simplify lookupmaster
parent
bf811e4d4a
commit
2cc3111a77
@ -0,0 +1,58 @@ |
||||
# frozen_string_literal: true |
||||
|
||||
module AccountFinderConcern |
||||
extend ActiveSupport::Concern |
||||
|
||||
class_methods do |
||||
def find_local!(username) |
||||
find_local(username) || raise(ActiveRecord::RecordNotFound) |
||||
end |
||||
|
||||
def find_remote!(username, domain) |
||||
find_remote(username, domain) || raise(ActiveRecord::RecordNotFound) |
||||
end |
||||
|
||||
def find_local(username) |
||||
find_remote(username, nil) |
||||
end |
||||
|
||||
def find_remote(username, domain) |
||||
AccountFinder.new(username, domain).account |
||||
end |
||||
end |
||||
|
||||
class AccountFinder |
||||
attr_reader :username, :domain |
||||
|
||||
def initialize(username, domain) |
||||
@username = username |
||||
@domain = domain |
||||
end |
||||
|
||||
def account |
||||
scoped_accounts.take |
||||
end |
||||
|
||||
private |
||||
|
||||
def scoped_accounts |
||||
Account.unscoped.tap do |scope| |
||||
scope.merge! matching_username |
||||
scope.merge! matching_domain |
||||
end |
||||
end |
||||
|
||||
def matching_username |
||||
raise(ActiveRecord::RecordNotFound) if username.blank? |
||||
Account.where(Account.arel_table[:username].lower.eq username.downcase) |
||||
end |
||||
|
||||
def matching_domain |
||||
if domain.nil? |
||||
Account.where(domain: nil) |
||||
else |
||||
Account.where(Account.arel_table[:domain].lower.eq domain.downcase) |
||||
end |
||||
end |
||||
end |
||||
end |
@ -0,0 +1,93 @@ |
||||
# frozen_string_literal: true |
||||
|
||||
require 'rails_helper' |
||||
|
||||
describe AccountFinderConcern do |
||||
describe 'local finders' do |
||||
before do |
||||
@account = Fabricate(:account, username: 'Alice') |
||||
end |
||||
|
||||
describe '.find_local' do |
||||
it 'returns case-insensitive result' do |
||||
expect(Account.find_local('alice')).to eq(@account) |
||||
end |
||||
|
||||
it 'returns correctly cased result' do |
||||
expect(Account.find_local('Alice')).to eq(@account) |
||||
end |
||||
|
||||
it 'returns nil without a match' do |
||||
expect(Account.find_local('a_ice')).to be_nil |
||||
end |
||||
|
||||
it 'returns nil for regex style username value' do |
||||
expect(Account.find_local('al%')).to be_nil |
||||
end |
||||
end |
||||
|
||||
describe '.find_local!' do |
||||
it 'returns matching result' do |
||||
expect(Account.find_local!('alice')).to eq(@account) |
||||
end |
||||
|
||||
it 'raises on non-matching result' do |
||||
expect { Account.find_local!('missing') }.to raise_error(ActiveRecord::RecordNotFound) |
||||
end |
||||
|
||||
it 'raises with blank username' do |
||||
expect { Account.find_local!('') }.to raise_error(ActiveRecord::RecordNotFound) |
||||
end |
||||
|
||||
it 'raises with nil username' do |
||||
expect { Account.find_local!(nil) }.to raise_error(ActiveRecord::RecordNotFound) |
||||
end |
||||
end |
||||
end |
||||
|
||||
describe 'remote finders' do |
||||
before do |
||||
@account = Fabricate(:account, username: 'Alice', domain: 'mastodon.social') |
||||
end |
||||
|
||||
describe '.find_remote' do |
||||
it 'returns exact match result' do |
||||
expect(Account.find_remote('alice', 'mastodon.social')).to eq(@account) |
||||
end |
||||
|
||||
it 'returns case-insensitive result' do |
||||
expect(Account.find_remote('ALICE', 'MASTODON.SOCIAL')).to eq(@account) |
||||
end |
||||
|
||||
it 'returns nil when username does not match' do |
||||
expect(Account.find_remote('a_ice', 'mastodon.social')).to be_nil |
||||
end |
||||
|
||||
it 'returns nil when domain does not match' do |
||||
expect(Account.find_remote('alice', 'm_stodon.social')).to be_nil |
||||
end |
||||
|
||||
it 'returns nil for regex style domain value' do |
||||
expect(Account.find_remote('alice', 'm%')).to be_nil |
||||
end |
||||
end |
||||
|
||||
describe '.find_remote!' do |
||||
it 'returns matching result' do |
||||
expect(Account.find_remote!('alice', 'mastodon.social')).to eq(@account) |
||||
end |
||||
|
||||
it 'raises on non-matching result' do |
||||
expect { Account.find_remote!('missing', 'mastodon.host') }.to raise_error(ActiveRecord::RecordNotFound) |
||||
end |
||||
|
||||
it 'raises with blank username' do |
||||
expect { Account.find_remote!('', '') }.to raise_error(ActiveRecord::RecordNotFound) |
||||
end |
||||
|
||||
it 'raises with nil username' do |
||||
expect { Account.find_remote!(nil, nil) }.to raise_error(ActiveRecord::RecordNotFound) |
||||
end |
||||
end |
||||
end |
||||
end |
Loading…
Reference in new issue