67 lines
		
	
	
	
		
			2.5 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
			
		
		
	
	
			67 lines
		
	
	
	
		
			2.5 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
| # frozen_string_literal: true
 | |
| 
 | |
| class VideoMetadataExtractor
 | |
|   attr_reader :duration, :bitrate, :video_codec, :audio_codec,
 | |
|               :colorspace, :width, :height, :frame_rate, :r_frame_rate
 | |
| 
 | |
|   def initialize(path)
 | |
|     @path     = path
 | |
|     @metadata = Oj.load(ffmpeg_command_output, mode: :strict, symbol_keys: true)
 | |
| 
 | |
|     parse_metadata
 | |
|   rescue Terrapin::ExitStatusError, Oj::ParseError
 | |
|     @invalid = true
 | |
|   rescue Terrapin::CommandNotFoundError
 | |
|     raise Paperclip::Errors::CommandNotFoundError, 'Could not run the `ffprobe` command. Please install ffmpeg.' # rubocop:disable I18n/RailsI18n/DecorateString -- This error is not user-facing
 | |
|   end
 | |
| 
 | |
|   def valid?
 | |
|     !@invalid
 | |
|   end
 | |
| 
 | |
|   private
 | |
| 
 | |
|   def ffmpeg_command_output
 | |
|     command = Terrapin::CommandLine.new(Rails.configuration.x.ffprobe_binary, '-i :path -print_format :format -show_format -show_streams -show_error -loglevel :loglevel')
 | |
|     command.run(path: @path, format: 'json', loglevel: 'fatal')
 | |
|   end
 | |
| 
 | |
|   def parse_metadata
 | |
|     if @metadata.key?(:format)
 | |
|       @duration = @metadata[:format][:duration].to_f
 | |
|       @bitrate  = @metadata[:format][:bit_rate].to_i
 | |
|     end
 | |
| 
 | |
|     if @metadata.key?(:streams)
 | |
|       video_streams = @metadata[:streams].select { |stream| stream[:codec_type] == 'video' }
 | |
|       audio_streams = @metadata[:streams].select { |stream| stream[:codec_type] == 'audio' }
 | |
| 
 | |
|       if (video_stream = video_streams.first)
 | |
|         @video_codec = video_stream[:codec_name]
 | |
|         @colorspace  = video_stream[:pix_fmt]
 | |
|         @width       = video_stream[:width]
 | |
|         @height      = video_stream[:height]
 | |
|         @frame_rate  = parse_framerate(video_stream[:avg_frame_rate])
 | |
|         @r_frame_rate = parse_framerate(video_stream[:r_frame_rate])
 | |
|         # For some video streams the frame_rate reported by `ffprobe` will be 0/0, but for these streams we
 | |
|         # should use `r_frame_rate` instead. Video screencast generated by Gnome Screencast have this issue.
 | |
|         @frame_rate ||= @r_frame_rate
 | |
|         # If the video has not been re-encoded by ffmpeg, it may contain rotation information,
 | |
|         # and we need to simulate applying it to the dimensions
 | |
|         @width, @height = @height, @width if video_stream[:side_data_list]&.any? { |x| x[:rotation]&.abs == 90 }
 | |
|       end
 | |
| 
 | |
|       if (audio_stream = audio_streams.first)
 | |
|         @audio_codec = audio_stream[:codec_name]
 | |
|       end
 | |
|     end
 | |
| 
 | |
|     @invalid = true if @metadata.key?(:error)
 | |
|   end
 | |
| 
 | |
|   def parse_framerate(raw)
 | |
|     Rational(raw)
 | |
|   rescue ZeroDivisionError
 | |
|     nil
 | |
|   end
 | |
| end
 |