2007-08-30 18:43:06 +08:00
|
|
|
#!/usr/bin/perl -wT
|
|
|
|
#
|
|
|
|
# ==========================================================================
|
|
|
|
#
|
|
|
|
# ZoneMinder Control Script, $Date$, $Revision$
|
|
|
|
# Copyright (C) 2003, 2004, 2005, 2006 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 continuously monitors the recorded events for the given
|
|
|
|
# monitor and applies any filters which would delete and/or upload
|
|
|
|
# matching events
|
|
|
|
#
|
|
|
|
use strict;
|
|
|
|
|
|
|
|
# ==========================================================================
|
|
|
|
#
|
|
|
|
# These are the elements you can edit to suit your installation
|
|
|
|
#
|
|
|
|
# ==========================================================================
|
|
|
|
|
|
|
|
use constant DBG_ID => "zmcontrol"; # Tag that appears in debug to identify source
|
|
|
|
use constant DBG_LEVEL => 1; # 0 is errors, warnings and info only, > 0 for debug
|
|
|
|
|
|
|
|
# ==========================================================================
|
|
|
|
|
|
|
|
use ZoneMinder;
|
|
|
|
use Getopt::Long;
|
|
|
|
use POSIX qw/strftime EPIPE/;
|
|
|
|
use Socket;
|
|
|
|
use Data::Dumper;
|
|
|
|
use Module::Load;
|
|
|
|
use PHP::Serialization qw(serialize unserialize);
|
|
|
|
|
|
|
|
use constant MAX_CONNECT_DELAY => 10;
|
|
|
|
use constant MAX_COMMAND_WAIT => 1800;
|
|
|
|
|
|
|
|
$| = 1;
|
|
|
|
|
|
|
|
$ENV{PATH} = '/bin:/usr/bin';
|
|
|
|
$ENV{SHELL} = '/bin/sh' if exists $ENV{SHELL};
|
|
|
|
delete @ENV{qw(IFS CDPATH ENV BASH_ENV)};
|
|
|
|
|
|
|
|
sub Usage
|
|
|
|
{
|
|
|
|
print( "
|
2007-11-05 01:30:29 +08:00
|
|
|
Usage: zmcontrol.pl --id <monitor_id> --command=<command> <various options>
|
2007-08-30 18:43:06 +08:00
|
|
|
");
|
|
|
|
exit( -1 );
|
|
|
|
}
|
|
|
|
|
|
|
|
zmDbgInit( DBG_ID, level=>DBG_LEVEL );
|
|
|
|
|
|
|
|
my $arg_string = join( " ", @ARGV );
|
|
|
|
|
|
|
|
my $id;
|
|
|
|
my %options;
|
|
|
|
|
|
|
|
if ( !GetOptions(
|
|
|
|
'id=i'=>\$id,
|
|
|
|
'command=s'=>\$options{command},
|
|
|
|
'xcoord=i'=>\$options{xcoord},
|
|
|
|
'ycoord=i'=>\$options{ycoord},
|
|
|
|
'speed=i'=>\$options{speed},
|
|
|
|
'step=i'=>\$options{step},
|
|
|
|
'panspeed=i'=>\$options{panspeed},
|
|
|
|
'tiltspeed=i'=>\$options{tiltspeed},
|
|
|
|
'panstep=i'=>\$options{panstep},
|
|
|
|
'tiltstep=i'=>\$options{tiltstep},
|
|
|
|
'preset=i'=>\$options{preset},
|
|
|
|
'autostop'=>\$options{autostop},
|
|
|
|
)
|
|
|
|
)
|
|
|
|
{
|
|
|
|
Usage();
|
|
|
|
}
|
|
|
|
|
2007-11-05 01:30:29 +08:00
|
|
|
if ( !$id || !$options{command} )
|
|
|
|
{
|
|
|
|
print( STDERR "Please give a valid monitor id and command\n" );
|
|
|
|
Usage();
|
|
|
|
}
|
|
|
|
|
2007-08-30 18:43:06 +08:00
|
|
|
( $id ) = $id =~ /^(\w+)$/;
|
|
|
|
|
|
|
|
Debug( $arg_string );
|
|
|
|
|
|
|
|
my $sock_file = ZM_PATH_SOCKS.'/zmcontrol-'.$id.'.sock';
|
|
|
|
|
|
|
|
socket( CLIENT, PF_UNIX, SOCK_STREAM, 0 ) or Fatal( "Can't open socket: $!" );
|
|
|
|
|
|
|
|
my $saddr = sockaddr_un( $sock_file );
|
|
|
|
my $server_up = connect( CLIENT, $saddr );
|
|
|
|
if ( !$server_up )
|
|
|
|
{
|
|
|
|
# The server isn't there
|
2007-11-05 01:30:29 +08:00
|
|
|
my $monitor = zmDbGetMonitorAndControl( $id );
|
|
|
|
if ( !$monitor )
|
|
|
|
{
|
|
|
|
Fatal( "Unable to load control data for monitor $id" );
|
|
|
|
}
|
|
|
|
my $protocol = $monitor->{Protocol};
|
|
|
|
|
2007-08-30 18:43:06 +08:00
|
|
|
Info( "Starting control server $id/$protocol" );
|
|
|
|
close( CLIENT );
|
|
|
|
|
|
|
|
if ( my $cpid = fork() )
|
|
|
|
{
|
|
|
|
zmDbgInit( DBG_ID, level=>DBG_LEVEL );
|
|
|
|
|
|
|
|
# Parent process just sleep and fall through
|
|
|
|
socket( CLIENT, PF_UNIX, SOCK_STREAM, 0 ) or die( "Can't open socket: $!" );
|
|
|
|
my $attempts = 0;
|
|
|
|
while (!connect( CLIENT, $saddr ))
|
|
|
|
{
|
|
|
|
$attempts++;
|
|
|
|
Fatal( "Can't connect: $!" ) if ($attempts > MAX_CONNECT_DELAY);
|
|
|
|
sleep(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
elsif ( defined($cpid) )
|
|
|
|
{
|
|
|
|
close( STDOUT );
|
|
|
|
close( STDERR );
|
|
|
|
|
|
|
|
setpgrp();
|
|
|
|
|
|
|
|
zmDbgInit( DBG_ID, level=>DBG_LEVEL );
|
|
|
|
|
|
|
|
Info( "Control server $id/$protocol starting at ".strftime( '%y/%m/%d %H:%M:%S', localtime() ) );
|
|
|
|
|
2007-11-05 01:30:29 +08:00
|
|
|
$0 = $0." --id $id";
|
2007-08-30 18:43:06 +08:00
|
|
|
|
|
|
|
load "ZoneMinder::Control::$protocol";
|
|
|
|
|
|
|
|
my $control = "ZoneMinder::Control::$protocol"->new( $id );
|
|
|
|
my $control_key = $control->getKey();
|
|
|
|
$control->loadMonitor();
|
|
|
|
|
|
|
|
$control->open();
|
|
|
|
|
|
|
|
socket( SERVER, PF_UNIX, SOCK_STREAM, 0 ) or Fatal( "Can't open socket: $!" );
|
|
|
|
unlink( $sock_file );
|
|
|
|
bind( SERVER, $saddr ) or Fatal( "Can't bind: $!" );
|
|
|
|
listen( SERVER, SOMAXCONN ) or Fatal( "Can't listen: $!" );
|
|
|
|
|
|
|
|
my $rin = '';
|
|
|
|
vec( $rin, fileno(SERVER), 1 ) = 1;
|
|
|
|
my $win = $rin;
|
|
|
|
my $ein = $win;
|
|
|
|
my $timeout = MAX_COMMAND_WAIT;
|
|
|
|
while( 1 )
|
|
|
|
{
|
|
|
|
my $nfound = select( my $rout = $rin, undef, undef, $timeout );
|
|
|
|
if ( $nfound > 0 )
|
|
|
|
{
|
|
|
|
if ( vec( $rout, fileno(SERVER), 1 ) )
|
|
|
|
{
|
|
|
|
my $paddr = accept( CLIENT, SERVER );
|
|
|
|
my $message = <CLIENT>;
|
|
|
|
|
|
|
|
next if ( !$message );
|
|
|
|
|
|
|
|
my $params = unserialize( $message );
|
|
|
|
#Debug( Dumper( $params ) );
|
|
|
|
|
|
|
|
my $command = $params->{command};
|
|
|
|
$control->$command( $params );
|
|
|
|
close( CLIENT );
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
Fatal( "Bogus descriptor" );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
elsif ( $nfound < 0 )
|
|
|
|
{
|
|
|
|
if ( $! == EPIPE )
|
|
|
|
{
|
|
|
|
Error( "Can't select: $!" );
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
Fatal( "Can't select: $!" );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
#print( "Select timed out\n" );
|
|
|
|
last;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Info( "Control server $id/$protocol exiting at ".strftime( '%y/%m/%d %H:%M:%S', localtime() ) );
|
|
|
|
unlink( $sock_file );
|
|
|
|
$control->close();
|
|
|
|
exit( 0 );
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
Fatal( "Can't fork: $!" );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
# The server is there, connect to it
|
|
|
|
#print( "Writing commands\n" );
|
|
|
|
CLIENT->autoflush();
|
|
|
|
|
|
|
|
my $message = serialize( \%options );
|
|
|
|
print( CLIENT $message );
|
|
|
|
shutdown( CLIENT, 1 );
|
|
|
|
|
|
|
|
exit( 0 );
|