334 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
			
		
		
	
	
			334 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
# 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 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
 |