2023-02-22 01:55:31 +01:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2016-09-05 17:46:36 +02:00
|
|
|
require 'rails_helper'
|
|
|
|
|
2024-07-08 18:01:08 +02:00
|
|
|
RSpec.describe MediaAttachment, :attachment_processing do
|
2017-11-08 07:29:07 +01:00
|
|
|
describe 'local?' do
|
|
|
|
subject { media_attachment.local? }
|
|
|
|
|
2023-08-07 17:58:12 +02:00
|
|
|
let(:media_attachment) { described_class.new(remote_url: remote_url) }
|
2023-02-20 05:24:14 +01:00
|
|
|
|
2023-05-04 05:49:08 +02:00
|
|
|
context 'when remote_url is blank' do
|
2017-11-08 07:29:07 +01:00
|
|
|
let(:remote_url) { '' }
|
|
|
|
|
|
|
|
it 'returns true' do
|
2023-02-20 05:00:48 +01:00
|
|
|
expect(subject).to be true
|
2017-11-08 07:29:07 +01:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2023-05-04 05:49:08 +02:00
|
|
|
context 'when remote_url is present' do
|
2017-11-08 07:29:07 +01:00
|
|
|
let(:remote_url) { 'remote_url' }
|
|
|
|
|
|
|
|
it 'returns false' do
|
2023-02-20 05:00:48 +01:00
|
|
|
expect(subject).to be false
|
2017-11-08 07:29:07 +01:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe 'needs_redownload?' do
|
|
|
|
subject { media_attachment.needs_redownload? }
|
|
|
|
|
2023-08-07 17:58:12 +02:00
|
|
|
let(:media_attachment) { described_class.new(remote_url: remote_url, file: file) }
|
2023-02-20 05:24:14 +01:00
|
|
|
|
2023-05-04 05:49:08 +02:00
|
|
|
context 'when file is blank' do
|
2017-11-08 07:29:07 +01:00
|
|
|
let(:file) { nil }
|
|
|
|
|
2023-05-04 05:49:08 +02:00
|
|
|
context 'when remote_url is present' do
|
2017-11-08 07:29:07 +01:00
|
|
|
let(:remote_url) { 'remote_url' }
|
|
|
|
|
|
|
|
it 'returns true' do
|
2023-02-20 05:00:48 +01:00
|
|
|
expect(subject).to be true
|
2017-11-08 07:29:07 +01:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2023-05-04 05:49:08 +02:00
|
|
|
context 'when file is present' do
|
2017-11-08 07:29:07 +01:00
|
|
|
let(:file) { attachment_fixture('avatar.gif') }
|
|
|
|
|
2023-05-04 05:49:08 +02:00
|
|
|
context 'when remote_url is blank' do
|
2017-11-08 07:29:07 +01:00
|
|
|
let(:remote_url) { '' }
|
|
|
|
|
|
|
|
it 'returns false' do
|
2023-02-20 05:00:48 +01:00
|
|
|
expect(subject).to be false
|
2017-11-08 07:29:07 +01:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2023-05-04 05:49:08 +02:00
|
|
|
context 'when remote_url is present' do
|
2017-11-08 07:29:07 +01:00
|
|
|
let(:remote_url) { 'remote_url' }
|
|
|
|
|
|
|
|
it 'returns true' do
|
2023-02-20 05:00:48 +01:00
|
|
|
expect(subject).to be false
|
2017-11-08 07:29:07 +01:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '#to_param' do
|
2023-08-07 17:58:12 +02:00
|
|
|
let(:media_attachment) { Fabricate.build(:media_attachment, shortcode: shortcode, id: id) }
|
2017-11-08 07:29:07 +01:00
|
|
|
|
2021-10-13 15:27:19 +02:00
|
|
|
context 'when media attachment has a shortcode' do
|
|
|
|
let(:shortcode) { 'foo' }
|
2023-08-07 17:58:12 +02:00
|
|
|
let(:id) { 123 }
|
2021-10-13 15:27:19 +02:00
|
|
|
|
|
|
|
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 }
|
2023-08-07 17:58:12 +02:00
|
|
|
let(:id) { 123 }
|
2021-10-13 15:27:19 +02:00
|
|
|
|
|
|
|
it 'returns string representation of id' do
|
2023-08-07 17:58:12 +02:00
|
|
|
expect(media_attachment.to_param).to eq id.to_s
|
2021-10-13 15:27:19 +02:00
|
|
|
end
|
2017-11-08 07:29:07 +01:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2023-08-01 19:34:11 +02:00
|
|
|
shared_examples 'static 600x400 image' do |content_type, extension|
|
|
|
|
after do
|
|
|
|
media.destroy
|
|
|
|
end
|
|
|
|
|
2024-12-11 03:28:21 +01:00
|
|
|
it 'saves metadata and generates styles' do
|
2024-02-16 14:00:09 +01:00
|
|
|
expect(media)
|
|
|
|
.to be_persisted
|
|
|
|
.and be_processing_complete
|
|
|
|
.and have_attributes(
|
|
|
|
file: be_present,
|
|
|
|
type: eq('image'),
|
|
|
|
file_content_type: eq(content_type),
|
2024-12-11 03:28:21 +01:00
|
|
|
file_file_name: end_with(extension),
|
|
|
|
blurhash: have_attributes(size: eq(36))
|
2024-02-16 14:00:09 +01:00
|
|
|
)
|
2023-09-04 09:46:33 +02:00
|
|
|
|
2024-07-18 17:23:46 +02:00
|
|
|
# Strip original file name
|
2024-02-16 14:00:09 +01:00
|
|
|
expect(media.file_file_name)
|
|
|
|
.to_not start_with '600x400'
|
|
|
|
|
2024-12-11 03:28:21 +01:00
|
|
|
# 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)
|
|
|
|
|
2024-07-18 17:23:46 +02:00
|
|
|
# Set meta for original and thumbnail
|
2024-12-11 03:28:21 +01:00
|
|
|
expect(media_metadata)
|
2024-02-16 14:00:09 +01:00
|
|
|
.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)
|
|
|
|
)
|
|
|
|
)
|
2024-12-11 03:28:21 +01:00
|
|
|
|
|
|
|
# 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)
|
|
|
|
)
|
|
|
|
)
|
2023-08-01 19:34:11 +02:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe 'jpeg' do
|
2023-08-07 17:58:12 +02:00
|
|
|
let(:media) { Fabricate(:media_attachment, file: attachment_fixture('600x400.jpeg')) }
|
2023-08-01 19:34:11 +02:00
|
|
|
|
|
|
|
it_behaves_like 'static 600x400 image', 'image/jpeg', '.jpeg'
|
|
|
|
end
|
|
|
|
|
|
|
|
describe 'png' do
|
2023-08-07 17:58:12 +02:00
|
|
|
let(:media) { Fabricate(:media_attachment, file: attachment_fixture('600x400.png')) }
|
2023-08-01 19:34:11 +02:00
|
|
|
|
|
|
|
it_behaves_like 'static 600x400 image', 'image/png', '.png'
|
|
|
|
end
|
|
|
|
|
2024-12-11 03:28:21 +01:00
|
|
|
describe 'gif' do
|
|
|
|
let(:media) { Fabricate(:media_attachment, file: attachment_fixture('600x400.gif')) }
|
2024-06-05 21:15:39 +02:00
|
|
|
|
2024-12-11 03:28:21 +01:00
|
|
|
it_behaves_like 'static 600x400 image', 'image/gif', '.gif'
|
2024-06-05 21:15:39 +02:00
|
|
|
end
|
|
|
|
|
2023-08-01 19:34:11 +02:00
|
|
|
describe 'webp' do
|
2023-08-07 17:58:12 +02:00
|
|
|
let(:media) { Fabricate(:media_attachment, file: attachment_fixture('600x400.webp')) }
|
2023-08-01 19:34:11 +02:00
|
|
|
|
|
|
|
it_behaves_like 'static 600x400 image', 'image/webp', '.webp'
|
|
|
|
end
|
|
|
|
|
|
|
|
describe 'avif' do
|
2023-08-07 17:58:12 +02:00
|
|
|
let(:media) { Fabricate(:media_attachment, file: attachment_fixture('600x400.avif')) }
|
2023-08-01 19:34:11 +02:00
|
|
|
|
|
|
|
it_behaves_like 'static 600x400 image', 'image/jpeg', '.jpeg'
|
|
|
|
end
|
|
|
|
|
|
|
|
describe 'heic' do
|
2023-08-07 17:58:12 +02:00
|
|
|
let(:media) { Fabricate(:media_attachment, file: attachment_fixture('600x400.heic')) }
|
2023-08-01 19:34:11 +02:00
|
|
|
|
|
|
|
it_behaves_like 'static 600x400 image', 'image/jpeg', '.jpeg'
|
|
|
|
end
|
|
|
|
|
2024-12-11 03:28:21 +01:00
|
|
|
describe 'monochrome jpg' do
|
|
|
|
let(:media) { Fabricate(:media_attachment, file: attachment_fixture('monochrome.png')) }
|
|
|
|
|
|
|
|
it_behaves_like 'static 600x400 image', 'image/png', '.png'
|
|
|
|
end
|
|
|
|
|
2023-08-01 19:34:11 +02:00
|
|
|
describe 'base64-encoded image' do
|
|
|
|
let(:base64_attachment) { "data:image/jpeg;base64,#{Base64.encode64(attachment_fixture('600x400.jpeg').read)}" }
|
2023-08-07 17:58:12 +02:00
|
|
|
let(:media) { Fabricate(:media_attachment, file: base64_attachment) }
|
2023-08-01 19:34:11 +02:00
|
|
|
|
|
|
|
it_behaves_like 'static 600x400 image', 'image/jpeg', '.jpeg'
|
|
|
|
end
|
|
|
|
|
|
|
|
describe 'animated gif' do
|
2024-12-11 03:28:21 +01:00
|
|
|
let(:media) { Fabricate(:media_attachment, file: attachment_fixture('600x400-animated.gif')) }
|
2016-09-09 20:04:34 +02:00
|
|
|
|
2024-12-11 03:28:21 +01:00
|
|
|
it_behaves_like 'animated 600x400 image'
|
2017-04-19 23:21:00 +02:00
|
|
|
end
|
|
|
|
|
2024-12-11 03:28:21 +01:00
|
|
|
describe 'animated png' do
|
|
|
|
let(:media) { Fabricate(:media_attachment, file: attachment_fixture('600x400-animated.png')) }
|
|
|
|
|
|
|
|
it_behaves_like 'animated 600x400 image'
|
2017-04-26 03:48:12 +02:00
|
|
|
end
|
|
|
|
|
2021-09-29 23:52:36 +02:00
|
|
|
describe 'ogg with cover art' do
|
2023-08-07 17:58:12 +02:00
|
|
|
let(:media) { Fabricate(:media_attachment, file: attachment_fixture('boop.ogg')) }
|
2024-07-18 17:23:46 +02:00
|
|
|
let(:expected_media_duration) { 0.235102 }
|
2021-09-29 23:52:36 +02:00
|
|
|
|
2024-07-18 17:23:46 +02:00
|
|
|
# The libvips and ImageMagick implementations produce different results
|
|
|
|
let(:expected_background_color) { Rails.configuration.x.use_vips ? '#268cd9' : '#3088d4' }
|
2024-06-05 21:15:39 +02:00
|
|
|
|
2024-07-18 17:23:46 +02:00
|
|
|
it 'sets correct file metadata' do
|
|
|
|
expect(media)
|
|
|
|
.to have_attributes(
|
|
|
|
type: eq('audio'),
|
|
|
|
thumbnail: be_present,
|
|
|
|
file_file_name: not_eq('boop.ogg')
|
|
|
|
)
|
2024-06-27 18:03:26 +02:00
|
|
|
|
2024-07-18 17:23:46 +02:00
|
|
|
expect(media_metadata)
|
|
|
|
.to include(
|
|
|
|
original: include(duration: be_within(0.05).of(expected_media_duration)),
|
|
|
|
colors: include(background: eq(expected_background_color))
|
|
|
|
)
|
2024-06-27 18:03:26 +02:00
|
|
|
end
|
2021-09-29 23:52:36 +02:00
|
|
|
end
|
|
|
|
|
2023-07-06 15:05:05 +02:00
|
|
|
describe 'mp3 with large cover art' do
|
2023-08-07 17:58:12 +02:00
|
|
|
let(:media) { Fabricate(:media_attachment, file: attachment_fixture('boop.mp3')) }
|
2024-07-18 17:23:46 +02:00
|
|
|
let(:expected_media_duration) { 0.235102 }
|
2023-07-06 15:05:05 +02:00
|
|
|
|
2024-07-18 17:23:46 +02:00
|
|
|
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))
|
|
|
|
)
|
2023-07-06 15:05:05 +02:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2024-09-05 17:36:05 +02:00
|
|
|
it { is_expected.to validate_presence_of(:file) }
|
2020-01-23 21:40:03 +01:00
|
|
|
|
2021-10-06 15:49:32 +02:00
|
|
|
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
|
2023-08-07 17:58:12 +02:00
|
|
|
expect { Fabricate(:media_attachment, file: attachment_fixture('attachment.webm')) }.to raise_error(ActiveRecord::RecordInvalid)
|
2021-10-06 15:49:32 +02:00
|
|
|
end
|
|
|
|
|
|
|
|
it 'accepts video files that are small enough' do
|
|
|
|
stub_const 'MediaAttachment::IMAGE_LIMIT', 1.kilobyte
|
|
|
|
stub_const 'MediaAttachment::VIDEO_LIMIT', 100.megabytes
|
2023-08-07 17:58:12 +02:00
|
|
|
media = Fabricate(:media_attachment, file: attachment_fixture('attachment.webm'))
|
2021-10-06 15:49:32 +02:00
|
|
|
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
|
2023-08-07 17:58:12 +02:00
|
|
|
expect { Fabricate(:media_attachment, file: attachment_fixture('attachment.jpg')) }.to raise_error(ActiveRecord::RecordInvalid)
|
2021-10-06 15:49:32 +02:00
|
|
|
end
|
|
|
|
|
|
|
|
it 'accepts image files that are small enough' do
|
|
|
|
stub_const 'MediaAttachment::IMAGE_LIMIT', 100.megabytes
|
|
|
|
stub_const 'MediaAttachment::VIDEO_LIMIT', 1.kilobyte
|
2023-08-07 17:58:12 +02:00
|
|
|
media = Fabricate(:media_attachment, file: attachment_fixture('attachment.jpg'))
|
2021-10-06 15:49:32 +02:00
|
|
|
expect(media.valid?).to be true
|
|
|
|
end
|
|
|
|
end
|
2024-07-18 17:23:46 +02:00
|
|
|
|
2024-08-09 14:48:34 +02:00
|
|
|
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
|
|
|
|
|
2024-07-18 17:23:46 +02:00
|
|
|
private
|
|
|
|
|
|
|
|
def media_metadata
|
|
|
|
media.file.meta.deep_symbolize_keys
|
|
|
|
end
|
2016-09-05 17:46:36 +02:00
|
|
|
end
|