2017-03-04 22:17:10 +01:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2019-10-03 01:09:12 +02:00
|
|
|
class GifReader
|
|
|
|
attr_reader :animated
|
|
|
|
|
|
|
|
EXTENSION_LABELS = [0xf9, 0x01, 0xff].freeze
|
|
|
|
GIF_HEADERS = %w(GIF87a GIF89a).freeze
|
|
|
|
|
2020-05-15 11:38:12 +02:00
|
|
|
class GifReaderException < StandardError; end
|
2019-10-03 01:09:12 +02:00
|
|
|
|
|
|
|
class UnknownImageType < GifReaderException; end
|
|
|
|
|
|
|
|
class CannotParseImage < GifReaderException; end
|
|
|
|
|
|
|
|
def self.animated?(path)
|
|
|
|
new(path).animated
|
|
|
|
rescue GifReaderException
|
|
|
|
false
|
|
|
|
end
|
|
|
|
|
|
|
|
def initialize(path, max_frames = 2)
|
|
|
|
@path = path
|
|
|
|
@nb_frames = 0
|
|
|
|
|
|
|
|
File.open(path, 'rb') do |s|
|
|
|
|
raise UnknownImageType unless GIF_HEADERS.include?(s.read(6))
|
|
|
|
|
|
|
|
# Skip to "packed byte"
|
|
|
|
s.seek(4, IO::SEEK_CUR)
|
|
|
|
|
|
|
|
# "Packed byte" gives us the size of the GIF color table
|
|
|
|
packed_byte, = s.read(1).unpack('C')
|
|
|
|
|
|
|
|
# Skip background color and aspect ratio
|
|
|
|
s.seek(2, IO::SEEK_CUR)
|
|
|
|
|
|
|
|
if packed_byte & 0x80 != 0
|
|
|
|
# GIF uses a global color table, skip it
|
|
|
|
s.seek(3 * (1 << ((packed_byte & 0x07) + 1)), IO::SEEK_CUR)
|
|
|
|
end
|
|
|
|
|
|
|
|
# Now read data
|
|
|
|
while @nb_frames < max_frames
|
|
|
|
separator = s.read(1)
|
|
|
|
|
|
|
|
case separator
|
|
|
|
when ',' # Image block
|
|
|
|
@nb_frames += 1
|
|
|
|
|
|
|
|
# Skip to "packed byte"
|
|
|
|
s.seek(8, IO::SEEK_CUR)
|
|
|
|
packed_byte, = s.read(1).unpack('C')
|
|
|
|
|
|
|
|
if packed_byte & 0x80 != 0
|
|
|
|
# Image uses a local color table, skip it
|
|
|
|
s.seek(3 * (1 << ((packed_byte & 0x07) + 1)), IO::SEEK_CUR)
|
|
|
|
end
|
|
|
|
|
|
|
|
# Skip lzw min code size
|
2023-02-20 06:51:43 +01:00
|
|
|
raise InvalidValue unless s.read(1).unpack1('C') >= 2
|
2019-10-03 01:09:12 +02:00
|
|
|
|
|
|
|
# Skip image data sub-blocks
|
|
|
|
skip_sub_blocks!(s)
|
|
|
|
when '!' # Extension block
|
|
|
|
skip_extension_block!(s)
|
|
|
|
when ';' # Trailer
|
|
|
|
break
|
|
|
|
else
|
|
|
|
raise CannotParseImage
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
@animated = @nb_frames > 1
|
|
|
|
end
|
|
|
|
|
|
|
|
private
|
|
|
|
|
|
|
|
def skip_extension_block!(file)
|
2023-02-20 06:51:43 +01:00
|
|
|
if EXTENSION_LABELS.include?(file.read(1).unpack1('C'))
|
2019-10-03 01:09:12 +02:00
|
|
|
block_size, = file.read(1).unpack('C')
|
|
|
|
file.seek(block_size, IO::SEEK_CUR)
|
|
|
|
end
|
|
|
|
|
|
|
|
# Read until extension block end marker
|
|
|
|
skip_sub_blocks!(file)
|
|
|
|
end
|
|
|
|
|
|
|
|
# Skip sub-blocks up until block end marker
|
|
|
|
def skip_sub_blocks!(file)
|
|
|
|
loop do
|
|
|
|
size, = file.read(1).unpack('C')
|
|
|
|
|
|
|
|
break if size.zero?
|
|
|
|
|
|
|
|
file.seek(size, IO::SEEK_CUR)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-03-04 22:17:10 +01:00
|
|
|
module Paperclip
|
|
|
|
# This transcoder is only to be used for the MediaAttachment model
|
2021-05-05 19:44:01 +02:00
|
|
|
# to convert animated GIFs to videos
|
|
|
|
|
2017-03-04 22:17:10 +01:00
|
|
|
class GifTranscoder < Paperclip::Processor
|
|
|
|
def make
|
2018-07-14 03:56:52 +02:00
|
|
|
return File.open(@file.path) unless needs_convert?
|
2017-03-04 22:17:10 +01:00
|
|
|
|
|
|
|
final_file = Paperclip::Transcoder.make(file, options, attachment)
|
|
|
|
|
2021-05-11 19:15:11 +02:00
|
|
|
if options[:style] == :original
|
|
|
|
attachment.instance.file_file_name = File.basename(attachment.instance.file_file_name, '.*') + '.mp4'
|
|
|
|
attachment.instance.file_content_type = 'video/mp4'
|
|
|
|
attachment.instance.type = MediaAttachment.types[:gifv]
|
|
|
|
end
|
2017-03-04 22:17:10 +01:00
|
|
|
|
|
|
|
final_file
|
|
|
|
end
|
2018-07-14 03:56:52 +02:00
|
|
|
|
|
|
|
private
|
|
|
|
|
|
|
|
def needs_convert?
|
2021-05-11 19:15:11 +02:00
|
|
|
GifReader.animated?(file.path)
|
2018-07-14 03:56:52 +02:00
|
|
|
end
|
2017-03-04 22:17:10 +01:00
|
|
|
end
|
|
|
|
end
|