# frozen_string_literal: true require 'rails_helper' RSpec.describe Account do include_examples 'Reviewable' context 'with an account record' do subject { Fabricate(:account) } let(:bob) { Fabricate(:account, username: 'bob') } describe '#suspend!' do it 'marks the account as suspended and creates a deletion request' do expect { subject.suspend! } .to change(subject, :suspended?).from(false).to(true) .and(change { AccountDeletionRequest.exists?(account: subject) }.from(false).to(true)) end context 'when the account is of a local user' do subject { local_user_account } let!(:local_user_account) { Fabricate(:user, email: 'foo+bar@domain.org').account } it 'creates a canonical domain block' do subject.suspend! expect(CanonicalEmailBlock.block?(subject.user_email)).to be true end context 'when a canonical domain block already exists for that email' do before do Fabricate(:canonical_email_block, email: subject.user_email) end it 'does not raise an error' do expect { subject.suspend! }.to_not raise_error end end end end describe '#follow!' do it 'creates a follow' do follow = subject.follow!(bob) expect(follow).to be_instance_of Follow expect(follow.account).to eq subject expect(follow.target_account).to eq bob end end describe '#unfollow!' do before do subject.follow!(bob) end it 'destroys a follow' do unfollow = subject.unfollow!(bob) expect(unfollow).to be_instance_of Follow expect(unfollow.account).to eq subject expect(unfollow.target_account).to eq bob expect(unfollow.destroyed?).to be true end end describe '#following?' do it 'returns true when the target is followed' do subject.follow!(bob) expect(subject.following?(bob)).to be true end it 'returns false if the target is not followed' do expect(subject.following?(bob)).to be false end end end describe '#local?' do it 'returns true when the account is local' do account = Fabricate(:account, domain: nil) expect(account.local?).to be true end it 'returns false when the account is on a different domain' do account = Fabricate(:account, domain: 'foreign.tld') expect(account.local?).to be false end end describe 'Local domain user methods' do subject { Fabricate(:account, domain: nil, username: 'alice') } around do |example| before = Rails.configuration.x.local_domain example.run Rails.configuration.x.local_domain = before end describe '#to_webfinger_s' do it 'returns a webfinger string for the account' do Rails.configuration.x.local_domain = 'example.com' expect(subject.to_webfinger_s).to eq 'acct:alice@example.com' end end describe '#local_username_and_domain' do it 'returns the username and local domain for the account' do Rails.configuration.x.local_domain = 'example.com' expect(subject.local_username_and_domain).to eq 'alice@example.com' end end end describe '#acct' do it 'returns username for local users' do account = Fabricate(:account, domain: nil, username: 'alice') expect(account.acct).to eql 'alice' end it 'returns username@domain for foreign users' do account = Fabricate(:account, domain: 'foreign.tld', username: 'alice') expect(account.acct).to eql 'alice@foreign.tld' end end describe '#save_with_optional_media!' do before do stub_request(:get, 'https://remote.test/valid_avatar').to_return(request_fixture('avatar.txt')) stub_request(:get, 'https://remote.test/invalid_avatar').to_return(request_fixture('feed.txt')) end let(:account) do Fabricate(:account, avatar_remote_url: 'https://remote.test/valid_avatar', header_remote_url: 'https://remote.test/valid_avatar') end let!(:expectation) { account.dup } context 'with valid properties' do before do account.save_with_optional_media! end it 'unchanges avatar, header, avatar_remote_url, and header_remote_url' do expect(account.avatar_remote_url).to eq expectation.avatar_remote_url expect(account.header_remote_url).to eq expectation.header_remote_url expect(account.avatar_file_name).to eq expectation.avatar_file_name expect(account.header_file_name).to eq expectation.header_file_name end end context 'with invalid properties' do before do account.avatar_remote_url = 'https://remote.test/invalid_avatar' account.save_with_optional_media! end it 'sets default avatar, header, avatar_remote_url, and header_remote_url' do expect(account.avatar_remote_url).to eq 'https://remote.test/invalid_avatar' expect(account.header_remote_url).to eq expectation.header_remote_url expect(account.avatar_file_name).to be_nil expect(account.header_file_name).to eq expectation.header_file_name end end end describe '#possibly_stale?' do let(:account) { Fabricate(:account, last_webfingered_at: last_webfingered_at) } context 'when last_webfingered_at is nil' do let(:last_webfingered_at) { nil } it 'returns true' do expect(account.possibly_stale?).to be true end end context 'when last_webfingered_at is more than 24 hours before' do let(:last_webfingered_at) { 25.hours.ago } it 'returns true' do expect(account.possibly_stale?).to be true end end context 'when last_webfingered_at is less than 24 hours before' do let(:last_webfingered_at) { 23.hours.ago } it 'returns false' do expect(account.possibly_stale?).to be false end end end describe '#refresh!' do let(:account) { Fabricate(:account, domain: domain) } let(:acct) { account.acct } context 'when domain is nil' do let(:domain) { nil } it 'returns nil' do expect(account.refresh!).to be_nil end it 'does not call ResolveAccountService#call' do service = instance_double(ResolveAccountService, call: nil) allow(ResolveAccountService).to receive(:new).and_return(service) account.refresh! expect(service).to_not have_received(:call).with(acct) end end context 'when domain is present' do let(:domain) { 'example.com' } it 'calls ResolveAccountService#call' do service = instance_double(ResolveAccountService, call: nil) allow(ResolveAccountService).to receive(:new).and_return(service) account.refresh! expect(service).to have_received(:call).with(acct).once end end end describe '#to_param' do it 'returns username' do account = Fabricate(:account, username: 'alice') expect(account.to_param).to eq 'alice' end end describe '#keypair' do it 'returns an RSA key pair' do account = Fabricate(:account) expect(account.keypair).to be_instance_of OpenSSL::PKey::RSA end end describe '#object_type' do it 'is always a person' do account = Fabricate(:account) expect(account.object_type).to be :person end end describe '#favourited?' do subject { Fabricate(:account) } let(:original_status) do author = Fabricate(:account, username: 'original') Fabricate(:status, account: author) end context 'when the status is a reblog of another status' do let(:original_reblog) do author = Fabricate(:account, username: 'original_reblogger') Fabricate(:status, reblog: original_status, account: author) end it 'is true when this account has favourited it' do Fabricate(:favourite, status: original_reblog, account: subject) expect(subject.favourited?(original_status)).to be true end it 'is false when this account has not favourited it' do expect(subject.favourited?(original_status)).to be false end end context 'when the status is an original status' do it 'is true when this account has favourited it' do Fabricate(:favourite, status: original_status, account: subject) expect(subject.favourited?(original_status)).to be true end it 'is false when this account has not favourited it' do expect(subject.favourited?(original_status)).to be false end end end describe '#reblogged?' do subject { Fabricate(:account) } let(:original_status) do author = Fabricate(:account, username: 'original') Fabricate(:status, account: author) end context 'when the status is a reblog of another status' do let(:original_reblog) do author = Fabricate(:account, username: 'original_reblogger') Fabricate(:status, reblog: original_status, account: author) end it 'is true when this account has reblogged it' do Fabricate(:status, reblog: original_reblog, account: subject) expect(subject.reblogged?(original_reblog)).to be true end it 'is false when this account has not reblogged it' do expect(subject.reblogged?(original_reblog)).to be false end end context 'when the status is an original status' do it 'is true when this account has reblogged it' do Fabricate(:status, reblog: original_status, account: subject) expect(subject.reblogged?(original_status)).to be true end it 'is false when this account has not reblogged it' do expect(subject.reblogged?(original_status)).to be false end end end describe '#excluded_from_timeline_account_ids' do it 'includes account ids of blockings, blocked_bys and mutes' do account = Fabricate(:account) block = Fabricate(:block, account: account) mute = Fabricate(:mute, account: account) block_by = Fabricate(:block, target_account: account) results = account.excluded_from_timeline_account_ids expect(results.size).to eq 3 expect(results).to include( block.target_account.id, mute.target_account.id, block_by.account.id ) end end describe '#excluded_from_timeline_domains' do it 'returns the domains blocked by the account' do account = Fabricate(:account) account.block_domain!('domain') expect(account.excluded_from_timeline_domains).to contain_exactly('domain') end end describe '.search_for' do before do _missing = Fabricate( :account, display_name: 'Missing', username: 'missing', domain: 'missing.com' ) end it 'does not return suspended users' do Fabricate( :account, display_name: 'Display Name', username: 'username', domain: 'example.com', suspended: true ) results = described_class.search_for('username') expect(results).to eq [] end it 'does not return unapproved users' do match = Fabricate( :account, display_name: 'Display Name', username: 'username' ) match.user.update(approved: false) results = described_class.search_for('username') expect(results).to eq [] end it 'does not return unconfirmed users' do match = Fabricate( :account, display_name: 'Display Name', username: 'username' ) match.user.update(confirmed_at: nil) results = described_class.search_for('username') expect(results).to eq [] end it 'accepts ?, \, : and space as delimiter' do match = Fabricate( :account, display_name: 'A & l & i & c & e', username: 'username', domain: 'example.com' ) results = described_class.search_for('A?l\i:c e') expect(results).to eq [match] end it 'finds accounts with matching display_name' do match = Fabricate( :account, display_name: 'Display Name', username: 'username', domain: 'example.com' ) results = described_class.search_for('display') expect(results).to eq [match] end it 'finds accounts with matching username' do match = Fabricate( :account, display_name: 'Display Name', username: 'username', domain: 'example.com' ) results = described_class.search_for('username') expect(results).to eq [match] end it 'finds accounts with matching domain' do match = Fabricate( :account, display_name: 'Display Name', username: 'username', domain: 'example.com' ) results = described_class.search_for('example') expect(results).to eq [match] end it 'limits via constant by default' do stub_const('Account::Search::DEFAULT_LIMIT', 1) 2.times.each { Fabricate(:account, display_name: 'Display Name') } results = described_class.search_for('display') expect(results.size).to eq 1 end it 'accepts arbitrary limits' do 2.times.each { Fabricate(:account, display_name: 'Display Name') } results = described_class.search_for('display', limit: 1) expect(results.size).to eq 1 end it 'ranks multiple matches higher' do matches = [ { username: 'username', display_name: 'username' }, { display_name: 'Display Name', username: 'username', domain: 'example.com' }, ].map(&method(:Fabricate).curry(2).call(:account)) results = described_class.search_for('username') expect(results).to eq matches end end describe '.advanced_search_for' do let(:account) { Fabricate(:account) } context 'when limiting search to followed accounts' do it 'accepts ?, \, : and space as delimiter' do match = Fabricate( :account, display_name: 'A & l & i & c & e', username: 'username', domain: 'example.com' ) account.follow!(match) results = described_class.advanced_search_for('A?l\i:c e', account, limit: 10, following: true) expect(results).to eq [match] end it 'does not return non-followed accounts' do Fabricate( :account, display_name: 'A & l & i & c & e', username: 'username', domain: 'example.com' ) results = described_class.advanced_search_for('A?l\i:c e', account, limit: 10, following: true) expect(results).to eq [] end it 'does not return suspended users' do Fabricate( :account, display_name: 'Display Name', username: 'username', domain: 'example.com', suspended: true ) results = described_class.advanced_search_for('username', account, limit: 10, following: true) expect(results).to eq [] end it 'does not return unapproved users' do match = Fabricate( :account, display_name: 'Display Name', username: 'username' ) match.user.update(approved: false) results = described_class.advanced_search_for('username', account, limit: 10, following: true) expect(results).to eq [] end it 'does not return unconfirmed users' do match = Fabricate( :account, display_name: 'Display Name', username: 'username' ) match.user.update(confirmed_at: nil) results = described_class.advanced_search_for('username', account, limit: 10, following: true) expect(results).to eq [] end end it 'does not return suspended users' do Fabricate( :account, display_name: 'Display Name', username: 'username', domain: 'example.com', suspended: true ) results = described_class.advanced_search_for('username', account) expect(results).to eq [] end it 'does not return unapproved users' do match = Fabricate( :account, display_name: 'Display Name', username: 'username' ) match.user.update(approved: false) results = described_class.advanced_search_for('username', account) expect(results).to eq [] end it 'does not return unconfirmed users' do match = Fabricate( :account, display_name: 'Display Name', username: 'username' ) match.user.update(confirmed_at: nil) results = described_class.advanced_search_for('username', account) expect(results).to eq [] end it 'accepts ?, \, : and space as delimiter' do match = Fabricate( :account, display_name: 'A & l & i & c & e', username: 'username', domain: 'example.com' ) results = described_class.advanced_search_for('A?l\i:c e', account) expect(results).to eq [match] end it 'limits result count by default value' do stub_const('Account::Search::DEFAULT_LIMIT', 1) 2.times { Fabricate(:account, display_name: 'Display Name') } results = described_class.advanced_search_for('display', account) expect(results.size).to eq 1 end it 'accepts arbitrary limits' do 2.times { Fabricate(:account, display_name: 'Display Name') } results = described_class.advanced_search_for('display', account, limit: 1) expect(results.size).to eq 1 end it 'ranks followed accounts higher' do match = Fabricate(:account, username: 'Matching') followed_match = Fabricate(:account, username: 'Matcher') Fabricate(:follow, account: account, target_account: followed_match) results = described_class.advanced_search_for('match', account) expect(results).to eq [followed_match, match] expect(results.first.rank).to be > results.last.rank end end describe '#statuses_count' do subject { Fabricate(:account) } it 'counts statuses' do Fabricate(:status, account: subject) Fabricate(:status, account: subject) expect(subject.statuses_count).to eq 2 end it 'does not count direct statuses' do Fabricate(:status, account: subject, visibility: :direct) expect(subject.statuses_count).to eq 0 end it 'is decremented when status is removed' do status = Fabricate(:status, account: subject) expect(subject.statuses_count).to eq 1 status.destroy expect(subject.statuses_count).to eq 0 end it 'is decremented when status is removed when account is not preloaded' do status = Fabricate(:status, account: subject) expect(subject.reload.statuses_count).to eq 1 clean_status = Status.find(status.id) expect(clean_status.association(:account).loaded?).to be false clean_status.destroy expect(subject.reload.statuses_count).to eq 0 end end describe '.following_map' do it 'returns an hash' do expect(described_class.following_map([], 1)).to be_a Hash end end describe '.followed_by_map' do it 'returns an hash' do expect(described_class.followed_by_map([], 1)).to be_a Hash end end describe '.blocking_map' do it 'returns an hash' do expect(described_class.blocking_map([], 1)).to be_a Hash end end describe '.requested_map' do it 'returns an hash' do expect(described_class.requested_map([], 1)).to be_a Hash end end describe '.requested_by_map' do it 'returns an hash' do expect(described_class.requested_by_map([], 1)).to be_a Hash end end describe 'MENTION_RE' do subject { described_class::MENTION_RE } it 'matches usernames in the middle of a sentence' do expect(subject.match('Hello to @alice from me')[1]).to eq 'alice' end it 'matches usernames in the beginning of status' do expect(subject.match('@alice Hey how are you?')[1]).to eq 'alice' end it 'matches full usernames' do expect(subject.match('@alice@example.com')[1]).to eq 'alice@example.com' end it 'matches full usernames with a dot at the end' do expect(subject.match('Hello @alice@example.com.')[1]).to eq 'alice@example.com' end it 'matches dot-prepended usernames' do expect(subject.match('.@alice I want everybody to see this')[1]).to eq 'alice' end it 'does not match e-mails' do expect(subject.match('Drop me an e-mail at alice@example.com')).to be_nil end it 'does not match URLs' do expect(subject.match('Check this out https://medium.com/@alice/some-article#.abcdef123')).to be_nil end it 'does not match URL query string' do expect(subject.match('https://example.com/?x=@alice')).to be_nil end it 'matches usernames immediately following the letter ß' do expect(subject.match('Hello toß @alice from me')[1]).to eq 'alice' end it 'matches usernames containing uppercase characters' do expect(subject.match('Hello to @aLice@Example.com from me')[1]).to eq 'aLice@Example.com' end end describe '#prepare_contents' do subject { Fabricate.build :account, domain: domain, note: ' padded note ', display_name: ' padded name ' } context 'with local account' do let(:domain) { nil } it 'strips values' do expect { subject.valid? } .to change(subject, :note).to('padded note') .and(change(subject, :display_name).to('padded name')) end end context 'with remote account' do let(:domain) { 'host.example' } it 'preserves values' do expect { subject.valid? } .to not_change(subject, :note) .and(not_change(subject, :display_name)) end end end describe 'Normalizations' do describe 'username' do it { is_expected.to normalize(:username).from(" \u3000bob \t \u00a0 \n ").to('bob') } end end describe 'validations' do it { is_expected.to validate_presence_of(:username) } context 'when is local' do it 'is invalid if the username is not unique in case-insensitive comparison among local accounts' do _account = Fabricate(:account, username: 'the_doctor') non_unique_account = Fabricate.build(:account, username: 'the_Doctor') non_unique_account.valid? expect(non_unique_account).to model_have_error_on_field(:username) end it 'is invalid if the username is reserved' do account = Fabricate.build(:account, username: 'support') account.valid? expect(account).to model_have_error_on_field(:username) end it 'is valid when username is reserved but record has already been created' do account = Fabricate.build(:account, username: 'support') account.save(validate: false) expect(account.valid?).to be true end it 'is valid if we are creating an instance actor account with a period' do account = Fabricate.build(:account, id: described_class::INSTANCE_ACTOR_ID, actor_type: 'Application', locked: true, username: 'example.com') expect(account.valid?).to be true end it 'is valid if we are creating a possibly-conflicting instance actor account' do _account = Fabricate(:account, username: 'examplecom') instance_account = Fabricate.build(:account, id: described_class::INSTANCE_ACTOR_ID, actor_type: 'Application', locked: true, username: 'example.com') expect(instance_account.valid?).to be true end it 'is invalid if the username doesn\'t only contains letters, numbers and underscores' do account = Fabricate.build(:account, username: 'the-doctor') account.valid? expect(account).to model_have_error_on_field(:username) end it 'is invalid if the username contains a period' do account = Fabricate.build(:account, username: 'the.doctor') account.valid? expect(account).to model_have_error_on_field(:username) end it 'is invalid if the username is longer than the character limit' do account = Fabricate.build(:account, username: username_over_limit) account.valid? expect(account).to model_have_error_on_field(:username) end it 'is invalid if the display name is longer than the character limit' do account = Fabricate.build(:account, display_name: display_name_over_limit) account.valid? expect(account).to model_have_error_on_field(:display_name) end it 'is invalid if the note is longer than the character limit' do account = Fabricate.build(:account, note: account_note_over_limit) account.valid? expect(account).to model_have_error_on_field(:note) end end context 'when is remote' do it 'is invalid if the username is same among accounts in the same normalized domain' do Fabricate(:account, domain: 'にゃん', username: 'username') account = Fabricate.build(:account, domain: 'xn--r9j5b5b', username: 'username') account.valid? expect(account).to model_have_error_on_field(:username) end 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') account = Fabricate.build(:account, domain: 'xn--r9j5b5b', username: 'Username') account.valid? expect(account).to model_have_error_on_field(:username) end it 'is valid even if the username contains hyphens' do account = Fabricate.build(:account, domain: 'domain', username: 'the-doctor') account.valid? expect(account).to_not model_have_error_on_field(:username) end it 'is invalid if the username doesn\'t only contains letters, numbers, underscores and hyphens' do account = Fabricate.build(:account, domain: 'domain', username: 'the doctor') account.valid? expect(account).to model_have_error_on_field(:username) end it 'is valid even if the username is longer than the character limit' do account = Fabricate.build(:account, domain: 'domain', username: username_over_limit) account.valid? expect(account).to_not model_have_error_on_field(:username) end it 'is valid even if the display name is longer than the character limit' do account = Fabricate.build(:account, domain: 'domain', display_name: display_name_over_limit) account.valid? expect(account).to_not model_have_error_on_field(:display_name) end it 'is valid even if the note is longer than the character limit' do account = Fabricate.build(:account, domain: 'domain', note: account_note_over_limit) account.valid? expect(account).to_not model_have_error_on_field(:note) end end def username_over_limit 'a' * described_class::USERNAME_LENGTH_LIMIT * 2 end def display_name_over_limit 'a' * described_class::DISPLAY_NAME_LENGTH_LIMIT * 2 end def account_note_over_limit 'a' * described_class::NOTE_LENGTH_LIMIT * 2 end end describe 'scopes' do describe 'matches_uri_prefix' do let!(:alice) { Fabricate :account, domain: 'host.example', uri: 'https://host.example/user/a' } let!(:bob) { Fabricate :account, domain: 'top-level.example', uri: 'https://top-level.example' } it 'returns accounts which start with the value' do results = described_class.matches_uri_prefix('https://host.example') expect(results.size) .to eq(1) expect(results) .to include(alice) .and not_include(bob) end it 'returns accounts which equal the value' do results = described_class.matches_uri_prefix('https://top-level.example') expect(results.size) .to eq(1) expect(results) .to include(bob) .and not_include(alice) end end describe 'auditable' do let!(:alice) { Fabricate :account } let!(:bob) { Fabricate :account } before do 2.times { Fabricate :action_log, account: alice } end it 'returns distinct accounts with action log records' do results = described_class.auditable expect(results.size) .to eq(1) expect(results) .to include(alice) .and not_include(bob) end end describe 'alphabetic' do it 'sorts by alphabetic order of domain and username' do matches = [ { username: 'a', domain: 'a' }, { username: 'b', domain: 'a' }, { username: 'a', domain: 'b' }, { username: 'b', domain: 'b' }, ].map(&method(:Fabricate).curry(2).call(:account)) expect(described_class.without_internal.alphabetic).to eq matches end end describe 'matches_display_name' do it 'matches display name which starts with the given string' do match = Fabricate(:account, display_name: 'pattern and suffix') Fabricate(:account, display_name: 'prefix and pattern') expect(described_class.matches_display_name('pattern')).to eq [match] end end describe 'matches_username' do it 'matches display name which starts with the given string' do match = Fabricate(:account, username: 'pattern_and_suffix') Fabricate(:account, username: 'prefix_and_pattern') expect(described_class.matches_username('pattern')).to eq [match] end end describe 'by_domain_and_subdomains' do it 'returns exact domain matches' do account = Fabricate(:account, domain: 'example.com') expect(described_class.by_domain_and_subdomains('example.com')).to eq [account] end it 'returns subdomains' do account = Fabricate(:account, domain: 'foo.example.com') expect(described_class.by_domain_and_subdomains('example.com')).to eq [account] end it 'does not return partially matching domains' do account = Fabricate(:account, domain: 'grexample.com') expect(described_class.by_domain_and_subdomains('example.com')).to_not eq [account] end end describe 'remote' do it 'returns an array of accounts who have a domain' do _account = Fabricate(:account, domain: nil) account_with_domain = Fabricate(:account, domain: 'example.com') expect(described_class.remote).to contain_exactly(account_with_domain) end end describe 'local' do it 'returns an array of accounts who do not have a domain' do local_account = Fabricate(:account, domain: nil) _account_with_domain = Fabricate(:account, domain: 'example.com') expect(described_class.without_internal.local).to contain_exactly(local_account) end end describe 'partitioned' do it 'returns a relation of accounts partitioned by domain' do matches = %w(a b a b) matches.size.times.to_a.shuffle.each do |index| matches[index] = Fabricate(:account, domain: matches[index]) end expect(described_class.without_internal.partitioned).to match_array(matches) end end describe 'recent' do it 'returns a relation of accounts sorted by recent creation' do matches = Array.new(2) { Fabricate(:account) } expect(described_class.without_internal.recent).to match_array(matches) end end describe 'silenced' do it 'returns an array of accounts who are silenced' do silenced_account = Fabricate(:account, silenced: true) _account = Fabricate(:account, silenced: false) expect(described_class.silenced).to contain_exactly(silenced_account) end end describe 'suspended' do it 'returns an array of accounts who are suspended' do suspended_account = Fabricate(:account, suspended: true) _account = Fabricate(:account, suspended: false) expect(described_class.suspended).to contain_exactly(suspended_account) end end describe 'searchable' do let!(:suspended_local) { Fabricate(:account, suspended: true, username: 'suspended_local') } let!(:suspended_remote) { Fabricate(:account, suspended: true, domain: 'example.org', username: 'suspended_remote') } let!(:silenced_local) { Fabricate(:account, silenced: true, username: 'silenced_local') } let!(:silenced_remote) { Fabricate(:account, silenced: true, domain: 'example.org', username: 'silenced_remote') } let!(:unconfirmed) { Fabricate(:user, confirmed_at: nil).account } let!(:unapproved) { Fabricate(:user, approved: false).account } let!(:unconfirmed_unapproved) { Fabricate(:user, confirmed_at: nil, approved: false).account } let!(:local_account) { Fabricate(:account, username: 'local_account') } let!(:remote_account) { Fabricate(:account, domain: 'example.org', username: 'remote_account') } before do # Accounts get automatically-approved depending on settings, so ensure they aren't approved unapproved.user.update(approved: false) unconfirmed_unapproved.user.update(approved: false) end it 'returns every usable non-suspended account' do expect(described_class.searchable).to contain_exactly(silenced_local, silenced_remote, local_account, remote_account) expect(described_class.searchable).to_not include(suspended_local, suspended_remote, unconfirmed, unapproved) end it 'does not mess with previously-applied scopes' do expect(described_class.where.not(id: remote_account.id).searchable).to contain_exactly(silenced_local, silenced_remote, local_account) end end end context 'when is local' do it 'generates keys' do account = described_class.create!(domain: nil, username: 'user_without_keys') expect(account) .to be_private_key .and be_public_key expect(account.keypair) .to be_private .and be_public end end context 'when is remote' do it 'does not generate keys' do key = OpenSSL::PKey::RSA.new(1024).public_key account = described_class.create!(domain: 'remote', uri: 'https://remote/actor', username: 'remote_user_with_public', public_key: key.to_pem) expect(account.keypair.params).to eq key.params end it 'normalizes domain' do account = described_class.create!(domain: 'にゃん', uri: 'https://xn--r9j5b5b/actor', username: 'remote_user_with_idn_domain') expect(account.domain).to eq 'xn--r9j5b5b' end end include_examples 'AccountAvatar', :account include_examples 'AccountHeader', :account describe '#increment_count!' do subject { Fabricate(:account) } it 'increments the count in multi-threaded an environment when account_stat is not yet initialized' do subject multi_threaded_execution(15) do described_class.find(subject.id).increment_count!(:followers_count) end expect(subject.reload.followers_count).to eq 15 end end end