2016-02-20 21:53:20 +00:00
|
|
|
require 'rails_helper'
|
|
|
|
|
|
|
|
RSpec.describe Account, type: :model do
|
2016-02-26 14:28:08 +00:00
|
|
|
context do
|
|
|
|
let(:bob) { Fabricate(:account, username: 'bob') }
|
2017-06-22 16:34:27 +00:00
|
|
|
subject { Fabricate(:account) }
|
2016-02-26 14:28:08 +00:00
|
|
|
|
|
|
|
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)
|
2016-02-24 23:17:01 +00:00
|
|
|
|
2016-02-26 14:28:08 +00:00
|
|
|
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
|
2016-02-24 23:17:01 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
describe '#local?' do
|
2016-02-26 14:28:08 +00:00
|
|
|
it 'returns true when the account is local' do
|
2017-06-22 16:34:27 +00:00
|
|
|
account = Fabricate(:account, domain: nil)
|
|
|
|
expect(account.local?).to be true
|
2016-02-26 14:28:08 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns false when the account is on a different domain' do
|
2017-06-22 16:34:27 +00:00
|
|
|
account = Fabricate(:account, domain: 'foreign.tld')
|
|
|
|
expect(account.local?).to be false
|
2016-02-26 14:28:08 +00:00
|
|
|
end
|
2017-04-10 20:58:06 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
describe 'Local domain user methods' do
|
|
|
|
around do |example|
|
|
|
|
before = Rails.configuration.x.local_domain
|
|
|
|
example.run
|
|
|
|
Rails.configuration.x.local_domain = before
|
|
|
|
end
|
|
|
|
|
2017-06-22 16:34:27 +00:00
|
|
|
subject { Fabricate(:account, domain: nil, username: 'alice') }
|
|
|
|
|
2017-04-10 20:58:06 +00:00
|
|
|
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
|
2016-02-24 23:17:01 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
describe '#acct' do
|
2016-02-26 14:28:08 +00:00
|
|
|
it 'returns username for local users' do
|
2017-06-22 16:34:27 +00:00
|
|
|
account = Fabricate(:account, domain: nil, username: 'alice')
|
|
|
|
expect(account.acct).to eql 'alice'
|
2016-02-26 14:28:08 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns username@domain for foreign users' do
|
2017-06-22 16:34:27 +00:00
|
|
|
account = Fabricate(:account, domain: 'foreign.tld', username: 'alice')
|
|
|
|
expect(account.acct).to eql 'alice@foreign.tld'
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '#save_with_optional_media!' do
|
2017-11-06 04:54:12 +00:00
|
|
|
before do
|
2018-04-25 12:12:28 +00:00
|
|
|
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'))
|
2017-11-06 04:54:12 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
let(:account) do
|
|
|
|
Fabricate(:account,
|
2018-04-25 12:12:28 +00:00
|
|
|
avatar_remote_url: 'https://remote.test/valid_avatar',
|
|
|
|
header_remote_url: 'https://remote.test/valid_avatar')
|
2017-11-06 04:54:12 +00:00
|
|
|
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
|
2018-04-25 12:12:28 +00:00
|
|
|
account.avatar_remote_url = 'https://remote.test/invalid_avatar'
|
2017-11-06 04:54:12 +00:00
|
|
|
account.save_with_optional_media!
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'sets default avatar, header, avatar_remote_url, and header_remote_url' do
|
2019-10-09 05:10:46 +00:00
|
|
|
expect(account.avatar_remote_url).to eq 'https://remote.test/invalid_avatar'
|
|
|
|
expect(account.header_remote_url).to eq expectation.header_remote_url
|
2017-11-06 04:54:12 +00:00
|
|
|
expect(account.avatar_file_name).to eq nil
|
|
|
|
expect(account.header_file_name).to eq nil
|
|
|
|
end
|
2016-02-26 14:28:08 +00:00
|
|
|
end
|
2016-02-24 23:17:01 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
describe '#subscribed?' do
|
2016-09-20 00:43:20 +00:00
|
|
|
it 'returns false when no subscription expiration information is present' do
|
2017-06-22 16:34:27 +00:00
|
|
|
account = Fabricate(:account, subscription_expires_at: nil)
|
|
|
|
expect(account.subscribed?).to be false
|
2016-02-26 14:28:08 +00:00
|
|
|
end
|
|
|
|
|
2016-09-20 00:43:20 +00:00
|
|
|
it 'returns true when subscription expiration has been set' do
|
2017-06-22 16:34:27 +00:00
|
|
|
account = Fabricate(:account, subscription_expires_at: 30.days.from_now)
|
|
|
|
expect(account.subscribed?).to be true
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-11-05 08:20:05 +00:00
|
|
|
describe '#possibly_stale?' do
|
|
|
|
let(:account) { Fabricate(:account, last_webfingered_at: last_webfingered_at) }
|
|
|
|
|
|
|
|
context '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 '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 '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
|
|
|
|
|
2017-11-06 04:54:41 +00:00
|
|
|
describe '#refresh!' do
|
|
|
|
let(:account) { Fabricate(:account, domain: domain) }
|
|
|
|
let(:acct) { account.acct }
|
|
|
|
|
|
|
|
context 'domain is nil' do
|
|
|
|
let(:domain) { nil }
|
|
|
|
|
|
|
|
it 'returns nil' do
|
|
|
|
expect(account.refresh!).to be_nil
|
|
|
|
end
|
|
|
|
|
2018-01-22 13:25:09 +00:00
|
|
|
it 'calls not ResolveAccountService#call' do
|
|
|
|
expect_any_instance_of(ResolveAccountService).not_to receive(:call).with(acct)
|
2017-11-06 04:54:41 +00:00
|
|
|
account.refresh!
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'domain is present' do
|
|
|
|
let(:domain) { 'example.com' }
|
|
|
|
|
2018-01-22 13:25:09 +00:00
|
|
|
it 'calls ResolveAccountService#call' do
|
|
|
|
expect_any_instance_of(ResolveAccountService).to receive(:call).with(acct).once
|
2017-11-06 04:54:41 +00:00
|
|
|
account.refresh!
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-06-22 16:34:27 +00:00
|
|
|
describe '#to_param' do
|
|
|
|
it 'returns username' do
|
|
|
|
account = Fabricate(:account, username: 'alice')
|
|
|
|
expect(account.to_param).to eq 'alice'
|
2016-02-26 14:28:08 +00:00
|
|
|
end
|
2016-02-24 23:17:01 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
describe '#keypair' do
|
2016-02-26 14:28:08 +00:00
|
|
|
it 'returns an RSA key pair' do
|
2017-06-22 16:34:27 +00:00
|
|
|
account = Fabricate(:account)
|
|
|
|
expect(account.keypair).to be_instance_of OpenSSL::PKey::RSA
|
2016-02-26 14:28:08 +00:00
|
|
|
end
|
2016-02-24 23:17:01 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
describe '#object_type' do
|
2016-02-26 14:28:08 +00:00
|
|
|
it 'is always a person' do
|
2017-06-22 16:34:27 +00:00
|
|
|
account = Fabricate(:account)
|
|
|
|
expect(account.object_type).to be :person
|
2016-02-26 14:28:08 +00:00
|
|
|
end
|
2016-02-24 23:17:01 +00:00
|
|
|
end
|
|
|
|
|
2016-03-19 11:13:47 +00:00
|
|
|
describe '#favourited?' do
|
2017-04-07 18:18:30 +00:00
|
|
|
let(:original_status) do
|
|
|
|
author = Fabricate(:account, username: 'original')
|
|
|
|
Fabricate(:status, account: author)
|
|
|
|
end
|
|
|
|
|
2017-06-22 16:34:27 +00:00
|
|
|
subject { Fabricate(:account) }
|
|
|
|
|
2017-04-07 18:18:30 +00:00
|
|
|
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 is true when this account has favourited it' do
|
|
|
|
Fabricate(:favourite, status: original_reblog, account: subject)
|
|
|
|
|
|
|
|
expect(subject.favourited?(original_status)).to eq true
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'is false when this account has not favourited it' do
|
|
|
|
expect(subject.favourited?(original_status)).to eq false
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when the status is an original status' do
|
|
|
|
it 'is is true when this account has favourited it' do
|
|
|
|
Fabricate(:favourite, status: original_status, account: subject)
|
|
|
|
|
|
|
|
expect(subject.favourited?(original_status)).to eq true
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'is false when this account has not favourited it' do
|
|
|
|
expect(subject.favourited?(original_status)).to eq false
|
|
|
|
end
|
|
|
|
end
|
2016-03-19 11:13:47 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
describe '#reblogged?' do
|
2017-04-07 18:18:30 +00:00
|
|
|
let(:original_status) do
|
|
|
|
author = Fabricate(:account, username: 'original')
|
|
|
|
Fabricate(:status, account: author)
|
|
|
|
end
|
|
|
|
|
2017-06-22 16:34:27 +00:00
|
|
|
subject { Fabricate(:account) }
|
|
|
|
|
2018-10-04 15:38:04 +00:00
|
|
|
context 'when the status is a reblog of another status' do
|
2017-04-07 18:18:30 +00:00
|
|
|
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 eq true
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'is false when this account has not reblogged it' do
|
|
|
|
expect(subject.reblogged?(original_reblog)).to eq 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 eq true
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'is false when this account has not reblogged it' do
|
|
|
|
expect(subject.reblogged?(original_status)).to eq false
|
|
|
|
end
|
|
|
|
end
|
2016-03-19 11:13:47 +00:00
|
|
|
end
|
|
|
|
|
2017-04-28 13:10:41 +00:00
|
|
|
describe '#excluded_from_timeline_account_ids' do
|
|
|
|
it 'includes account ids of blockings, blocked_bys and mutes' do
|
|
|
|
account = Fabricate(:account)
|
2017-04-30 12:49:24 +00:00
|
|
|
block = Fabricate(:block, account: account)
|
|
|
|
mute = Fabricate(:mute, account: account)
|
|
|
|
block_by = Fabricate(:block, target_account: account)
|
2017-04-28 13:10:41 +00:00
|
|
|
|
|
|
|
results = account.excluded_from_timeline_account_ids
|
|
|
|
expect(results.size).to eq 3
|
|
|
|
expect(results).to include(block.target_account.id)
|
|
|
|
expect(results).to include(mute.target_account.id)
|
|
|
|
expect(results).to include(block_by.account.id)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-06-22 16:34:27 +00:00
|
|
|
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 match_array ['domain']
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-04-09 12:45:01 +00:00
|
|
|
describe '.search_for' do
|
|
|
|
before do
|
|
|
|
_missing = Fabricate(
|
|
|
|
:account,
|
|
|
|
display_name: "Missing",
|
|
|
|
username: "missing",
|
|
|
|
domain: "missing.com"
|
|
|
|
)
|
|
|
|
end
|
|
|
|
|
2017-06-22 16:34:27 +00:00
|
|
|
it 'accepts ?, \, : and space as delimiter' do
|
|
|
|
match = Fabricate(
|
|
|
|
:account,
|
|
|
|
display_name: 'A & l & i & c & e',
|
|
|
|
username: 'username',
|
|
|
|
domain: 'example.com'
|
|
|
|
)
|
|
|
|
|
|
|
|
results = Account.search_for('A?l\i:c e')
|
|
|
|
expect(results).to eq [match]
|
|
|
|
end
|
|
|
|
|
2017-04-09 12:45:01 +00:00
|
|
|
it 'finds accounts with matching display_name' do
|
2017-06-22 16:34:27 +00:00
|
|
|
match = Fabricate(
|
|
|
|
:account,
|
|
|
|
display_name: "Display Name",
|
|
|
|
username: "username",
|
|
|
|
domain: "example.com"
|
|
|
|
)
|
|
|
|
|
2017-04-09 12:45:01 +00:00
|
|
|
results = Account.search_for("display")
|
2017-06-22 16:34:27 +00:00
|
|
|
expect(results).to eq [match]
|
2017-04-09 12:45:01 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
it 'finds accounts with matching username' do
|
2017-06-22 16:34:27 +00:00
|
|
|
match = Fabricate(
|
|
|
|
:account,
|
|
|
|
display_name: "Display Name",
|
|
|
|
username: "username",
|
|
|
|
domain: "example.com"
|
|
|
|
)
|
|
|
|
|
2017-04-09 12:45:01 +00:00
|
|
|
results = Account.search_for("username")
|
2017-06-22 16:34:27 +00:00
|
|
|
expect(results).to eq [match]
|
2017-04-09 12:45:01 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
it 'finds accounts with matching domain' do
|
2017-06-22 16:34:27 +00:00
|
|
|
match = Fabricate(
|
|
|
|
:account,
|
|
|
|
display_name: "Display Name",
|
|
|
|
username: "username",
|
|
|
|
domain: "example.com"
|
|
|
|
)
|
|
|
|
|
2017-04-09 12:45:01 +00:00
|
|
|
results = Account.search_for("example")
|
2017-06-22 16:34:27 +00:00
|
|
|
expect(results).to eq [match]
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'limits by 10 by default' do
|
|
|
|
11.times.each { Fabricate(:account, display_name: "Display Name") }
|
|
|
|
results = Account.search_for("display")
|
|
|
|
expect(results.size).to eq 10
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'accepts arbitrary limits' do
|
|
|
|
2.times.each { Fabricate(:account, display_name: "Display Name") }
|
|
|
|
results = Account.search_for("display", 1)
|
|
|
|
expect(results.size).to eq 1
|
2017-04-09 12:45:01 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
it 'ranks multiple matches higher' do
|
2017-06-22 16:34:27 +00:00
|
|
|
matches = [
|
|
|
|
{ username: "username", display_name: "username" },
|
|
|
|
{ display_name: "Display Name", username: "username", domain: "example.com" },
|
|
|
|
].map(&method(:Fabricate).curry(2).call(:account))
|
|
|
|
|
2017-04-09 12:45:01 +00:00
|
|
|
results = Account.search_for("username")
|
2017-06-22 16:34:27 +00:00
|
|
|
expect(results).to eq matches
|
2017-04-09 12:45:01 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '.advanced_search_for' do
|
2017-06-22 16:34:27 +00:00
|
|
|
it 'accepts ?, \, : and space as delimiter' do
|
|
|
|
account = Fabricate(:account)
|
|
|
|
match = Fabricate(
|
|
|
|
:account,
|
|
|
|
display_name: 'A & l & i & c & e',
|
|
|
|
username: 'username',
|
|
|
|
domain: 'example.com'
|
|
|
|
)
|
|
|
|
|
|
|
|
results = Account.advanced_search_for('A?l\i:c e', account)
|
|
|
|
expect(results).to eq [match]
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'limits by 10 by default' do
|
|
|
|
11.times { Fabricate(:account, display_name: "Display Name") }
|
|
|
|
results = Account.search_for("display")
|
|
|
|
expect(results.size).to eq 10
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'accepts arbitrary limits' do
|
|
|
|
2.times { Fabricate(:account, display_name: "Display Name") }
|
|
|
|
results = Account.search_for("display", 1)
|
|
|
|
expect(results.size).to eq 1
|
|
|
|
end
|
|
|
|
|
2017-04-09 12:45:01 +00:00
|
|
|
it 'ranks followed accounts higher' do
|
|
|
|
account = Fabricate(:account)
|
|
|
|
match = Fabricate(:account, username: "Matching")
|
|
|
|
followed_match = Fabricate(:account, username: "Matcher")
|
|
|
|
Fabricate(:follow, account: account, target_account: followed_match)
|
|
|
|
|
|
|
|
results = Account.advanced_search_for("match", account)
|
|
|
|
expect(results).to eq [followed_match, match]
|
|
|
|
expect(results.first.rank).to be > results.last.rank
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-06-22 16:34:27 +00:00
|
|
|
describe '.domains' do
|
|
|
|
it 'returns domains' do
|
|
|
|
Fabricate(:account, domain: 'domain')
|
2019-07-18 23:44:42 +00:00
|
|
|
expect(Account.remote.domains).to match_array(['domain'])
|
2017-06-22 16:34:27 +00:00
|
|
|
end
|
|
|
|
end
|
2017-05-16 10:06:38 +00:00
|
|
|
|
2018-05-30 00:50:23 +00:00
|
|
|
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
|
|
|
|
|
2017-01-09 13:00:55 +00:00
|
|
|
describe '.following_map' do
|
|
|
|
it 'returns an hash' do
|
|
|
|
expect(Account.following_map([], 1)).to be_a Hash
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '.followed_by_map' do
|
|
|
|
it 'returns an hash' do
|
|
|
|
expect(Account.followed_by_map([], 1)).to be_a Hash
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '.blocking_map' do
|
|
|
|
it 'returns an hash' do
|
|
|
|
expect(Account.blocking_map([], 1)).to be_a Hash
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '.requested_map' do
|
|
|
|
it 'returns an hash' do
|
|
|
|
expect(Account.requested_map([], 1)).to be_a Hash
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-03-27 21:51:18 +00:00
|
|
|
describe 'MENTION_RE' do
|
|
|
|
subject { Account::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
|
|
|
|
|
2017-03-05 17:08:19 +00:00
|
|
|
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
|
|
|
|
|
2016-03-27 21:51:18 +00:00
|
|
|
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
|
2017-05-05 17:48:22 +00:00
|
|
|
|
|
|
|
xit 'does not match URL querystring' do
|
|
|
|
expect(subject.match('https://example.com/?x=@alice')).to be_nil
|
|
|
|
end
|
2016-03-27 21:51:18 +00:00
|
|
|
end
|
2017-04-04 22:29:56 +00:00
|
|
|
|
|
|
|
describe 'validations' do
|
|
|
|
it 'has a valid fabricator' do
|
|
|
|
account = Fabricate.build(:account)
|
|
|
|
account.valid?
|
|
|
|
expect(account).to be_valid
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'is invalid without a username' do
|
|
|
|
account = Fabricate.build(:account, username: nil)
|
|
|
|
account.valid?
|
|
|
|
expect(account).to model_have_error_on_field(:username)
|
|
|
|
end
|
|
|
|
|
2019-03-11 19:48:24 +00:00
|
|
|
it 'squishes the username before validation' do
|
|
|
|
account = Fabricate(:account, domain: nil, username: " \u3000bob \t \u00a0 \n ")
|
|
|
|
expect(account.username).to eq 'bob'
|
|
|
|
end
|
|
|
|
|
2017-06-22 16:34:27 +00:00
|
|
|
context 'when is local' do
|
2018-09-13 22:53:09 +00:00
|
|
|
it 'is invalid if the username is not unique in case-insensitive comparison among local accounts' do
|
2017-06-22 16:34:27 +00:00
|
|
|
account_1 = Fabricate(:account, username: 'the_doctor')
|
|
|
|
account_2 = Fabricate.build(:account, username: 'the_Doctor')
|
|
|
|
account_2.valid?
|
|
|
|
expect(account_2).to model_have_error_on_field(:username)
|
|
|
|
end
|
2017-04-04 22:29:56 +00:00
|
|
|
|
2017-06-22 16:34:27 +00:00
|
|
|
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
|
2017-06-04 23:03:45 +00:00
|
|
|
|
2017-06-22 16:34:27 +00:00
|
|
|
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
|
2017-06-08 13:22:01 +00:00
|
|
|
|
2019-07-24 12:19:17 +00:00
|
|
|
it 'is valid if we are creating an instance actor account with a period' do
|
|
|
|
account = Fabricate.build(:account, id: -99, 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_1 = Fabricate(:account, username: 'examplecom')
|
|
|
|
account_2 = Fabricate.build(:account, id: -99, actor_type: 'Application', locked: true, username: 'example.com')
|
|
|
|
expect(account_2.valid?).to be true
|
|
|
|
end
|
|
|
|
|
2017-04-04 22:29:56 +00:00
|
|
|
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
|
|
|
|
|
2019-07-24 12:19:17 +00:00
|
|
|
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
|
|
|
|
|
2017-04-04 22:29:56 +00:00
|
|
|
it 'is invalid if the username is longer then 30 characters' do
|
2019-08-06 13:33:03 +00:00
|
|
|
account = Fabricate.build(:account, username: Faker::Lorem.characters(number: 31))
|
2017-04-04 22:29:56 +00:00
|
|
|
account.valid?
|
|
|
|
expect(account).to model_have_error_on_field(:username)
|
|
|
|
end
|
2017-06-22 16:34:27 +00:00
|
|
|
|
|
|
|
it 'is invalid if the display name is longer than 30 characters' do
|
2019-08-06 13:33:03 +00:00
|
|
|
account = Fabricate.build(:account, display_name: Faker::Lorem.characters(number: 31))
|
2017-06-22 16:34:27 +00:00
|
|
|
account.valid?
|
|
|
|
expect(account).to model_have_error_on_field(:display_name)
|
|
|
|
end
|
|
|
|
|
2019-05-19 20:51:44 +00:00
|
|
|
it 'is invalid if the note is longer than 500 characters' do
|
2019-08-06 13:33:03 +00:00
|
|
|
account = Fabricate.build(:account, note: Faker::Lorem.characters(number: 501))
|
2017-06-22 16:34:27 +00:00
|
|
|
account.valid?
|
|
|
|
expect(account).to model_have_error_on_field(:note)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when is remote' do
|
2020-02-01 14:42:24 +00:00
|
|
|
it 'is invalid if the username is same among accounts in the same normalized domain' do
|
2017-06-22 16:34:27 +00:00
|
|
|
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
|
|
|
|
|
2020-02-01 14:42:24 +00:00
|
|
|
it 'is invalid if the username is not unique in case-insensitive comparison among accounts in the same normalized domain' do
|
2017-06-22 16:34:27 +00:00
|
|
|
Fabricate(:account, domain: 'にゃん', username: 'username')
|
|
|
|
account = Fabricate.build(:account, domain: 'xn--r9j5b5b', username: 'Username')
|
|
|
|
account.valid?
|
2020-02-01 14:42:24 +00:00
|
|
|
expect(account).to model_have_error_on_field(:username)
|
2017-06-22 16:34:27 +00:00
|
|
|
end
|
|
|
|
|
2018-11-27 11:28:01 +00:00
|
|
|
it 'is valid even if the username contains hyphens' do
|
2017-06-22 16:34:27 +00:00
|
|
|
account = Fabricate.build(:account, domain: 'domain', username: 'the-doctor')
|
|
|
|
account.valid?
|
2018-11-27 11:28:01 +00:00
|
|
|
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?
|
2018-07-30 20:29:52 +00:00
|
|
|
expect(account).to model_have_error_on_field(:username)
|
2017-06-22 16:34:27 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
it 'is valid even if the username is longer then 30 characters' do
|
2019-08-06 13:33:03 +00:00
|
|
|
account = Fabricate.build(:account, domain: 'domain', username: Faker::Lorem.characters(number: 31))
|
2017-06-22 16:34:27 +00:00
|
|
|
account.valid?
|
|
|
|
expect(account).not_to model_have_error_on_field(:username)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'is valid even if the display name is longer than 30 characters' do
|
2019-08-06 13:33:03 +00:00
|
|
|
account = Fabricate.build(:account, domain: 'domain', display_name: Faker::Lorem.characters(number: 31))
|
2017-06-22 16:34:27 +00:00
|
|
|
account.valid?
|
|
|
|
expect(account).not_to model_have_error_on_field(:display_name)
|
|
|
|
end
|
|
|
|
|
2019-05-21 11:29:06 +00:00
|
|
|
it 'is valid even if the note is longer than 500 characters' do
|
2019-08-06 13:33:03 +00:00
|
|
|
account = Fabricate.build(:account, domain: 'domain', note: Faker::Lorem.characters(number: 501))
|
2017-06-22 16:34:27 +00:00
|
|
|
account.valid?
|
|
|
|
expect(account).not_to model_have_error_on_field(:note)
|
|
|
|
end
|
2017-04-04 22:29:56 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe 'scopes' do
|
2017-06-22 16:34:27 +00:00
|
|
|
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))
|
|
|
|
|
2019-07-18 23:44:42 +00:00
|
|
|
expect(Account.where('id > 0').alphabetic).to eq matches
|
2017-06-22 16:34:27 +00:00
|
|
|
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(Account.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(Account.matches_username('pattern')).to eq [match]
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-06-21 22:13:10 +00:00
|
|
|
describe 'by_domain_and_subdomains' do
|
|
|
|
it 'returns exact domain matches' do
|
|
|
|
account = Fabricate(:account, domain: 'example.com')
|
|
|
|
expect(Account.by_domain_and_subdomains('example.com')).to eq [account]
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns subdomains' do
|
|
|
|
account = Fabricate(:account, domain: 'foo.example.com')
|
|
|
|
expect(Account.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(Account.by_domain_and_subdomains('example.com')).to_not eq [account]
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-06-22 16:34:27 +00:00
|
|
|
describe 'expiring' do
|
|
|
|
it 'returns remote accounts with followers whose subscription expiration date is past or not given' do
|
|
|
|
local = Fabricate(:account, domain: nil)
|
|
|
|
matches = [
|
|
|
|
{ domain: 'remote', subscription_expires_at: '2000-01-01T00:00:00Z' },
|
|
|
|
].map(&method(:Fabricate).curry(2).call(:account))
|
|
|
|
matches.each(&local.method(:follow!))
|
|
|
|
Fabricate(:account, domain: 'remote', subscription_expires_at: nil)
|
|
|
|
local.follow!(Fabricate(:account, domain: 'remote', subscription_expires_at: '2000-01-03T00:00:00Z'))
|
|
|
|
local.follow!(Fabricate(:account, domain: nil, subscription_expires_at: nil))
|
|
|
|
|
|
|
|
expect(Account.expiring('2000-01-02T00:00:00Z').recent).to eq matches.reverse
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-04-04 22:29:56 +00:00
|
|
|
describe 'remote' do
|
|
|
|
it 'returns an array of accounts who have a domain' do
|
|
|
|
account_1 = Fabricate(:account, domain: nil)
|
|
|
|
account_2 = Fabricate(:account, domain: 'example.com')
|
|
|
|
expect(Account.remote).to match_array([account_2])
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-04-18 19:09:07 +00:00
|
|
|
describe 'by_domain_accounts' do
|
|
|
|
it 'returns accounts grouped by domain sorted by accounts' do
|
|
|
|
2.times { Fabricate(:account, domain: 'example.com') }
|
|
|
|
Fabricate(:account, domain: 'example2.com')
|
|
|
|
|
2019-07-18 23:44:42 +00:00
|
|
|
results = Account.where('id > 0').by_domain_accounts
|
2017-04-18 19:09:07 +00:00
|
|
|
expect(results.length).to eq 2
|
|
|
|
expect(results.first.domain).to eq 'example.com'
|
|
|
|
expect(results.first.accounts_count).to eq 2
|
|
|
|
expect(results.last.domain).to eq 'example2.com'
|
|
|
|
expect(results.last.accounts_count).to eq 1
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-04-04 22:29:56 +00:00
|
|
|
describe 'local' do
|
|
|
|
it 'returns an array of accounts who do not have a domain' do
|
|
|
|
account_1 = Fabricate(:account, domain: nil)
|
|
|
|
account_2 = Fabricate(:account, domain: 'example.com')
|
2019-07-18 23:44:42 +00:00
|
|
|
expect(Account.where('id > 0').local).to match_array([account_1])
|
2017-04-04 22:29:56 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-06-22 16:34:27 +00:00
|
|
|
describe 'partitioned' do
|
|
|
|
it 'returns a relation of accounts partitioned by domain' do
|
|
|
|
matches = ['a', 'b', 'a', 'b']
|
|
|
|
matches.size.times.to_a.shuffle.each do |index|
|
|
|
|
matches[index] = Fabricate(:account, domain: matches[index])
|
|
|
|
end
|
|
|
|
|
2019-07-18 23:44:42 +00:00
|
|
|
expect(Account.where('id > 0').partitioned).to match_array(matches)
|
2017-06-22 16:34:27 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe 'recent' do
|
|
|
|
it 'returns a relation of accounts sorted by recent creation' do
|
|
|
|
matches = 2.times.map { Fabricate(:account) }
|
2019-07-18 23:44:42 +00:00
|
|
|
expect(Account.where('id > 0').recent).to match_array(matches)
|
2017-06-22 16:34:27 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-04-04 22:29:56 +00:00
|
|
|
describe 'silenced' do
|
|
|
|
it 'returns an array of accounts who are silenced' do
|
|
|
|
account_1 = Fabricate(:account, silenced: true)
|
|
|
|
account_2 = Fabricate(:account, silenced: false)
|
|
|
|
expect(Account.silenced).to match_array([account_1])
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe 'suspended' do
|
|
|
|
it 'returns an array of accounts who are suspended' do
|
|
|
|
account_1 = Fabricate(:account, suspended: true)
|
|
|
|
account_2 = Fabricate(:account, suspended: false)
|
|
|
|
expect(Account.suspended).to match_array([account_1])
|
|
|
|
end
|
|
|
|
end
|
2017-06-22 16:34:27 +00:00
|
|
|
end
|
2017-04-10 22:38:58 +00:00
|
|
|
|
2017-06-22 16:34:27 +00:00
|
|
|
context 'when is local' do
|
2018-05-02 13:45:24 +00:00
|
|
|
# Test disabled because test environment omits autogenerating keys for performance
|
|
|
|
xit 'generates keys' do
|
2019-08-06 13:33:03 +00:00
|
|
|
account = Account.create!(domain: nil, username: Faker::Internet.user_name(separators: ['_']))
|
2017-06-22 16:34:27 +00:00
|
|
|
expect(account.keypair.private?).to eq true
|
|
|
|
end
|
|
|
|
end
|
2017-04-10 22:38:58 +00:00
|
|
|
|
2017-06-22 16:34:27 +00:00
|
|
|
context 'when is remote' do
|
|
|
|
it 'does not generate keys' do
|
|
|
|
key = OpenSSL::PKey::RSA.new(1024).public_key
|
2019-08-06 13:33:03 +00:00
|
|
|
account = Account.create!(domain: 'remote', username: Faker::Internet.user_name(separators: ['_']), public_key: key.to_pem)
|
2017-06-22 16:34:27 +00:00
|
|
|
expect(account.keypair.params).to eq key.params
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'normalizes domain' do
|
2019-08-06 13:33:03 +00:00
|
|
|
account = Account.create!(domain: 'にゃん', username: Faker::Internet.user_name(separators: ['_']))
|
2017-06-22 16:34:27 +00:00
|
|
|
expect(account.domain).to eq 'xn--r9j5b5b'
|
2017-04-10 22:38:58 +00:00
|
|
|
end
|
|
|
|
end
|
2017-06-22 16:34:27 +00:00
|
|
|
|
|
|
|
include_examples 'AccountAvatar', :account
|
2020-01-04 00:54:07 +00:00
|
|
|
include_examples 'AccountHeader', :account
|
Backport fixes to 3.2 (#15360)
* Fix 2FA/sign-in token sessions being valid after password change (#14802)
If someone tries logging in to an account and is prompted for a 2FA
code or sign-in token, even if the account's password or e-mail is
updated in the meantime, the session will show the prompt and allow
the login process to complete with a valid 2FA code or sign-in token
* Fix Move handler not being triggered when failing to fetch target (#15107)
When failing to fetch the target account, the ProcessingWorker fails
as expected, but since it hasn't cleared the `move_in_progress` flag,
the next attempt at processing skips the `Move` activity altogether.
This commit changes it to clear the flag when encountering any
unexpected error on fetching the target account. This is likely to
occur because, of, e.g., a timeout, when many instances query the
same actor at the same time.
* Fix slow distinct queries where grouped queries are faster (#15287)
About 2x speed-up on inboxes query
* Fix possible inconsistencies in tag search (#14906)
Do not downcase the queried tag before passing it to postgres when searching:
- tags are not downcased on creation
- `arel_table[:name].lower.matches(pattern)` generates an ILIKE anyway
- if Postgres and Rails happen to use different case-folding rules,
downcasing before query but not before insertion may mean that some
tags with some casings are not searchable
* Fix updating account counters when account_stat is not yet created (#15108)
* Fix account processing failing because of large collections (#15027)
Fixes #15025
* Fix downloading remote media files when server returns empty filename (#14867)
Fixes #14817
* Fix webfinger redirect handling in ResolveAccountService (#15187)
* Fix webfinger redirect handling in ResolveAccountService
ResolveAccountService#process_webfinger! handled a one-step webfinger
redirection, but only accepting the result if it matched the exact URI passed
as input, defeating the point of a redirection check.
Instead, use the same logic as in `ActivityPub::FetchRemoteAccountService`,
updating the resulting `acct:` URI with the result of the first webfinger
query.
* Add tests
* Remove dependency on unused and unmaintained http_parser.rb gem (#14574)
It seems that years ago, the “http” gem dependend on the “http_parser.rb” gem
(it now depends on the “http-parser” gem), and, still years ago, we pulled
it from git in order to benefit from a bugfix that wasn't released yet (#7467).
* Add tootctl maintenance fix-duplicates (#14860, #15201, #15264, #15349, #15359)
* Fix old migration script not being able to run if it fails midway (#15361)
* Fix old migration script not being able to run if it fails midway
Improve the robustness of a migration script likely to fail because of database
corruption so it can run again once database corruptions are fixed.
* Display a specific error message in case of index corruption
Co-authored-by: Eugen Rochko <eugen@zeonfederated.com>
Co-authored-by: Claire <claire.github-309c@sitedethib.com>
Co-authored-by: Eugen Rochko <eugen@zeonfederated.com>
Co-authored-by: Claire <claire.github-309c@sitedethib.com>
2020-12-18 22:31:14 +00:00
|
|
|
|
|
|
|
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
|
|
|
|
|
|
|
|
increment_by = 15
|
|
|
|
wait_for_start = true
|
|
|
|
|
|
|
|
threads = Array.new(increment_by) do
|
|
|
|
Thread.new do
|
|
|
|
true while wait_for_start
|
|
|
|
Account.find(subject.id).increment_count!(:followers_count)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
wait_for_start = false
|
|
|
|
threads.each(&:join)
|
|
|
|
|
|
|
|
expect(subject.reload.followers_count).to eq 15
|
|
|
|
end
|
|
|
|
end
|
2016-02-20 21:53:20 +00:00
|
|
|
end
|