1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
|
# frozen_string_literal: true
require "active_support/core_ext/hash/compact"
module ActiveStorage
# Extracts the following from a video blob:
#
# * Width (pixels)
# * Height (pixels)
# * Duration (seconds)
# * Angle (degrees)
# * Aspect ratio
#
# Example:
#
# ActiveStorage::VideoAnalyzer.new(blob).metadata
# # => { width: 640, height: 480, duration: 5.0, angle: 0, 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, aspect_ratio: aspect_ratio }.compact
end
private
def width
rotated? ? raw_height : raw_width
end
def height
rotated? ? raw_width : raw_height
end
def raw_width
Integer(video_stream["width"]) if video_stream["width"]
end
def raw_height
Integer(video_stream["height"]) if video_stream["height"]
end
def duration
Float(video_stream["duration"]) if video_stream["duration"]
end
def angle
Integer(tags["rotate"]) if tags["rotate"]
end
def aspect_ratio
if descriptor = video_stream["display_aspect_ratio"]
descriptor.split(":", 2).collect(&:to_i)
end
end
def rotated?
angle == 90 || angle == 270
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
|