You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
345 lines
11 KiB
345 lines
11 KiB
# frozen_string_literal: true |
|
|
|
require 'rails_helper' |
|
|
|
RSpec.describe StatusCacheHydrator do |
|
let(:status) { Fabricate(:status) } |
|
let(:account) { Fabricate(:account) } |
|
|
|
describe '#hydrate' do |
|
let(:compare_to_hash) { InlineRenderer.render(status, account, :status) } |
|
|
|
shared_examples 'shared behavior' do |
|
context 'when handling a new status' do |
|
let(:poll) { Fabricate(:poll) } |
|
let(:status) { Fabricate(:status, poll: poll) } |
|
|
|
it 'renders the same attributes as a full render' do |
|
expect(subject).to eql(compare_to_hash) |
|
end |
|
end |
|
|
|
context 'when handling a new status with own poll' do |
|
let(:poll) { Fabricate(:poll, account: account) } |
|
let(:status) { Fabricate(:status, poll: poll, account: account) } |
|
|
|
it 'renders the same attributes as a full render' do |
|
expect(subject).to eql(compare_to_hash) |
|
end |
|
end |
|
|
|
context 'when handling a status with a quote policy' do |
|
let(:status) { Fabricate(:status, quote_approval_policy: Status::QUOTE_APPROVAL_POLICY_FLAGS[:followers] << 16) } |
|
|
|
before do |
|
account.follow!(status.account) |
|
end |
|
|
|
it 'renders the same attributes as a full render' do |
|
expect(subject).to eql(compare_to_hash) |
|
end |
|
end |
|
|
|
context 'when handling a filtered status' do |
|
let(:status) { Fabricate(:status, text: 'this toot is about that banned word') } |
|
|
|
before do |
|
account.custom_filters.create!(phrase: 'filter1', context: %w(home), action: :hide, keywords_attributes: [{ keyword: 'banned' }, { keyword: 'irrelevant' }]) |
|
end |
|
|
|
it 'renders the same attributes as a full render' do |
|
expect(subject).to eql(compare_to_hash) |
|
end |
|
end |
|
|
|
context 'when handling an unapproved quote' do |
|
let(:quoted_status) { Fabricate(:status) } |
|
|
|
before do |
|
Fabricate(:quote, status: status, quoted_status: quoted_status, state: :pending) |
|
end |
|
|
|
it 'renders the same attributes as full render' do |
|
expect(subject).to eql(compare_to_hash) |
|
expect(subject[:quote]).to_not be_nil |
|
expect(subject[:quote_status]).to be_nil |
|
end |
|
end |
|
|
|
context 'when handling an approved quote' do |
|
let(:quoted_status) { Fabricate(:status) } |
|
let(:legacy) { false } |
|
|
|
before do |
|
Fabricate(:quote, status: status, quoted_status: quoted_status, state: :accepted, legacy: legacy) |
|
end |
|
|
|
it 'renders the same attributes as full render' do |
|
expect(subject).to eql(compare_to_hash) |
|
expect(subject[:quote]).to_not be_nil |
|
end |
|
|
|
context 'when the quote post is recursive' do |
|
let(:quoted_status) { status } |
|
|
|
it 'renders the same attributes as full render' do |
|
expect(subject).to eql(compare_to_hash) |
|
expect(subject[:quote]).to_not be_nil |
|
end |
|
end |
|
|
|
context 'when the quote post is a legacy quote' do |
|
let(:legacy) { true } |
|
|
|
it 'renders the same attributes as full render' do |
|
expect(subject).to eql(compare_to_hash) |
|
expect(subject[:quote]).to_not be_nil |
|
end |
|
end |
|
|
|
context 'when the quoted post is a private post the viewer is not authorized to see' do |
|
let(:quoted_status) { Fabricate(:status, account: status.account, visibility: :private) } |
|
|
|
it 'renders the same attributes as full render' do |
|
expect(subject).to eql(compare_to_hash) |
|
expect(subject[:quote]).to_not be_nil |
|
expect(subject[:quote][:quoted_status]).to be_nil |
|
end |
|
end |
|
|
|
context 'when the quoted post is a private post the viewer is authorized to see' do |
|
let(:quoted_status) { Fabricate(:status, account: status.account, visibility: :private) } |
|
|
|
before do |
|
account.follow!(quoted_status.account) |
|
end |
|
|
|
it 'renders the same attributes as full render' do |
|
expect(subject).to eql(compare_to_hash) |
|
expect(subject[:quote]).to_not be_nil |
|
expect(subject[:quote][:quoted_status]).to_not be_nil |
|
end |
|
end |
|
|
|
context 'when the quoted post has been deleted' do |
|
let(:quoted_status) { nil } |
|
|
|
it 'returns the same attributes as full render' do |
|
expect(subject).to eql(compare_to_hash) |
|
expect(subject[:quote]).to_not be_nil |
|
expect(subject[:quote][:quoted_status]).to be_nil |
|
end |
|
end |
|
|
|
context 'when the quoted post author has blocked the viewer' do |
|
before do |
|
quoted_status.account.block!(account) |
|
end |
|
|
|
it 'returns the same attributes as full render' do |
|
expect(subject).to eql(compare_to_hash) |
|
expect(subject[:quote]).to_not be_nil |
|
expect(subject[:quote][:quoted_status]).to be_nil |
|
end |
|
end |
|
|
|
context 'when the viewer has blocked the quoted post author' do |
|
before do |
|
account.block!(quoted_status.account) |
|
end |
|
|
|
it 'returns the same attributes as full render' do |
|
expect(subject).to eql(compare_to_hash) |
|
expect(subject[:quote]).to_not be_nil |
|
end |
|
end |
|
|
|
context 'when the quoted post has been favourited' do |
|
before do |
|
FavouriteService.new.call(account, quoted_status) |
|
end |
|
|
|
it 'renders the same attributes as full render' do |
|
expect(subject).to eql(compare_to_hash) |
|
expect(subject[:quote]).to_not be_nil |
|
end |
|
end |
|
|
|
context 'when the quoted post has been reblogged' do |
|
before do |
|
ReblogService.new.call(account, quoted_status) |
|
end |
|
|
|
it 'renders the same attributes as full render' do |
|
expect(subject).to eql(compare_to_hash) |
|
expect(subject[:quote]).to_not be_nil |
|
end |
|
end |
|
|
|
context 'when the quoted post matches account filters' do |
|
let(:quoted_status) { Fabricate(:status, text: 'this toot is about that banned word') } |
|
|
|
before do |
|
account.custom_filters.create!(phrase: 'filter1', context: %w(home), action: :hide, keywords_attributes: [{ keyword: 'banned' }, { keyword: 'irrelevant' }]) |
|
end |
|
|
|
it 'renders the same attributes as a full render' do |
|
expect(subject).to eql(compare_to_hash) |
|
expect(subject[:quote]).to_not be_nil |
|
end |
|
end |
|
end |
|
|
|
context 'when handling a reblog' do |
|
let(:reblog) { Fabricate(:status) } |
|
let(:status) { Fabricate(:status, reblog: reblog) } |
|
|
|
context 'when the reblog has an approved quote' do |
|
let(:quoted_status) { Fabricate(:status) } |
|
|
|
before do |
|
Fabricate(:quote, status: reblog, quoted_status: quoted_status, state: :accepted) |
|
end |
|
|
|
it 'renders the same attributes as full render' do |
|
expect(subject).to eql(compare_to_hash) |
|
expect(subject[:reblog][:quote]).to_not be_nil |
|
end |
|
|
|
context 'when the quoted post has been favourited' do |
|
before do |
|
FavouriteService.new.call(account, quoted_status) |
|
end |
|
|
|
it 'renders the same attributes as full render' do |
|
expect(subject).to eql(compare_to_hash) |
|
expect(subject[:reblog][:quote]).to_not be_nil |
|
end |
|
end |
|
|
|
context 'when the quoted post has been reblogged' do |
|
before do |
|
ReblogService.new.call(account, quoted_status) |
|
end |
|
|
|
it 'renders the same attributes as full render' do |
|
expect(subject).to eql(compare_to_hash) |
|
expect(subject[:reblog][:quote]).to_not be_nil |
|
end |
|
end |
|
|
|
context 'when the quoted post matches account filters' do |
|
let(:quoted_status) { Fabricate(:status, text: 'this toot is about that banned word') } |
|
|
|
before do |
|
account.custom_filters.create!(phrase: 'filter1', context: %w(home), action: :hide, keywords_attributes: [{ keyword: 'banned' }, { keyword: 'irrelevant' }]) |
|
end |
|
|
|
it 'renders the same attributes as a full render' do |
|
expect(subject).to eql(compare_to_hash) |
|
expect(subject[:reblog][:quote]).to_not be_nil |
|
end |
|
end |
|
end |
|
|
|
context 'when it has been favourited' do |
|
before do |
|
FavouriteService.new.call(account, reblog) |
|
end |
|
|
|
it 'renders the same attributes as a full render' do |
|
expect(subject).to eql(compare_to_hash) |
|
end |
|
end |
|
|
|
context 'when it has been reblogged' do |
|
before do |
|
ReblogService.new.call(account, reblog) |
|
end |
|
|
|
it 'renders the same attributes as a full render' do |
|
expect(subject).to eql(compare_to_hash) |
|
end |
|
end |
|
|
|
context 'when it has been pinned' do |
|
let(:reblog) { Fabricate(:status, account: account) } |
|
|
|
before do |
|
StatusPin.create!(account: account, status: reblog) |
|
end |
|
|
|
it 'renders the same attributes as a full render' do |
|
expect(subject).to eql(compare_to_hash) |
|
end |
|
end |
|
|
|
context 'when it has been followed tags' do |
|
let(:followed_tag) { Fabricate(:tag) } |
|
|
|
before do |
|
reblog.tags << Fabricate(:tag) |
|
reblog.tags << followed_tag |
|
TagFollow.create!(tag: followed_tag, account: account, rate_limit: false) |
|
end |
|
|
|
it 'renders the same attributes as a full render' do |
|
expect(subject).to eql(compare_to_hash) |
|
end |
|
end |
|
|
|
context 'when it has a poll authored by the user' do |
|
let(:poll) { Fabricate(:poll, account: account) } |
|
let(:reblog) { Fabricate(:status, poll: poll, account: account) } |
|
|
|
it 'renders the same attributes as a full render' do |
|
expect(subject).to eql(compare_to_hash) |
|
end |
|
end |
|
|
|
context 'when it has been voted in' do |
|
let(:poll) { Fabricate(:poll, options: %w(Yellow Blue)) } |
|
let(:reblog) { Fabricate(:status, poll: poll) } |
|
|
|
before do |
|
VoteService.new.call(account, poll, [0]) |
|
end |
|
|
|
it 'renders the same attributes as a full render' do |
|
expect(subject).to eql(compare_to_hash) |
|
end |
|
end |
|
|
|
context 'when it matches account filters' do |
|
let(:reblog) { Fabricate(:status, text: 'this toot is about that banned word') } |
|
|
|
before do |
|
account.custom_filters.create!(phrase: 'filter1', context: %w(home), action: :hide, keywords_attributes: [{ keyword: 'banned' }, { keyword: 'irrelevant' }]) |
|
end |
|
|
|
it 'renders the same attributes as a full render' do |
|
expect(subject).to eql(compare_to_hash) |
|
end |
|
end |
|
end |
|
end |
|
|
|
context 'when cache is warm' do |
|
subject do |
|
Rails.cache.write("fan-out/#{status.id}", InlineRenderer.render(status, nil, :status)) |
|
described_class.new(status).hydrate(account.id) |
|
end |
|
|
|
it_behaves_like 'shared behavior' |
|
end |
|
|
|
context 'when cache is cold' do |
|
subject do |
|
Rails.cache.delete("fan-out/#{status.id}") |
|
described_class.new(status).hydrate(account.id) |
|
end |
|
|
|
it_behaves_like 'shared behavior' |
|
end |
|
end |
|
end
|
|
|