2023-02-22 01:55:31 +01:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2016-09-05 17:46:36 +02:00
|
|
|
require 'rails_helper'
|
|
|
|
|
2023-05-04 05:49:53 +02:00
|
|
|
RSpec.describe MediaAttachment do
|
2017-11-08 07:29:07 +01:00
|
|
|
describe 'local?' do
|
|
|
|
subject { media_attachment.local? }
|
|
|
|
|
2023-02-20 05:24:14 +01:00
|
|
|
let(:media_attachment) { Fabricate(:media_attachment, remote_url: remote_url) }
|
|
|
|
|
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-02-20 05:24:14 +01:00
|
|
|
let(:media_attachment) { Fabricate(:media_attachment, remote_url: remote_url, file: file) }
|
|
|
|
|
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
|
2021-10-13 15:27:19 +02:00
|
|
|
let(:media_attachment) { Fabricate(:media_attachment, shortcode: shortcode) }
|
|
|
|
let(:shortcode) { nil }
|
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' }
|
|
|
|
|
|
|
|
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 }
|
|
|
|
|
|
|
|
it 'returns string representation of id' do
|
|
|
|
expect(media_attachment.to_param).to eq media_attachment.id.to_s
|
|
|
|
end
|
2017-11-08 07:29:07 +01:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-04-19 23:21:00 +02:00
|
|
|
describe 'animated gif conversion' do
|
2023-06-06 13:58:33 +02:00
|
|
|
let(:media) { described_class.create(account: Fabricate(:account), file: attachment_fixture('avatar.gif')) }
|
2016-09-09 20:04:34 +02:00
|
|
|
|
2017-04-19 23:21:00 +02:00
|
|
|
it 'sets type to gifv' do
|
|
|
|
expect(media.type).to eq 'gifv'
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'converts original file to mp4' do
|
|
|
|
expect(media.file_content_type).to eq 'video/mp4'
|
|
|
|
end
|
2017-04-26 03:48:12 +02:00
|
|
|
|
|
|
|
it 'sets meta' do
|
2023-02-18 23:38:14 +01:00
|
|
|
expect(media.file.meta['original']['width']).to eq 128
|
|
|
|
expect(media.file.meta['original']['height']).to eq 128
|
2017-04-26 03:48:12 +02:00
|
|
|
end
|
2017-04-19 23:21:00 +02:00
|
|
|
end
|
|
|
|
|
|
|
|
describe 'non-animated gif non-conversion' do
|
2017-10-26 15:48:35 +02:00
|
|
|
fixtures = [
|
|
|
|
{ filename: 'attachment.gif', width: 600, height: 400, aspect: 1.5 },
|
|
|
|
{ filename: 'mini-static.gif', width: 32, height: 32, aspect: 1.0 },
|
|
|
|
]
|
2017-04-19 23:21:00 +02:00
|
|
|
|
2017-10-26 15:48:35 +02:00
|
|
|
fixtures.each do |fixture|
|
|
|
|
context fixture[:filename] do
|
2023-06-06 13:58:33 +02:00
|
|
|
let(:media) { described_class.create(account: Fabricate(:account), file: attachment_fixture(fixture[:filename])) }
|
2017-04-19 23:21:00 +02:00
|
|
|
|
2017-10-26 15:48:35 +02:00
|
|
|
it 'sets type to image' do
|
|
|
|
expect(media.type).to eq 'image'
|
|
|
|
end
|
2017-04-26 03:48:12 +02:00
|
|
|
|
2017-10-26 15:48:35 +02:00
|
|
|
it 'leaves original file as-is' do
|
|
|
|
expect(media.file_content_type).to eq 'image/gif'
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'sets meta' do
|
2023-02-18 23:38:14 +01:00
|
|
|
expect(media.file.meta['original']['width']).to eq fixture[:width]
|
|
|
|
expect(media.file.meta['original']['height']).to eq fixture[:height]
|
|
|
|
expect(media.file.meta['original']['aspect']).to eq fixture[:aspect]
|
2017-10-26 15:48:35 +02:00
|
|
|
end
|
|
|
|
end
|
2017-04-26 03:48:12 +02:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2021-09-29 23:52:36 +02:00
|
|
|
describe 'ogg with cover art' do
|
2023-06-06 13:58:33 +02:00
|
|
|
let(:media) { described_class.create(account: Fabricate(:account), file: attachment_fixture('boop.ogg')) }
|
2021-09-29 23:52:36 +02:00
|
|
|
|
|
|
|
it 'detects it as an audio file' do
|
|
|
|
expect(media.type).to eq 'audio'
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'sets meta for the duration' do
|
|
|
|
expect(media.file.meta['original']['duration']).to be_within(0.05).of(0.235102)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'extracts thumbnail' do
|
2023-02-20 06:14:50 +01:00
|
|
|
expect(media.thumbnail.present?).to be true
|
2021-09-29 23:52:36 +02:00
|
|
|
end
|
|
|
|
|
|
|
|
it 'extracts colors from thumbnail' do
|
|
|
|
expect(media.file.meta['colors']['background']).to eq '#3088d4'
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'gives the file a random name' do
|
|
|
|
expect(media.file_file_name).to_not eq 'boop.ogg'
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-04-26 03:48:12 +02:00
|
|
|
describe 'jpeg' do
|
2023-06-06 13:58:33 +02:00
|
|
|
let(:media) { described_class.create(account: Fabricate(:account), file: attachment_fixture('attachment.jpg')) }
|
2017-04-26 03:48:12 +02:00
|
|
|
|
|
|
|
it 'sets meta for different style' do
|
2023-02-18 23:38:14 +01:00
|
|
|
expect(media.file.meta['original']['width']).to eq 600
|
|
|
|
expect(media.file.meta['original']['height']).to eq 400
|
|
|
|
expect(media.file.meta['original']['aspect']).to eq 1.5
|
|
|
|
expect(media.file.meta['small']['width']).to eq 588
|
|
|
|
expect(media.file.meta['small']['height']).to eq 392
|
|
|
|
expect(media.file.meta['small']['aspect']).to eq 1.5
|
2017-04-26 03:48:12 +02:00
|
|
|
end
|
2020-01-04 01:54:07 +01:00
|
|
|
|
|
|
|
it 'gives the file a random name' do
|
|
|
|
expect(media.file_file_name).to_not eq 'attachment.jpg'
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe 'base64-encoded jpeg' do
|
|
|
|
let(:base64_attachment) { "data:image/jpeg;base64,#{Base64.encode64(attachment_fixture('attachment.jpg').read)}" }
|
2023-06-06 13:58:33 +02:00
|
|
|
let(:media) { described_class.create(account: Fabricate(:account), file: base64_attachment) }
|
2020-01-04 01:54:07 +01:00
|
|
|
|
|
|
|
it 'saves media attachment' do
|
|
|
|
expect(media.persisted?).to be true
|
|
|
|
expect(media.file).to_not be_nil
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'gives the file a file name' do
|
|
|
|
expect(media.file_file_name).to_not be_blank
|
|
|
|
end
|
2017-04-19 23:21:00 +02:00
|
|
|
end
|
2017-09-28 15:31:31 +02:00
|
|
|
|
2020-01-23 21:40:03 +01:00
|
|
|
it 'is invalid without file' do
|
2023-06-06 13:58:33 +02:00
|
|
|
media = described_class.new(account: Fabricate(:account))
|
2020-01-23 21:40:03 +01:00
|
|
|
expect(media.valid?).to be false
|
|
|
|
end
|
|
|
|
|
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-06-06 13:58:33 +02:00
|
|
|
expect { described_class.create!(account: Fabricate(:account), 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-06-06 13:58:33 +02:00
|
|
|
media = described_class.create!(account: Fabricate(:account), 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-06-06 13:58:33 +02:00
|
|
|
expect { described_class.create!(account: Fabricate(:account), 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-06-06 13:58:33 +02:00
|
|
|
media = described_class.create!(account: Fabricate(:account), file: attachment_fixture('attachment.jpg'))
|
2021-10-06 15:49:32 +02:00
|
|
|
expect(media.valid?).to be true
|
|
|
|
end
|
|
|
|
end
|
2016-09-05 17:46:36 +02:00
|
|
|
end
|