scope :without_replies, -> { where('statuses.reply = FALSE OR statuses.in_reply_to_account_id = statuses.account_id') }
scope :without_reblogs, -> { where('statuses.reblog_of_id IS NULL') }
+ scope :with_public_visibility, -> { where(visibility: :public) }
+ scope :tagged_with, ->(tag) { joins(:statuses_tags).where(statuses_tags: { tag_id: tag }) }
+ scope :local_only, -> { left_outer_joins(:account).where(accounts: { domain: nil }) }
+ scope :excluding_silenced_accounts, -> { left_outer_joins(:account).where(accounts: { silenced: false }) }
+ scope :including_silenced_accounts, -> { left_outer_joins(:account).where(accounts: { silenced: true }) }
+ scope :not_excluded_by_account, ->(account) { where.not(account_id: account.excluded_from_timeline_account_ids) }
cache_associated :account, :application, :media_attachments, :tags, :stream_entry, mentions: :account, reblog: [:account, :application, :stream_entry, :tags, :media_attachments, mentions: :account], thread: :account
end
def as_public_timeline(account = nil, local_only = false)
- query = joins('LEFT OUTER JOIN accounts ON statuses.account_id = accounts.id')
- .where(visibility: :public)
- .without_replies
- .without_reblogs
+ query = timeline_scope(local_only).without_replies
- query = query.where('accounts.domain IS NULL') if local_only
-
- account.nil? ? filter_timeline_default(query) : filter_timeline_default(filter_timeline(query, account))
+ apply_timeline_filters(query, account)
end
def as_tag_timeline(tag, account = nil, local_only = false)
- query = tag.statuses
- .joins('LEFT OUTER JOIN accounts ON statuses.account_id = accounts.id')
- .where(visibility: :public)
- .without_reblogs
-
- query = query.where('accounts.domain IS NULL') if local_only
+ query = timeline_scope(local_only).tagged_with(tag)
- account.nil? ? filter_timeline_default(query) : filter_timeline_default(filter_timeline(query, account))
+ apply_timeline_filters(query, account)
end
def as_outbox_timeline(account)
private
- def filter_timeline(query, account)
- blocked = Rails.cache.fetch("exclude_account_ids_for:#{account.id}") { Block.where(account: account).pluck(:target_account_id) + Block.where(target_account: account).pluck(:account_id) + Mute.where(account: account).pluck(:target_account_id) }
- query = query.where('statuses.account_id NOT IN (?)', blocked) unless blocked.empty? # Only give us statuses from people we haven't blocked, or muted, or that have blocked us
- query = query.where('accounts.silenced = TRUE') if account.silenced? # and if we're hellbanned, only people who are also hellbanned
- query
+ def timeline_scope(local_only = false)
+ starting_scope = local_only ? Status.local_only : Status
+ starting_scope
+ .with_public_visibility
+ .without_reblogs
+ end
+
+ def apply_timeline_filters(query, account)
+ if account.nil?
+ filter_timeline_default(query)
+ else
+ filter_timeline_for_account(query, account)
+ end
+ end
+
+ def filter_timeline_for_account(query, account)
+ query = query.not_excluded_by_account(account)
+ query.merge(account_silencing_filter(account))
end
def filter_timeline_default(query)
- query.where('accounts.silenced = FALSE')
+ query.excluding_silenced_accounts
+ end
+
+ def account_silencing_filter(account)
+ if account.silenced?
+ including_silenced_accounts
+ else
+ excluding_silenced_accounts
+ end
end
end
pending
end
+ describe '.local_only' do
+ it 'returns only statuses from local accounts' do
+ local_account = Fabricate(:account, domain: nil)
+ remote_account = Fabricate(:account, domain: 'test.com')
+ local_status = Fabricate(:status, account: local_account)
+ remote_status = Fabricate(:status, account: remote_account)
+
+ results = described_class.local_only
+ expect(results).to include(local_status)
+ expect(results).not_to include(remote_status)
+ end
+ end
+
describe '.as_home_timeline' do
before do
account = Fabricate(:account)
expect(@results).not_to include(@not_followed_status)
end
end
+
+ describe '.as_public_timeline' do
+ it 'only includes statuses with public visibility' do
+ public_status = Fabricate(:status, visibility: :public)
+ private_status = Fabricate(:status, visibility: :private)
+
+ results = Status.as_public_timeline
+ expect(results).to include(public_status)
+ expect(results).not_to include(private_status)
+ end
+
+ it 'does not include replies' do
+ status = Fabricate(:status)
+ reply = Fabricate(:status, in_reply_to_id: status.id)
+
+ results = Status.as_public_timeline
+ expect(results).to include(status)
+ expect(results).not_to include(reply)
+ end
+
+ it 'does not include boosts' do
+ status = Fabricate(:status)
+ boost = Fabricate(:status, reblog_of_id: status.id)
+
+ results = Status.as_public_timeline
+ expect(results).to include(status)
+ expect(results).not_to include(boost)
+ end
+
+ it 'filters out silenced accounts' do
+ account = Fabricate(:account)
+ silenced_account = Fabricate(:account, silenced: true)
+ status = Fabricate(:status, account: account)
+ silenced_status = Fabricate(:status, account: silenced_account)
+
+ results = Status.as_public_timeline
+ expect(results).to include(status)
+ expect(results).not_to include(silenced_status)
+ end
+
+ context 'with a local_only option set' do
+ it 'does not include remote instances statuses' do
+ local_account = Fabricate(:account, domain: nil)
+ remote_account = Fabricate(:account, domain: 'test.com')
+ local_status = Fabricate(:status, account: local_account)
+ remote_status = Fabricate(:status, account: remote_account)
+
+ results = Status.as_public_timeline(nil, true)
+ expect(results).to include(local_status)
+ expect(results).not_to include(remote_status)
+ end
+ end
+
+ describe 'with an account passed in' do
+ before do
+ @account = Fabricate(:account)
+ end
+
+ it 'excludes statuses from accounts blocked by the account' do
+ blocked = Fabricate(:account)
+ Fabricate(:block, account: @account, target_account: blocked)
+ blocked_status = Fabricate(:status, account: blocked)
+
+ results = Status.as_public_timeline(@account)
+ expect(results).not_to include(blocked_status)
+ end
+
+ it 'excludes statuses from accounts who have blocked the account' do
+ blocked = Fabricate(:account)
+ Fabricate(:block, account: blocked, target_account: @account)
+ blocked_status = Fabricate(:status, account: blocked)
+
+ results = Status.as_public_timeline(@account)
+ expect(results).not_to include(blocked_status)
+ end
+
+ it 'excludes statuses from accounts muted by the account' do
+ muted = Fabricate(:account)
+ Fabricate(:mute, account: @account, target_account: muted)
+ muted_status = Fabricate(:status, account: muted)
+
+ results = Status.as_public_timeline(@account)
+ expect(results).not_to include(muted_status)
+ end
+
+ context 'where that account is silenced' do
+ it 'includes statuses from other accounts that are silenced' do
+ @account.update(silenced: true)
+ other_silenced_account = Fabricate(:account, silenced: true)
+ other_status = Fabricate(:status, account: other_silenced_account)
+
+ results = Status.as_public_timeline(@account)
+ expect(results).to include(other_status)
+ end
+ end
+ end
+ end
+
+ describe '.as_tag_timeline' do
+ it 'includes statuses with a tag' do
+ tag = Fabricate(:tag)
+ status = Fabricate(:status, tags: [tag])
+ other = Fabricate(:status)
+
+ results = Status.as_tag_timeline(tag)
+ expect(results).to include(status)
+ expect(results).not_to include(other)
+ end
+
+ it 'allows replies to be included' do
+ original = Fabricate(:status)
+ tag = Fabricate(:tag)
+ status = Fabricate(:status, tags: [tag], in_reply_to_id: original.id)
+
+ results = Status.as_tag_timeline(tag)
+ expect(results).to include(status)
+ end
+ end
end