2005-12-16 10:05:29 +00:00
#!/usr/bin/perl -wT
#
# ==========================================================================
#
# ZoneMinder Video Creation Script, $Date$, $Revision$
2008-07-25 09:48:16 +00:00
# Copyright (C) 2001-2008 Philip Coombes
2005-12-16 10:05:29 +00:00
#
# This program is free software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License
# as published by the Free Software Foundation; either version 2
# of the License, or (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
#
# ==========================================================================
#
# This script is used to create MPEG videos of events for the web pages
# or as email attachments.
#
use strict ;
use bytes ;
# ==========================================================================
#
# You shouldn't need to change anything from here downwards
#
# ==========================================================================
2009-06-08 09:11:56 +00:00
@ EXTRA_PERL_LIB @
2005-12-16 10:05:29 +00:00
use ZoneMinder ;
use DBI ;
use Data::Dumper ;
2007-12-29 18:24:26 +00:00
use POSIX qw( strftime ) ;
2005-12-16 10:05:29 +00:00
use Getopt::Long qw( :config no_ignore_case ) ;
$| = 1 ;
$ ENV { PATH } = '/bin:/usr/bin' ;
$ ENV { SHELL } = '/bin/sh' if exists $ ENV { SHELL } ;
delete @ ENV { qw( IFS CDPATH ENV BASH_ENV ) } ;
2011-06-21 09:19:10 +00:00
logInit ( ) ;
2005-12-20 17:03:33 +00:00
2005-12-16 10:05:29 +00:00
my $ event_id ;
my $ format = 'mpg' ;
my $ rate = '' ;
my $ scale = '' ;
my $ fps = '' ;
my $ size = '' ;
my $ overwrite = 0 ;
2013-12-16 21:32:02 +00:00
my @ formats = split ( /\s+/ , $ Config { ZM_FFMPEG_FORMATS } ) ;
2005-12-16 10:05:29 +00:00
for ( my $ i = 0 ; $ i < @ formats ; $ i + + )
{
if ( $ i =~ /^(.+)\*$/ )
{
$ format = $ formats [ $ i ] = $ 1 ;
}
}
sub Usage
{
print ( "
Usage: zmvideo . pl - e <event_id> , - - event = <event_id> [ - - format <format> ] [ - - rate = <rate> ] [ - - scale = <scale> ] [ - - fps = <fps> ] [ - - size = <size> ] [ - - overwrite ]
Parameters are : -
- e <event_id> , - - event = <event_id> - What event to create the video for
- f <format> , - - format = <format> - What format to create the video in , default is mpg . For ffmpeg only .
- r <rate> , - - rate = <rate> - Relative rate , 1 = realtime , 2 = double speed , 0.5 = half speed etc
- s<scale> , - - s c a l e = <scale> - Scale , 1 = normal , 2 = double size , 0.5 = half size etc
- F <fps> , - - fps = <fps> - Absolute frame rate , in frames per second
- S <size> , - - size = <size> - Absolute video size , WxH or other specification supported by ffmpeg
- o , - - overwrite - Whether to overwrite an existing file , off by default .
" ) ;
exit ( - 1 ) ;
}
if ( ! GetOptions ( 'event=i' = > \ $ event_id , 'format|f=s' = > \ $ format , 'rate|r=f' = > \ $ rate , 'scale|s=f' = > \ $ scale , 'fps|F=f' = > \ $ fps , 'size|S=s' = > \ $ size , 'overwrite' = > \ $ overwrite ) )
{
Usage ( ) ;
}
if ( ! $ event_id || $ event_id < 0 )
{
print ( STDERR "Please give a valid event id\n" ) ;
Usage ( ) ;
}
2013-12-16 21:32:02 +00:00
if ( ! $ Config { ZM_OPT_FFMPEG } )
2005-12-16 10:05:29 +00:00
{
print ( STDERR "Mpeg encoding is not currently enabled\n" ) ;
exit ( - 1 ) ;
}
if ( ! $ rate && ! $ fps )
{
$ rate = 1 ;
}
if ( ! $ scale && ! $ size )
{
$ scale = 1 ;
}
if ( $ rate && ( $ rate < 0.25 || $ rate > 100 ) )
{
print ( STDERR "Rate is out of range, 0.25 >= rate <= 100\n" ) ;
Usage ( ) ;
}
if ( $ scale && ( $ scale < 0.25 || $ scale > 4 ) )
{
print ( STDERR "Scale is out of range, 0.25 >= scale <= 4\n" ) ;
Usage ( ) ;
}
if ( $ fps && ( $ fps > 30 ) )
{
print ( STDERR "FPS is out of range, <= 30\n" ) ;
Usage ( ) ;
}
my ( $ detaint_format ) = $ format =~ /^(\w+)$/ ;
my ( $ detaint_rate ) = $ rate =~ /^(-?\d+(?:\.\d+)?)$/ ;
my ( $ detaint_scale ) = $ scale =~ /^(-?\d+(?:\.\d+)?)$/ ;
my ( $ detaint_fps ) = $ fps =~ /^(-?\d+(?:\.\d+)?)$/ ;
my ( $ detaint_size ) = $ size =~ /^(\w+)$/ ;
$ format = $ detaint_format ;
$ rate = $ detaint_rate ;
$ scale = $ detaint_scale ;
$ fps = $ detaint_fps ;
$ size = $ detaint_size ;
2007-09-07 15:39:44 +00:00
my $ dbh = zmDbConnect ( ) ;
2005-12-16 10:05:29 +00:00
my @ filters ;
2007-09-04 16:08:55 +00:00
my $ sql = "select max(F.Delta)-min(F.Delta) as FullLength, E.*, unix_timestamp(E.StartTime) as Time, M.Name as MonitorName, M.Width as MonitorWidth, M.Height as MonitorHeight, M.Palette from Frames as F inner join Events as E on F.EventId = E.Id inner join Monitors as M on E.MonitorId = M.Id where EventId = '$event_id' group by F.EventId" ;
2006-04-23 19:51:28 +00:00
my $ sth = $ dbh - > prepare_cached ( $ sql ) or Fatal ( "Can't prepare '$sql': " . $ dbh - > errstr ( ) ) ;
my $ res = $ sth - > execute ( ) or Fatal ( "Can't execute: " . $ sth - > errstr ( ) ) ;
2005-12-16 10:05:29 +00:00
my $ event = $ sth - > fetchrow_hashref ( ) ;
$ sth - > finish ( ) ;
2008-02-13 22:41:17 +00:00
my $ event_path = getEventPath ( $ event ) ;
2006-04-23 19:51:28 +00:00
chdir ( $ event_path ) ;
2005-12-16 10:05:29 +00:00
( my $ video_name = $ event - > { Name } ) =~ s/\s/_/g ;
my @ file_parts ;
if ( $ rate )
{
my $ file_rate = $ rate ;
$ file_rate =~ s/\./_/ ;
$ file_rate =~ s/_00// ;
$ file_rate =~ s/(_\d+)0+$/$1/ ;
$ file_rate = 'r' . $ file_rate ;
push ( @ file_parts , $ file_rate ) ;
}
elsif ( $ fps )
{
my $ file_fps = $ fps ;
$ file_fps =~ s/\./_/ ;
$ file_fps =~ s/_00// ;
$ file_fps =~ s/(_\d+)0+$/$1/ ;
$ file_fps = 'R' . $ file_fps ;
push ( @ file_parts , $ file_fps ) ;
}
if ( $ scale )
{
my $ file_scale = $ scale ;
$ file_scale =~ s/\./_/ ;
$ file_scale =~ s/_00// ;
$ file_scale =~ s/(_\d+)0+$/$1/ ;
$ file_scale = 's' . $ file_scale ;
push ( @ file_parts , $ file_scale ) ;
}
elsif ( $ size )
{
my $ file_size = 'S' . $ size ;
push ( @ file_parts , $ file_size ) ;
}
my $ video_file = "$video_name-" . $ file_parts [ 0 ] . "-" . $ file_parts [ 1 ] . ".$format" ;
if ( $ overwrite || ! - s $ video_file )
{
2006-01-14 13:33:20 +00:00
Info ( "Creating video file $video_file for event $event->{Id}\n" ) ;
2005-12-16 10:05:29 +00:00
2008-07-14 15:56:52 +00:00
my $ frame_rate = sprintf ( "%.2f" , $ event - > { Frames } / $ event - > { FullLength } ) ;
if ( $ rate )
{
if ( $ rate != 1.0 )
{
$ frame_rate *= $ rate ;
}
}
elsif ( $ fps )
{
$ frame_rate = $ fps ;
}
my $ width = $ event - > { MonitorWidth } ;
my $ height = $ event - > { MonitorHeight } ;
my $ video_size = " ${width}x${height}" ;
if ( $ scale )
{
if ( $ scale != 1.0 )
{
$ width = int ( $ width * $ scale ) ;
$ height = int ( $ height * $ scale ) ;
$ video_size = " ${width}x${height}" ;
}
}
elsif ( $ size )
{
$ video_size = $ size ;
}
2013-12-16 21:32:02 +00:00
my $ command = $ Config { ZM_PATH_FFMPEG } . " -y -r $frame_rate " . $ Config { ZM_FFMPEG_INPUT_OPTIONS } . " -i %0" . $ Config { ZM_EVENT_IMAGE_DIGITS } . "d-capture.jpg -s $video_size " . $ Config { ZM_FFMPEG_OUTPUT_OPTIONS } . " '$video_file' > ffmpeg.log 2>&1" ;
2008-07-14 15:56:52 +00:00
Debug ( $ command . "\n" ) ;
my $ output = qx( $command ) ;
my $ status = $? >> 8 ;
if ( $ status )
{
Error ( "Unable to generate video, check " . $ event_path . "/ffmpeg.log for details" ) ;
exit ( - 1 ) ;
}
2005-12-16 10:05:29 +00:00
2006-01-14 13:33:20 +00:00
Info ( "Finished $video_file\n" ) ;
2005-12-16 10:05:29 +00:00
}
else
{
2006-01-14 13:33:20 +00:00
Info ( "Video file $video_file already exists for event $event->{Id}\n" ) ;
2005-12-16 10:05:29 +00:00
}
#print( STDOUT $event->{MonitorId}.'/'.$event->{Id}.'/'.$video_file."\n" );
print ( STDOUT $ video_file . "\n" ) ;
exit ( 0 ) ;