Skip to content
Permalink
main
Switch branches/tags
Go to file
 
 
Cannot retrieve contributors at this time
842 lines (666 sloc) 18 KB
package SReview::Video;
our $VERSION;
use SReview;
=head1 NAME
SReview::Video - SReview internal representation of an asset
=head1 SYNOPSIS
use SReview::Video;
use SReview::Video::ProfileFactory;
use SReview::Videopipe;
# convert any input file to VP9 at recommended settings for vertical resolution and frame rate
my $input = SReview::Video->new(url => $input_filename);
my $profile = SReview::Video::ProfileFactory->new("vp9", $input);
my $output = SReview::Video->new(url => $output_filename, reference => $profile);
SReview::Videopipe->new(inputs => [$input], output => $output)->run();
# do that again; but this time, force vorbis audio:
$output = SReview::Video->new(url => $other_filename, reference => $profile);
$output->audio_codec("libvorbis");
SReview::Videopipe->new(inputs => [$input], output => $output)->run();
=head1 DESCRIPTION
The SReview::Video package is used to represent media assets inside
SReview. It is a C<Moose>-based base class for much of the other Video*
classes in SReview.
There is one required attribute, C<url>, which represents the filename
of the video (however, for SReview::Video::NGinX, it should be an HTTP
URL instead).
If the C<url> attribute points to an existing file and an attempt is
made to read any of the codec, framerate, bit rate, or similar
attributes (without explicitly writing to them first), then
C<SReview::Video> will call C<ffprobe> on the file in question, and use
that to populate the requested attributes. If it does not, or C<ffprobe>
is incapable of detecting the requested attribute (which may be the case
for things like audio or video bitrate), then the attribute in question
will resolve to C<undef>.
If the C<url> attribute does not point to an existing file and an
attempt is made to read any of the codec, framerate, bit rate, or
similar attributes (without explicitly writing to them first), then they
will resolve to C<undef>. However, if the C<reference> attribute is
populated with another C<SReview::Video> object, then reading any of the
codec, framerate, bit rate, or similar attributes (without explicitly
writing to them first) will resolve to the value of the requested
attribute that is set or detected on the C<reference> object.
The return value of C<SReview::Video::ProfileFactory-E<gt>create()> is
also an SReview::Video object, but with different implementations of
some of the probing methods; this allows it to choose the correct values
for things like bitrate and encoder speed based on properties set in the
input object provided to the
C<SReview::Video::ProfileFactory-E<gt>create()> method.
For more information on how to use the files referred to in the
C<SReview::Video> object in an ffmpeg command line, please see
C<SReview::Videopipe>.
=head1 ATTRIBUTES
The following attributes are supported by SReview::Video. All attributes
will be probed from ffprobe output unless noted otherwise.
=cut
use Mojo::JSON qw(decode_json);
use SReview::CodecMap qw/detect_to_write/;
use Moose;
=head2 url
The filename of the asset this object should deal with. Required at
construction time. Will not be probed.
=cut
has 'url' => (
is => 'rw',
required => 1,
);
=head2 mtime
The mtime of the file backing this asset. Only defined if the file
exists at the time the attribute is first read, and is not updated later
on.
=cut
has 'mtime' => (
is => 'ro',
lazy => 1,
builder => '_probe_mtime',
);
sub _probe_mtime {
my $self = shift;
if($self->has_reference) {
return $self->reference->mtime;
}
my @statdata = stat($self->url);
if(scalar(@statdata) == 0) {
return undef;
}
return $statdata[9];
}
=head2 duration
The duration of this asset.
=cut
has 'duration' => (
is => 'rw',
builder => '_probe_duration',
lazy => 1,
);
sub _probe_duration {
my $self = shift;
if($self->has_reference) {
return $self->reference->duration;
}
return $self->_get_probedata->{format}{duration};
}
=head2 duration_style
The time unit is used for the C<duration> attribute. One of 'seconds'
(default) or 'frames'. Will not be probed.
=cut
has 'duration_style' => (
is => 'rw',
default => 'seconds',
);
=head2 video_codec
The codec in use for the video stream. Note that C<ffprobe> will
sometimes use a string (e.g., "vp8") that is not the best choice when
instructing C<ffmpeg> to transcode video to the said codec (for vp8, the
use of "libvpx" is recommended). C<SReview::CodecMap> is used to map
detected codecs to output codecs and resolve this issue.
=cut
has 'video_codec' => (
is => 'rw',
builder => '_probe_videocodec',
lazy => 1,
);
sub _probe_videocodec {
my $self = shift;
if($self->has_reference) {
return $self->reference->video_codec;
}
return $self->_get_videodata->{codec_name};
}
=head2 audio_codec
The codec in use for the audio stream. Note that C<ffprobe> will
sometimes use a string (e.g., "vorbis") that is not the best choice when
instructing C<ffmpeg> to transcode audio to the said codec (for vorbis,
the use of "libvorbis" is recommended). C<SReview::CodecMap> is used to
map detected codecs to output codecs and resolve this issue.
=cut
has 'audio_codec' => (
is => 'rw',
builder => '_probe_audiocodec',
lazy => 1,
);
sub _probe_audiocodec {
my $self = shift;
if($self->has_reference) {
return $self->reference->audio_codec;
}
return $self->_get_audiodata->{codec_name};
}
=head2 video_size
A string representing the resolution of the video in C<WxH> format,
where W is the width and H is the height.
This attribute is special in that in contrast to all the other
attributes, it is not provided directly by C<ffprobe>; instead, when
this parameter is read, the C<video_width> and C<video_height>
attributes are read and combined.
That does mean that you should not read this attribute, and based on
that possibly set the height and/or width attributes of a video (or vice
versa). Instead, you should read I<either> the C<video_width> and
C<video_height> attribute, I<or> this one.
Failure to follow this rule will result in undefined behaviour.
=cut
has 'video_size' => (
is => 'rw',
builder => '_probe_videosize',
lazy => 1,
predicate => 'has_video_size',
);
sub _probe_videosize {
my $self = shift;
if($self->has_reference) {
return $self->reference->video_size;
}
my $width = $self->video_width;
my $height = $self->video_height;
return undef unless defined($width) && defined($height);
return $self->video_width . "x" . $self->video_height;
}
=head2 video_width
The width of the video, in pixels.
=cut
has 'video_width' => (
is => 'rw',
builder => '_probe_width',
lazy => 1,
);
sub _probe_width {
my $self = shift;
if($self->has_reference) {
return $self->reference->video_width;
}
if($self->has_video_size) {
return (split /x/, $self->video_size)[0];
} else {
return $self->_get_videodata->{width};
}
}
=head2 video_height
The height of the video, in pixels.
=cut
has 'video_height' => (
is => 'rw',
builder => '_probe_height',
lazy => 1,
);
sub _probe_height {
my $self = shift;
if($self->has_reference) {
return $self->reference->video_height;
}
if($self->has_video_size) {
return (split /x/, $self->video_size)[1];
} else {
return $self->_get_videodata->{height};
}
}
=head2 video_bitrate
The bit rate of this video, in bits per second.
Note that not all container formats support probing the bitrate of the
encoded video or audio; when read on input objects with those that do
not, this will resolve to C<undef>.
=cut
has 'video_bitrate' => (
is => 'rw',
builder => '_probe_videobitrate',
lazy => 1,
);
sub _probe_videobitrate {
my $self = shift;
if($self->has_reference) {
return $self->reference->video_bitrate;
}
return $self->_get_videodata->{bit_rate};
}
=head2 video_minrate
The minimum bit rate for this video, in bits per second.
Defaults to 0.5 * video_bitrate
=cut
has 'video_minrate' => (
is => 'rw',
builder => '_probe_videominrate',
lazy => 1,
);
sub _probe_videominrate {
my $self = shift;
my $rate;
if($self->has_reference) {
$rate = $self->reference->video_minrate;
if(defined($rate)) {
return $rate;
}
}
$rate = $self->video_bitrate;
if(defined($rate)) {
return $rate * 0.5;
}
return undef;
}
=head2 video_maxrate
The maximum bit rate for this video, in bits per second.
Defaults to 1.45 * video_bitrate
=cut
has 'video_maxrate' => (
is => 'rw',
builder => '_probe_videomaxrate',
lazy => 1,
);
sub _probe_videomaxrate {
my $self = shift;
my $rate;
if($self->has_reference) {
$rate = $self->reference->video_maxrate;
if(defined($rate)) {
return $rate;
}
}
$rate = $self->video_bitrate;
if(defined($rate)) {
return $rate * 1.45;
}
return undef;
}
=head2 aspect_ratio
The Display Aspect Ratio of a video. Note that with non-square pixels,
this is not guaranteed to be what one would expect when reading the
C<video_size> attribute.
=cut
has 'aspect_ratio' => (
is => 'rw',
builder => '_probe_aspect_ratio',
lazy => 1,
);
sub _probe_aspect_ratio {
my $self = shift;
if($self->has_reference) {
return $self->reference->aspect_ratio;
}
return $self->_get_videodata->{display_aspect_ratio};
}
=head2 audio_bitrate
The bit rate of the audio stream on this video, in bits per second
=cut
has 'audio_bitrate' => (
is => 'rw',
builder => '_probe_audiobitrate',
lazy => 1,
);
sub _probe_audiobitrate {
my $self = shift;
if($self->has_reference) {
return $self->reference->audio_bitrate;
}
return $self->_get_audiodata->{bit_rate};
}
=head2 audio_samplerate
The sample rate of the audio, in samples per second
=cut
has 'audio_samplerate' => (
is => 'rw',
builder => '_probe_audiorate',
lazy => 1,
);
sub _probe_audiorate {
my $self = shift;
if($self->has_reference) {
return $self->reference->audio_samplerate;
}
return $self->_get_audiodata->{sample_rate};
}
=head2 video_framerate
The frame rate of the video, as a fraction.
Note that in the weird US frame rate, this could be 30000/1001.
=cut
has 'video_framerate' => (
is => 'rw',
builder => '_probe_framerate',
lazy => 1,
);
sub _probe_framerate {
my $self = shift;
if($self->has_reference) {
return $self->reference->video_framerate;
}
my $framerate = $self->_get_videodata->{r_frame_rate};
return $framerate;
}
=head2 fragment_start
If set, this instructs SReview on read to only read a particular part of the
video from this file. Should be specified in seconds; will not be probed.
=cut
has 'fragment_start' => (
is => 'rw',
predicate => 'has_fragment_start',
);
=head2 quality
The quality used for the video encoding, i.e., the value passed to the C<-crf>
parameter. Mostly for use by a profile. Will not be probed.
=cut
has 'quality' => (
is => 'rw',
builder => '_probe_quality',
lazy => 1,
);
sub _probe_quality {
my $self = shift;
if($self->has_reference) {
return $self->reference->quality;
}
return undef;
}
=head2 metadata
Can be used to set video metadata (as per C<ffmpeg>'s C<-metadata>
parameter). Functions C<add_metadata> and C<drop_metadata> can be used
to add or remove individual metedata values. Will not be probed.
=cut
has 'metadata' => (
traits => ['Hash'],
isa => 'HashRef[Str]',
is => 'ro',
handles => {
add_metadata => 'set',
drop_metadata => 'delete',
},
predicate => 'has_metadata',
);
=head2 reference
If set to any C<SReview::Video> object, then when any value is being
probed, rather than trying to run C<ffprobe> on the file pointed to by
our C<url> attribute, we will use the value reported by the referenced
object.
Can be used in "build a file almost like this one, but with these things
different" kind of scenarios.
Will not be probed (obviously).
=cut
has 'reference' => (
isa => 'SReview::Video',
is => 'ro',
predicate => 'has_reference',
);
=head2 pix_fmt
The pixel format (e.g., C<yuv420p> or the likes) of the video.
=cut
has 'pix_fmt' => (
is => 'rw',
builder => '_probe_pix_fmt',
lazy => 1,
);
sub _probe_pix_fmt {
my $self = shift;
if($self->has_reference) {
return $self->reference->pix_fmt;
}
return $self->_get_videodata->{pix_fmt};
}
=head2 astream_id
Returns the numeric ID for the first audio stream in this file. Useful for the
implementation of stream mappings etc; see C<SReview::Map>
=cut
has 'astream_id' => (
is => 'rw',
builder => '_probe_astream_id',
lazy => 1,
);
sub _probe_astream_id {
my $self = shift;
return $self->_get_audiodata->{index};
}
=head2 blackspots
Returns an array of hashes. Each hash contains a member C<start>,
C<end>, and C<duration>, containing the start, end, and duration,
respectively, of locations in the video file that are (almost) entirely
black.
Could be used by a script for automatic review.
Note that the ffmpeg run required to detect blackness is CPU intensive
and may require a very long time to finish.
=cut
has blackspots => (
is => 'ro',
isa => 'ArrayRef[HashRef[Num]]',
builder => '_probe_blackspots',
lazy => 1,
);
sub _probe_blackspots {
my $self = shift;
my $blacks = [];
pipe R, W;
if(fork == 0) {
open STDERR, ">&W";
open STDOUT, ">&W";
my @cmd = ("ffmpeg", "-threads", "1", "-nostats", "-i", $self->url, "-vf", "blackdetect=d=0:pix_th=.01", "-f", "null", "/dev/null");
exec @cmd;
die "exec failed";
}
close W;
while(<R>) {
if(/blackdetect.*black_start:(?<start>[\d\.]+)\sblack_end:(?<end>[\d\.]+)\sblack_duration:(?<duration>[\d\.]+)/) {
push @$blacks, { %+ };
}
}
close(R);
return $blacks;
}
=head2 astream_ids
Returns an array with the IDs for the audio streams in this file.
=head2 astream_count
Returns the number of audio streams in this file.
=cut
has 'astream_ids' => (
is => 'rw',
traits => ['Array'],
isa => 'ArrayRef[Int]',
builder => '_probe_astream_ids',
lazy => 1,
handles => {
astream_count => "count",
},
);
sub _probe_astream_ids {
my $self = shift;
my @rv;
foreach my $stream(@{$self->_get_probedata->{streams}}) {
if($stream->{codec_type} eq "audio") {
push @rv, $stream->{index};
}
}
return \@rv;
}
=head2 vstream_id
Returns the numeric ID for the first video stream in this file. Useful
for the implementation of stream mappings etc; see C<SReview::Map>
=cut
has 'vstream_id' => (
is => 'rw',
builder => '_probe_vstream_id',
lazy => 1,
);
sub _probe_vstream_id {
my $self = shift;
return $self->_get_videodata->{index};
}
=head2 extra_params
Add extra parameters. This should be used sparingly, rather add some
abstraction.
=cut
has 'extra_params' => (
traits => ['Hash'],
isa => 'HashRef[Str]',
is => 'ro',
handles => {
add_param => 'set',
drop_param => 'delete',
},
builder => "_probe_extra_params",
lazy => 1,
);
sub _probe_extra_params {
my $self = shift;
if($self->has_reference) {
return $self->reference->extra_params;
}
return {};
}
# Only to be used by the Videopipe class when doing multiple passes
has 'pass' => (
is => 'rw',
predicate => 'has_pass',
clearer => 'clear_pass',
);
## The below exist to help autodetect sizes, and are not meant for the end user
has 'videodata' => (
is => 'bare',
reader => '_get_videodata',
builder => '_probe_videodata',
lazy => 1,
);
has 'audiodata' => (
is => 'bare',
reader => '_get_audiodata',
builder => '_probe_audiodata',
lazy => 1,
);
has 'probedata' => (
is => 'bare',
reader => '_get_probedata',
builder => '_probe',
clearer => 'clear_probedata',
lazy => 1,
);
sub readopts {
my $self = shift;
my @opts = ();
push @opts, ("-i", $self->url);
return @opts;
}
sub writeopts {
my $self = shift;
my $pipe = shift;
my @opts = ();
if(!$pipe->vcopy && !$pipe->vskip) {
push @opts, ('-threads', '1');
if(defined($self->video_codec)) {
push @opts, ('-c:v', detect_to_write($self->video_codec));
}
if(defined($self->video_bitrate)) {
push @opts, ('-b:v', $self->video_bitrate . "k", '-minrate', $self->video_minrate . "k", '-maxrate', $self->video_maxrate . "k");
}
if(defined($self->video_framerate)) {
push @opts, ('-r:v', $self->video_framerate);
}
if(defined($self->quality)) {
push @opts, ('-crf', $self->quality);
}
if(defined($self->speed)) {
push @opts, ('-speed', $self->speed);
}
if($self->has_pass) {
push @opts, ('-pass', $self->pass, '-passlogfile', $self->url . '-multipass');
}
}
if(!$pipe->acopy && !$pipe->askip) {
if(defined($self->audio_codec)) {
push @opts, ('-c:a', detect_to_write($self->audio_codec));
}
if(defined($self->audio_bitrate)) {
push @opts, ('-b:a', $self->audio_bitrate);
}
if(defined($self->audio_samplerate)) {
push @opts, ('-ar', $self->audio_samplerate);
}
}
if($self->has_fragment_start) {
push @opts, ('-ss', $self->fragment_start);
}
if(defined($self->duration)) {
if($self->duration_style eq 'seconds') {
push @opts, ('-t', $self->duration);
} else {
push @opts, ('-frames:v', $self->duration);
}
}
if(defined($self->pix_fmt)) {
push @opts, ('-pix_fmt', $self->pix_fmt);
}
if($self->has_metadata) {
foreach my $meta(keys %{$self->metadata}) {
push @opts, ('-metadata', $meta . '=' . $self->metadata->{$meta});
}
}
if(!defined($self->duration) && $#{$pipe->inputs}>0) {
push @opts, '-shortest';
}
if(scalar(keys(%{$self->extra_params}))>0) {
foreach my $param(keys %{$self->extra_params}) {
push @opts, ("-$param", $self->extra_params->{$param});
}
}
if(exists($ENV{SREVIEW_NONSTRICT})) {
push @opts, ("-strict", "-2");
}
push @opts, $self->url;
return @opts;
}
sub _probe {
my $self = shift;
if($self->has_reference) {
return $self->reference->_get_probedata;
}
open JSON, "-|:encoding(UTF-8)", "ffprobe", "-loglevel", "quiet", "-print_format", "json", "-show_format", "-show_streams", $self->url;
my $json = "";
while(<JSON>) {
$json .= $_;
}
close JSON;
return decode_json($json);
}
sub _probe_audiodata {
my $self = shift;
if(!exists($self->_get_probedata->{streams})) {
return {};
}
foreach my $stream(@{$self->_get_probedata->{streams}}) {
if($stream->{codec_type} eq "audio") {
return $stream;
}
}
return {};
}
sub _probe_videodata {
my $self = shift;
if(!exists($self->_get_probedata->{streams})) {
return {};
}
foreach my $stream(@{$self->_get_probedata->{streams}}) {
if($stream->{codec_type} eq "video") {
return $stream;
}
}
return {};
}
sub speed {
my $self = shift;
if($self->has_reference) {
return $self->reference->speed;
}
return 4;
}
no Moose;
1;