#!/usr/bin/perl -wT # # ========================================================================== # # ZoneMinder Video Creation Script, $Date$, $Revision$ # Copyright (C) 2003 Philip Coombes # # 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; # ========================================================================== # # These are the elements you need to edit to suit your installation # # ========================================================================== use constant ZM_PATH_BIN => ""; use constant ZM_PATH_WEB => ""; use constant ZM_PATH_CGI => ""; use constant ZM_DB_SERVER => ""; use constant ZM_DB_NAME => ""; use constant ZM_DB_USERB => ""; use constant ZM_DB_PASSB => ""; # Load the config from the database into the symbol table BEGIN { use DBI; no strict 'refs'; my $dbh = DBI->connect( "DBI:mysql:database=".ZM_DB_NAME.";host=".ZM_DB_SERVER, ZM_DB_USERB, ZM_DB_PASSB ); my $sql = "select * from Config"; my $sth = $dbh->prepare_cached( $sql ) or die( "Can't prepare '$sql': ".$dbh->errstr() ); my $res = $sth->execute() or die( "Can't execute: ".$sth->errstr() ); while( my $config = $sth->fetchrow_hashref() ) { *{$config->{Name}} = sub { $config->{Value} }; } $sth->finish(); $dbh->disconnect(); } use constant LOG_FILE => ZM_PATH_LOGS.'/zmvideo.log'; use constant VERBOSE => 0; # Whether to output more verbose debug # ========================================================================== # # You shouldn't need to change anything from here downwards # # ========================================================================== use DBI; use Data::Dumper; use Getopt::Long; $| = 1; $ENV{PATH} = '/bin:/usr/bin'; $ENV{SHELL} = '/bin/sh' if exists $ENV{SHELL}; delete @ENV{qw(IFS CDPATH ENV BASH_ENV)}; my $event_id; sub Usage { print( " Usage: zmvideo.pl -e ,--event= Parameters are :- -e, --event= - What event to start create the video for "); exit( -1 ); } if ( !GetOptions( 'event=i'=>\$event_id ) ) { Usage(); } if ( !$event_id || $event_id < 0 ) { print( STDERR "Please give a valid event id\n" ); Usage(); } if ( ZM_OPT_MPEG eq "no" ) { print( STDERR "Mpeg encoding is not currently enabled\n" ); exit(-1); } my $log_file = LOG_FILE; open( LOG, ">>$log_file" ) or die( "Can't open log file: $!" ); #open( STDOUT, ">&LOG" ) || die( "Can't dup stdout: $!" ); #select( STDOUT ); $| = 1; open( STDERR, ">&LOG" ) || die( "Can't dup stderr: $!" ); select( STDERR ); $| = 1; select( LOG ); $| = 1; my $dbh = DBI->connect( "DBI:mysql:database=".ZM_DB_NAME.";host=".ZM_DB_SERVER, ZM_DB_USERB, ZM_DB_PASSB ); my @filters; my $sql = "select max(F.Delta)-min(F.Delta) as FullLength,E.*,M.Name as MonitorName, 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"; my $sth = $dbh->prepare_cached( $sql ) or die( "Can't prepare '$sql': ".$dbh->errstr() ); my $res = $sth->execute() or die( "Can't execute: ".$sth->errstr() ); my $event = $sth->fetchrow_hashref(); $sth->finish(); chdir( ZM_PATH_WEB.'/'.ZM_DIR_EVENTS.'/'.$event->{MonitorName}.'/'.$event->{Id} ); ( my $video_name = $event->{Name} ) =~ s/\s/_/g; my $video_file = "$video_name.mpg"; if ( !-s $video_file ) { print( LOG "Creating video file $video_file for event $event->{Id}\n" ); if ( ZM_OPT_MPEG eq "mpeg_encode" ) { my $param_file = "$video_name.mpe"; open( PARAMS, ">$param_file" ) or die( "Can't open '$param_file': $!" ); print( PARAMS "PATTERN IBBPBBPBBPBBPBB\n" ); print( PARAMS "FORCE_ENCODE_LAST_FRAME\n" ); print( PARAMS "OUTPUT $video_file\n" ); print( PARAMS "BASE_FILE_FORMAT JPEG\n" ); print( PARAMS "GOP_SIZE 30\n" ); print( PARAMS "SLICES_PER_FRAME 1\n" ); print( PARAMS "PIXEL HALF\n" ); print( PARAMS "RANGE 10\n" ); print( PARAMS "PSEARCH_ALG LOGARITHMIC\n" ); print( PARAMS "BSEARCH_ALG CROSS2\n" ); print( PARAMS "IQSCALE 8\n" ); print( PARAMS "PQSCALE 10\n" ); print( PARAMS "BQSCALE 25\n" ); print( PARAMS "REFERENCE_FRAME ORIGINAL\n" ); print( PARAMS "FRAME_RATE 24\n" ); if ( $event->{Palette} == 1 && !ZM_COLOUR_JPEG_FILES ) { print( PARAMS "INPUT_CONVERT ".ZM_PATH_NETPBM."/jpegtopnm * | ".ZM_PATH_NETPBM."/pgmtoppm white | ".ZM_PATH_NETPBM."/ppmtojpeg\n" ); } else { print( PARAMS "INPUT_CONVERT *\n" ); } print( PARAMS "INPUT_DIR .\n" ); print( PARAMS "INPUT\n" ); for ( my $i = 1; $i <= $event->{Frames}; $i++ ) { printf( PARAMS "capture-%03d.jpg\n", $i ); } print( PARAMS "END_INPUT\n" ); close( PARAMS ); my $command = ZM_PATH_MPEG_ENCODE." $param_file >mpeg.log"; my $output = qx($command); } elsif ( ZM_OPT_MPEG eq "ffmpeg" ) { my $frame_rate = sprintf( "%.2f", $event->{Frames}/$event->{FullLength} ); my $command = ZM_PATH_FFMPEG." -y ".ZM_FFMPEG_OPTIONS." -r $frame_rate -i capture-%03d.jpg $video_file > mpeg.log"; my $output = qx($command); } else { die( "Bogus mpeg option ".ZM_OPT_MPEG."\n" ); } my $status = $? >> 8; if ( $status ) { die( "Error: $status" ); } print( LOG "Finished $video_file\n" ); } else { print( LOG "Video file $video_file already exists for event $event->{Id}\n" ); } #print( STDOUT $event->{MonitorName}.'/'.$event->{Id}.'/'.$video_file."\n" ); print( STDOUT $video_file."\n" ); exit( 0 );