@ -1,10 +1,9 @@
require 'rails_helper'
require 'rails_helper'
RSpec . describe Account , type : :model do
RSpec . describe Account , type : :model do
subject { Fabricate ( :account , username : 'alice' ) }
context do
context do
let ( :bob ) { Fabricate ( :account , username : 'bob' ) }
let ( :bob ) { Fabricate ( :account , username : 'bob' ) }
subject { Fabricate ( :account ) }
describe '#follow!' do
describe '#follow!' do
it 'creates a follow' do
it 'creates a follow' do
@ -45,12 +44,13 @@ RSpec.describe Account, type: :model do
describe '#local?' do
describe '#local?' do
it 'returns true when the account is local' do
it 'returns true when the account is local' do
expect ( subject . local? ) . to be true
account = Fabricate ( :account , domain : nil )
expect ( account . local? ) . to be true
end
end
it 'returns false when the account is on a different domain' do
it 'returns false when the account is on a different domain' do
subject . domain = 'foreign.tld'
account = Fabricate ( :account , domain : 'foreign.tld' )
expect ( subjec t. local? ) . to be false
expect ( accoun t. local? ) . to be false
end
end
end
end
@ -61,6 +61,8 @@ RSpec.describe Account, type: :model do
Rails . configuration . x . local_domain = before
Rails . configuration . x . local_domain = before
end
end
subject { Fabricate ( :account , domain : nil , username : 'alice' ) }
describe '#to_webfinger_s' do
describe '#to_webfinger_s' do
it 'returns a webfinger string for the account' do
it 'returns a webfinger string for the account' do
Rails . configuration . x . local_domain = 'example.com'
Rails . configuration . x . local_domain = 'example.com'
@ -80,41 +82,72 @@ RSpec.describe Account, type: :model do
describe '#acct' do
describe '#acct' do
it 'returns username for local users' do
it 'returns username for local users' do
expect ( subject . acct ) . to eql 'alice'
account = Fabricate ( :account , domain : nil , username : 'alice' )
expect ( account . acct ) . to eql 'alice'
end
end
it 'returns username@domain for foreign users' do
it 'returns username@domain for foreign users' do
subject . domain = 'foreign.tld'
account = Fabricate ( :account , domain : 'foreign.tld' , username : 'alice' )
expect ( subject . acct ) . to eql 'alice@foreign.tld'
expect ( account . acct ) . to eql 'alice@foreign.tld'
end
end
describe '#save_with_optional_media!' do
it 'sets default avatar, header, avatar_remote_url, and header_remote_url if some of them are invalid' do
stub_request ( :get , 'https://remote/valid_avatar' ) . to_return ( request_fixture ( 'avatar.txt' ) )
stub_request ( :get , 'https://remote/invalid_avatar' ) . to_return ( request_fixture ( 'feed.txt' ) )
account = Fabricate ( :account ,
avatar_remote_url : 'https://remote/valid_avatar' ,
header_remote_url : 'https://remote/valid_avatar' )
account . avatar_remote_url = 'https://remote/invalid_avatar'
account . save_with_optional_media!
account . reload
expect ( account . avatar_remote_url ) . to eq ''
expect ( account . header_remote_url ) . to eq ''
expect ( account . avatar_file_name ) . to eq nil
expect ( account . header_file_name ) . to eq nil
end
end
end
end
describe '#subscribed?' do
describe '#subscribed?' do
it 'returns false when no subscription expiration information is present' do
it 'returns false when no subscription expiration information is present' do
expect ( subject . subscribed? ) . to be false
account = Fabricate ( :account , subscription_expires_at : nil )
expect ( account . subscribed? ) . to be false
end
end
it 'returns true when subscription expiration has been set' do
it 'returns true when subscription expiration has been set' do
subject . subscription_expires_at = 30 . days . from_now
account = Fabricate ( :account , subscription_expires_at : 30 . days . from_now )
expect ( subject . subscribed? ) . to be true
expect ( account . subscribed? ) . to be true
end
end
describe '#to_param' do
it 'returns username' do
account = Fabricate ( :account , username : 'alice' )
expect ( account . to_param ) . to eq 'alice'
end
end
end
end
describe '#keypair' do
describe '#keypair' do
it 'returns an RSA key pair' do
it 'returns an RSA key pair' do
expect ( subject . keypair ) . to be_instance_of OpenSSL :: PKey :: RSA
account = Fabricate ( :account )
expect ( account . keypair ) . to be_instance_of OpenSSL :: PKey :: RSA
end
end
end
end
describe '#subscription' do
describe '#subscription' do
it 'returns an OStatus subscription' do
it 'returns an OStatus subscription' do
expect ( subject . subscription ( '' ) ) . to be_instance_of OStatus2 :: Subscription
account = Fabricate ( :account )
expect ( account . subscription ( '' ) ) . to be_instance_of OStatus2 :: Subscription
end
end
end
end
describe '#object_type' do
describe '#object_type' do
it 'is always a person' do
it 'is always a person' do
expect ( subject . object_type ) . to be :person
account = Fabricate ( :account )
expect ( account . object_type ) . to be :person
end
end
end
end
@ -124,6 +157,8 @@ RSpec.describe Account, type: :model do
Fabricate ( :status , account : author )
Fabricate ( :status , account : author )
end
end
subject { Fabricate ( :account ) }
context 'when the status is a reblog of another status' do
context 'when the status is a reblog of another status' do
let ( :original_reblog ) do
let ( :original_reblog ) do
author = Fabricate ( :account , username : 'original_reblogger' )
author = Fabricate ( :account , username : 'original_reblogger' )
@ -160,6 +195,8 @@ RSpec.describe Account, type: :model do
Fabricate ( :status , account : author )
Fabricate ( :status , account : author )
end
end
subject { Fabricate ( :account ) }
context 'when the status is a reblog of another status' do
context 'when the status is a reblog of another status' do
let ( :original_reblog ) do
let ( :original_reblog ) do
author = Fabricate ( :account , username : 'original_reblogger' )
author = Fabricate ( :account , username : 'original_reblogger' )
@ -205,14 +242,16 @@ RSpec.describe Account, type: :model do
end
end
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 match_array [ 'domain' ]
end
end
describe '.search_for' do
describe '.search_for' do
before do
before do
@match = Fabricate (
:account ,
display_name : " Display Name " ,
username : " username " ,
domain : " example.com "
)
_missing = Fabricate (
_missing = Fabricate (
:account ,
:account ,
display_name : " Missing " ,
display_name : " Missing " ,
@ -221,33 +260,103 @@ RSpec.describe Account, type: :model do
)
)
end
end
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
it 'finds accounts with matching display_name' do
it 'finds accounts with matching display_name' do
match = Fabricate (
:account ,
display_name : " Display Name " ,
username : " username " ,
domain : " example.com "
)
results = Account . search_for ( " display " )
results = Account . search_for ( " display " )
expect ( results ) . to eq [ @match ]
expect ( results ) . to eq [ match ]
end
end
it 'finds accounts with matching username' do
it 'finds accounts with matching username' do
match = Fabricate (
:account ,
display_name : " Display Name " ,
username : " username " ,
domain : " example.com "
)
results = Account . search_for ( " username " )
results = Account . search_for ( " username " )
expect ( results ) . to eq [ @match ]
expect ( results ) . to eq [ match ]
end
end
it 'finds accounts with matching domain' do
it 'finds accounts with matching domain' do
match = Fabricate (
:account ,
display_name : " Display Name " ,
username : " username " ,
domain : " example.com "
)
results = Account . search_for ( " example " )
results = Account . search_for ( " example " )
expect ( results ) . to eq [ @match ]
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
end
end
it 'ranks multiple matches higher' do
it 'ranks multiple matches higher' do
account = Fabricate (
matches = [
:account ,
{ username : " username " , display_name : " username " } ,
username : " username " ,
{ display_name : " Display Name " , username : " username " , domain : " example.com " } ,
display_name : " username "
] . map ( & method ( :Fabricate ) . curry ( 2 ) . call ( :account ) )
)
results = Account . search_for ( " username " )
results = Account . search_for ( " username " )
expect ( results ) . to eq [ account , @match ]
expect ( results ) . to eq matches
end
end
end
end
describe '.advanced_search_for' do
describe '.advanced_search_for' do
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
it 'ranks followed accounts higher' do
it 'ranks followed accounts higher' do
account = Fabricate ( :account )
account = Fabricate ( :account )
match = Fabricate ( :account , username : " Matching " )
match = Fabricate ( :account , username : " Matching " )
@ -260,9 +369,14 @@ RSpec.describe Account, type: :model do
end
end
end
end
describe '.triadic_closures' do
describe '.domains' do
subject { described_class . triadic_closures ( me ) }
it 'returns domains' do
Fabricate ( :account , domain : 'domain' )
expect ( Account . domains ) . to match_array ( [ 'domain' ] )
end
end
describe '.triadic_closures' do
let! ( :me ) { Fabricate ( :account ) }
let! ( :me ) { Fabricate ( :account ) }
let! ( :friend ) { Fabricate ( :account ) }
let! ( :friend ) { Fabricate ( :account ) }
let! ( :friends_friend ) { Fabricate ( :account ) }
let! ( :friends_friend ) { Fabricate ( :account ) }
@ -277,7 +391,39 @@ RSpec.describe Account, type: :model do
end
end
it 'finds accounts you dont follow which are followed by accounts you do follow' do
it 'finds accounts you dont follow which are followed by accounts you do follow' do
is_expected . to eq [ friends_friend ]
expect ( described_class . triadic_closures ( me ) ) . to eq [ friends_friend ]
end
it 'limits by 5 with offset 0 by defualt' do
first_degree = 6 . times . map { Fabricate ( :account ) }
matches = 5 . times . map { Fabricate ( :account ) }
first_degree . each { | account | me . follow! ( account ) }
matches . each do | match |
first_degree . each { | account | account . follow! ( match ) }
first_degree . shift
end
expect ( described_class . triadic_closures ( me ) ) . to eq matches
end
it 'accepts arbitrary limits' do
another_friend = Fabricate ( :account )
higher_friends_friend = Fabricate ( :account )
me . follow! ( another_friend )
friend . follow! ( higher_friends_friend )
another_friend . follow! ( higher_friends_friend )
expect ( described_class . triadic_closures ( me , limit : 1 ) ) . to eq [ higher_friends_friend ]
end
it 'acceps arbitrary offset' do
another_friend = Fabricate ( :account )
higher_friends_friend = Fabricate ( :account )
me . follow! ( another_friend )
friend . follow! ( higher_friends_friend )
another_friend . follow! ( higher_friends_friend )
expect ( described_class . triadic_closures ( me , offset : 1 ) ) . to eq [ friends_friend ]
end
end
context 'when you block account' do
context 'when you block account' do
@ -286,7 +432,7 @@ RSpec.describe Account, type: :model do
end
end
it 'rejects blocked accounts' do
it 'rejects blocked accounts' do
is_ expected. to be_empty
expect ( describ ed_class . triadic_closures ( me ) ) . to be_empty
end
end
end
end
@ -296,7 +442,7 @@ RSpec.describe Account, type: :model do
end
end
it 'rejects muted accounts' do
it 'rejects muted accounts' do
is_ expected. to be_empty
expect ( describ ed_class . triadic_closures ( me ) ) . to be_empty
end
end
end
end
end
end
@ -374,9 +520,10 @@ RSpec.describe Account, type: :model do
expect ( account ) . to model_have_error_on_field ( :username )
expect ( account ) . to model_have_error_on_field ( :username )
end
end
it 'is invalid if the username already exists' do
context 'when is local' do
it 'is invalid if the username is not unique in case-insensitive comparsion among local accounts' do
account_1 = Fabricate ( :account , username : 'the_doctor' )
account_1 = Fabricate ( :account , username : 'the_doctor' )
account_2 = Fabricate . build ( :account , username : 'the_d octor' )
account_2 = Fabricate . build ( :account , username : 'the_D octor' )
account_2 . valid?
account_2 . valid?
expect ( account_2 ) . to model_have_error_on_field ( :username )
expect ( account_2 ) . to model_have_error_on_field ( :username )
end
end
@ -393,7 +540,6 @@ RSpec.describe Account, type: :model do
expect ( account . valid? ) . to be true
expect ( account . valid? ) . to be true
end
end
context 'when is local' do
it 'is invalid if the username doesn\'t only contains letters, numbers and underscores' do
it 'is invalid if the username doesn\'t only contains letters, numbers and underscores' do
account = Fabricate . build ( :account , username : 'the-doctor' )
account = Fabricate . build ( :account , username : 'the-doctor' )
account . valid?
account . valid?
@ -405,10 +551,109 @@ RSpec.describe Account, type: :model do
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 invalid if the display name is longer than 30 characters' do
account = Fabricate . build ( :account , display_name : Faker :: Lorem . characters ( 31 ) )
account . valid?
expect ( account ) . to model_have_error_on_field ( :display_name )
end
it 'is invalid if the note is longer than 160 characters' do
account = Fabricate . build ( :account , note : Faker :: Lorem . characters ( 161 ) )
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 not unique in case-sensitive 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 is unique only in case-sensitive 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 ) . not_to model_have_error_on_field ( :username )
end
it 'is valid even if the username doesn\'t only contains letters, numbers and underscores' do
account = Fabricate . build ( :account , domain : 'domain' , username : 'the-doctor' )
account . valid?
expect ( account ) . not_to model_have_error_on_field ( :username )
end
it 'is valid even if the username is longer then 30 characters' do
account = Fabricate . build ( :account , domain : 'domain' , username : Faker :: Lorem . characters ( 31 ) )
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
account = Fabricate . build ( :account , domain : 'domain' , display_name : Faker :: Lorem . characters ( 31 ) )
account . valid?
expect ( account ) . not_to model_have_error_on_field ( :display_name )
end
it 'is valid even if the note is longer than 160 characters' do
account = Fabricate . build ( :account , domain : 'domain' , note : Faker :: Lorem . characters ( 161 ) )
account . valid?
expect ( account ) . not_to model_have_error_on_field ( :note )
end
end
end
end
end
describe 'scopes' do
describe 'scopes' do
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 ( Account . 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 ( 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
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 : nil } ,
{ 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
describe 'remote' do
describe 'remote' do
it 'returns an array of accounts who have a domain' do
it 'returns an array of accounts who have a domain' do
account_1 = Fabricate ( :account , domain : nil )
account_1 = Fabricate ( :account , domain : nil )
@ -439,6 +684,24 @@ RSpec.describe Account, type: :model do
end
end
end
end
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
expect ( Account . partitioned ) . to match_array ( matches )
end
end
describe 'recent' do
it 'returns a relation of accounts sorted by recent creation' do
matches = 2 . times . map { Fabricate ( :account ) }
expect ( Account . recent ) . to match_array ( matches )
end
end
describe 'silenced' do
describe 'silenced' do
it 'returns an array of accounts who are silenced' do
it 'returns an array of accounts who are silenced' do
account_1 = Fabricate ( :account , silenced : true )
account_1 = Fabricate ( :account , silenced : true )
@ -454,25 +717,45 @@ RSpec.describe Account, type: :model do
expect ( Account . suspended ) . to match_array ( [ account_1 ] )
expect ( Account . suspended ) . to match_array ( [ account_1 ] )
end
end
end
end
end
describe 'static avatars' do
describe 'without_followers' do
describe 'when GIF' do
it 'returns a relation of accounts without followers' do
it 'creates a png static style' do
account_1 = Fabricate ( :account )
subject . avatar = attachment_fixture ( 'avatar.gif' )
account_2 = Fabricate ( :account )
subject . save
Fabricate ( :follow , account : account_1 , target_account : account_2 )
expect ( Account . without_followers ) . to match_array ( [ account_1 ] )
end
end
expect ( subject . avatar_static_url ) . to_not eq subject . avatar_original_url
describe 'with_followers' do
it 'returns a relation of accounts with followers' do
account_1 = Fabricate ( :account )
account_2 = Fabricate ( :account )
Fabricate ( :follow , account : account_1 , target_account : account_2 )
expect ( Account . with_followers ) . to match_array ( [ account_2 ] )
end
end
end
end
end
describe 'when non-GIF' do
context 'when is local' do
it 'does not create extra static style' do
it 'generates keys' do
subject . avatar = attachment_fixture ( 'attachment.jpg' )
account = Account . create! ( domain : nil , username : Faker :: Internet . user_name ( nil , [ '_' ] ) )
subject . save
expect ( account . keypair . private? ) . to eq true
end
end
expect ( subject . avatar_static_url ) . to eq subject . avatar_original_url
context 'when is remote' do
it 'does not generate keys' do
key = OpenSSL :: PKey :: RSA . new ( 1024 ) . public_key
account = Account . create! ( domain : 'remote' , username : Faker :: Internet . user_name ( nil , [ '_' ] ) , public_key : key . to_pem )
expect ( account . keypair . params ) . to eq key . params
end
end
it 'normalizes domain' do
account = Account . create! ( domain : 'にゃん' , username : Faker :: Internet . user_name ( nil , [ '_' ] ) )
expect ( account . domain ) . to eq 'xn--r9j5b5b'
end
end
end
end
include_examples 'AccountAvatar' , :account
end
end