# frozen_string_literal: true require 'rails_helper' RSpec.describe MediaAttachment, :attachment_processing do describe 'local?' do subject { media_attachment.local? } let(:media_attachment) { described_class.new(remote_url: remote_url) } context 'when remote_url is blank' do let(:remote_url) { '' } it 'returns true' do expect(subject).to be true end end context 'when remote_url is present' do let(:remote_url) { 'remote_url' } it 'returns false' do expect(subject).to be false end end end describe 'needs_redownload?' do subject { media_attachment.needs_redownload? } let(:media_attachment) { described_class.new(remote_url: remote_url, file: file) } context 'when file is blank' do let(:file) { nil } context 'when remote_url is present' do let(:remote_url) { 'remote_url' } it 'returns true' do expect(subject).to be true end end end context 'when file is present' do let(:file) { attachment_fixture('avatar.gif') } context 'when remote_url is blank' do let(:remote_url) { '' } it 'returns false' do expect(subject).to be false end end context 'when remote_url is present' do let(:remote_url) { 'remote_url' } it 'returns true' do expect(subject).to be false end end end end describe '#to_param' do let(:media_attachment) { Fabricate.build(:media_attachment, shortcode: shortcode, id: id) } context 'when media attachment has a shortcode' do let(:shortcode) { 'foo' } let(:id) { 123 } it 'returns shortcode' do expect(media_attachment.to_param).to eq shortcode end end context 'when media attachment does not have a shortcode' do let(:shortcode) { nil } let(:id) { 123 } it 'returns string representation of id' do expect(media_attachment.to_param).to eq id.to_s end end end shared_examples 'static 600x400 image' do |content_type, extension| after do media.destroy end it 'saves metadata and generates styles' do expect(media) .to be_persisted .and be_processing_complete .and have_attributes( file: be_present, type: eq('image'), file_content_type: eq(content_type), file_file_name: end_with(extension), blurhash: have_attributes(size: eq(36)) ) # Strip original file name expect(media.file_file_name) .to_not start_with '600x400' # Generate styles expect(FastImage.size(media.file.path(:original))) .to eq [600, 400] expect(FastImage.size(media.file.path(:small))) .to eq [588, 392] # Use extension recognized by Rack::Mime (used by PublicFileServerMiddleware) expect(media.file.path(:original)) .to end_with(extension) expect(media.file.path(:small)) .to end_with(extension) # Set meta for original and thumbnail expect(media_metadata) .to include( original: include( width: eq(600), height: eq(400), aspect: eq(1.5) ), small: include( width: eq(588), height: eq(392), aspect: eq(1.5) ) ) # Rack::Mime (used by PublicFileServerMiddleware) recognizes file extension expect(Rack::Mime.mime_type(extension, nil)).to eq content_type end end shared_examples 'animated 600x400 image' do after do media.destroy end it 'saves metadata and generates styles' do expect(media) .to be_persisted .and be_processing_complete .and have_attributes( file: be_present, type: eq('gifv'), file_content_type: eq('video/mp4'), file_file_name: end_with('.mp4'), blurhash: have_attributes(size: eq(36)) ) # Strip original file name expect(media.file_file_name) .to_not start_with '600x400' # Transcode to MP4 expect(media.file.path(:original)) .to end_with('.mp4') # Generate static thumbnail expect(FastImage.size(media.file.path(:small))) .to eq [600, 400] expect(FastImage.animated?(media.file.path(:small))) .to be false expect(media.file.path(:small)) .to end_with('.png') # Set meta for styles expect(media_metadata) .to include( original: include( width: eq(600), height: eq(400), duration: eq(3), frame_rate: '1/1' ), small: include( width: eq(600), height: eq(400), aspect: eq(1.5) ) ) end end describe 'jpeg' do let(:media) { Fabricate(:media_attachment, file: attachment_fixture('600x400.jpeg')) } it_behaves_like 'static 600x400 image', 'image/jpeg', '.jpeg' end describe 'png' do let(:media) { Fabricate(:media_attachment, file: attachment_fixture('600x400.png')) } it_behaves_like 'static 600x400 image', 'image/png', '.png' end describe 'gif' do let(:media) { Fabricate(:media_attachment, file: attachment_fixture('600x400.gif')) } it_behaves_like 'static 600x400 image', 'image/gif', '.gif' end describe 'webp' do let(:media) { Fabricate(:media_attachment, file: attachment_fixture('600x400.webp')) } it_behaves_like 'static 600x400 image', 'image/webp', '.webp' end describe 'avif' do let(:media) { Fabricate(:media_attachment, file: attachment_fixture('600x400.avif')) } it_behaves_like 'static 600x400 image', 'image/jpeg', '.jpeg' end describe 'heic' do let(:media) { Fabricate(:media_attachment, file: attachment_fixture('600x400.heic')) } it_behaves_like 'static 600x400 image', 'image/jpeg', '.jpeg' end describe 'monochrome jpg' do let(:media) { Fabricate(:media_attachment, file: attachment_fixture('monochrome.png')) } it_behaves_like 'static 600x400 image', 'image/png', '.png' end describe 'base64-encoded image' do let(:base64_attachment) { "data:image/jpeg;base64,#{Base64.encode64(attachment_fixture('600x400.jpeg').read)}" } let(:media) { Fabricate(:media_attachment, file: base64_attachment) } it_behaves_like 'static 600x400 image', 'image/jpeg', '.jpeg' end describe 'animated gif' do let(:media) { Fabricate(:media_attachment, file: attachment_fixture('600x400-animated.gif')) } it_behaves_like 'animated 600x400 image' end describe 'animated png' do let(:media) { Fabricate(:media_attachment, file: attachment_fixture('600x400-animated.png')) } it_behaves_like 'animated 600x400 image' end describe 'ogg with cover art' do let(:media) { Fabricate(:media_attachment, file: attachment_fixture('boop.ogg')) } let(:expected_media_duration) { 0.235102 } # The libvips and ImageMagick implementations produce different results let(:expected_background_color) { Rails.configuration.x.use_vips ? '#268cd9' : '#3088d4' } it 'sets correct file metadata' do expect(media) .to have_attributes( type: eq('audio'), thumbnail: be_present, file_file_name: not_eq('boop.ogg') ) expect(media_metadata) .to include( original: include(duration: be_within(0.05).of(expected_media_duration)), colors: include(background: eq(expected_background_color)) ) end end describe 'mp3 with large cover art' do let(:media) { Fabricate(:media_attachment, file: attachment_fixture('boop.mp3')) } let(:expected_media_duration) { 0.235102 } it 'detects file type and sets correct metadata' do expect(media) .to have_attributes( type: eq('audio'), thumbnail: be_present, file_file_name: not_eq('boop.mp3') ) expect(media_metadata) .to include( original: include(duration: be_within(0.05).of(expected_media_duration)) ) end end it { is_expected.to validate_presence_of(:file) } describe 'size limit validation' do it 'rejects video files that are too large' do stub_const 'MediaAttachment::IMAGE_LIMIT', 100.megabytes stub_const 'MediaAttachment::VIDEO_LIMIT', 1.kilobyte expect { Fabricate(:media_attachment, file: attachment_fixture('attachment.webm')) }.to raise_error(ActiveRecord::RecordInvalid) end it 'accepts video files that are small enough' do stub_const 'MediaAttachment::IMAGE_LIMIT', 1.kilobyte stub_const 'MediaAttachment::VIDEO_LIMIT', 100.megabytes media = Fabricate(:media_attachment, file: attachment_fixture('attachment.webm')) expect(media.valid?).to be true end it 'rejects image files that are too large' do stub_const 'MediaAttachment::IMAGE_LIMIT', 1.kilobyte stub_const 'MediaAttachment::VIDEO_LIMIT', 100.megabytes expect { Fabricate(:media_attachment, file: attachment_fixture('attachment.jpg')) }.to raise_error(ActiveRecord::RecordInvalid) end it 'accepts image files that are small enough' do stub_const 'MediaAttachment::IMAGE_LIMIT', 100.megabytes stub_const 'MediaAttachment::VIDEO_LIMIT', 1.kilobyte media = Fabricate(:media_attachment, file: attachment_fixture('attachment.jpg')) expect(media.valid?).to be true end end describe 'cache deletion hooks' do let(:media) { Fabricate(:media_attachment) } before do allow(Rails.configuration.x).to receive(:cache_buster_enabled).and_return(true) end it 'queues CacheBusterWorker jobs' do original_path = media.file.path(:original) small_path = media.file.path(:small) expect { media.destroy } .to enqueue_sidekiq_job(CacheBusterWorker).with(original_path) .and enqueue_sidekiq_job(CacheBusterWorker).with(small_path) end end private def media_metadata media.file.meta.deep_symbolize_keys end end