mirror of
https://github.com/rails/rails.git
synced 2022-11-09 12:12:34 -05:00
f70defa1ba
The code snippet within the usage documentation comment used the wrong object namespace for the ActiveStorage::Analyzer::VideoAnalyzer
118 lines
3 KiB
Ruby
118 lines
3 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
module ActiveStorage
|
|
# Extracts the following from a video blob:
|
|
#
|
|
# * Width (pixels)
|
|
# * Height (pixels)
|
|
# * Duration (seconds)
|
|
# * Angle (degrees)
|
|
# * Display aspect ratio
|
|
#
|
|
# Example:
|
|
#
|
|
# ActiveStorage::Analyzer::VideoAnalyzer.new(blob).metadata
|
|
# # => { width: 640.0, height: 480.0, duration: 5.0, angle: 0, display_aspect_ratio: [4, 3] }
|
|
#
|
|
# When a video's angle is 90 or 270 degrees, its width and height are automatically swapped for convenience.
|
|
#
|
|
# This analyzer requires the {FFmpeg}[https://www.ffmpeg.org] system library, which is not provided by Rails.
|
|
class Analyzer::VideoAnalyzer < Analyzer
|
|
def self.accept?(blob)
|
|
blob.video?
|
|
end
|
|
|
|
def metadata
|
|
{ width: width, height: height, duration: duration, angle: angle, display_aspect_ratio: display_aspect_ratio }.compact
|
|
end
|
|
|
|
private
|
|
def width
|
|
if rotated?
|
|
computed_height || encoded_height
|
|
else
|
|
encoded_width
|
|
end
|
|
end
|
|
|
|
def height
|
|
if rotated?
|
|
encoded_width
|
|
else
|
|
computed_height || encoded_height
|
|
end
|
|
end
|
|
|
|
def duration
|
|
Float(video_stream["duration"]) if video_stream["duration"]
|
|
end
|
|
|
|
def angle
|
|
Integer(tags["rotate"]) if tags["rotate"]
|
|
end
|
|
|
|
def display_aspect_ratio
|
|
if descriptor = video_stream["display_aspect_ratio"]
|
|
if terms = descriptor.split(":", 2)
|
|
numerator = Integer(terms[0])
|
|
denominator = Integer(terms[1])
|
|
|
|
[numerator, denominator] unless numerator == 0
|
|
end
|
|
end
|
|
end
|
|
|
|
|
|
def rotated?
|
|
angle == 90 || angle == 270
|
|
end
|
|
|
|
def computed_height
|
|
if encoded_width && display_height_scale
|
|
encoded_width * display_height_scale
|
|
end
|
|
end
|
|
|
|
def encoded_width
|
|
@encoded_width ||= Float(video_stream["width"]) if video_stream["width"]
|
|
end
|
|
|
|
def encoded_height
|
|
@encoded_height ||= Float(video_stream["height"]) if video_stream["height"]
|
|
end
|
|
|
|
def display_height_scale
|
|
@display_height_scale ||= Float(display_aspect_ratio.last) / display_aspect_ratio.first if display_aspect_ratio
|
|
end
|
|
|
|
|
|
def tags
|
|
@tags ||= video_stream["tags"] || {}
|
|
end
|
|
|
|
def video_stream
|
|
@video_stream ||= streams.detect { |stream| stream["codec_type"] == "video" } || {}
|
|
end
|
|
|
|
def streams
|
|
probe["streams"] || []
|
|
end
|
|
|
|
def probe
|
|
download_blob_to_tempfile { |file| probe_from(file) }
|
|
end
|
|
|
|
def probe_from(file)
|
|
IO.popen([ ffprobe_path, "-print_format", "json", "-show_streams", "-v", "error", file.path ]) do |output|
|
|
JSON.parse(output.read)
|
|
end
|
|
rescue Errno::ENOENT
|
|
logger.info "Skipping video analysis because FFmpeg isn't installed"
|
|
{}
|
|
end
|
|
|
|
def ffprobe_path
|
|
ActiveStorage.paths[:ffprobe] || "ffprobe"
|
|
end
|
|
end
|
|
end
|