mirror of
https://github.com/ZoneMinder/zoneminder.git
synced 2026-03-24 16:51:47 -04:00
Add --protocol, --address, and --device CLI options that bypass the database and socket machinery to load a control module directly and execute a single command. This enables standalone testing of control modules (e.g. ONVIF.pm get_config) without a running ZoneMinder instance. Also update POD with full documentation and examples. Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
299 lines
8.2 KiB
Perl
299 lines
8.2 KiB
Perl
#!@PERL_EXECUTABLE@ -wT
|
|
#
|
|
# ==========================================================================
|
|
#
|
|
# ZoneMinder Control Script, $Date$, $Revision$
|
|
# Copyright (C) 2001-2008 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
#
|
|
# ==========================================================================
|
|
|
|
use strict;
|
|
use warnings;
|
|
|
|
@EXTRA_PERL_LIB@
|
|
use ZoneMinder;
|
|
use Getopt::Long;
|
|
use autouse 'Pod::Usage'=>qw(pod2usage);
|
|
use POSIX qw/strftime EPIPE EINTR/;
|
|
use Socket;
|
|
use Data::Dumper;
|
|
|
|
use constant MAX_CONNECT_DELAY => 15;
|
|
use constant MAX_COMMAND_WAIT => 1800;
|
|
|
|
$| = 1;
|
|
|
|
$ENV{PATH} = '/bin:/usr/bin:/usr/local/bin';
|
|
$ENV{SHELL} = '/bin/sh' if exists $ENV{SHELL};
|
|
delete @ENV{qw(IFS CDPATH ENV BASH_ENV)};
|
|
|
|
my $arg_string = join(' ', @ARGV);
|
|
|
|
my $id;
|
|
my $protocol;
|
|
my $address;
|
|
my $device;
|
|
my %options;
|
|
|
|
GetOptions(
|
|
'id=i' =>\$id,
|
|
'protocol=s' =>\$protocol,
|
|
'address=s' =>\$address,
|
|
'device=s' =>\$device,
|
|
'command=s' =>\$options{command},
|
|
'xcoord=f' =>\$options{xcoord},
|
|
'ycoord=f' =>\$options{ycoord},
|
|
'speed=f' =>\$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},
|
|
) or pod2usage(-exitstatus => -1);
|
|
|
|
if ($protocol) {
|
|
# Direct protocol mode: bypass DB and socket, load module and run command directly
|
|
logInit(id=>'zmcontrol_protocol');
|
|
|
|
my $command = $options{command};
|
|
if (!$command) {
|
|
print(STDERR "Please supply a --command when using --protocol mode\n");
|
|
pod2usage(-exitstatus => -1);
|
|
}
|
|
|
|
# Validate protocol name (alphanumeric and underscore only)
|
|
($protocol) = $protocol =~ /^(\w+)$/;
|
|
if (!$protocol) {
|
|
Fatal('Invalid protocol name');
|
|
}
|
|
|
|
my $module = "ZoneMinder::Control::$protocol";
|
|
eval "require $module"
|
|
or Fatal("Failed to load protocol module $module: $@");
|
|
|
|
my $control = bless {
|
|
Monitor => {
|
|
ControlDevice => $device // '',
|
|
ControlAddress => $address // '',
|
|
AutoStopTimeout => $options{autostop} // 0,
|
|
},
|
|
}, $module;
|
|
|
|
if (!$control->open()) {
|
|
Fatal('Failed to open control');
|
|
}
|
|
|
|
Info("Direct protocol mode: $module -> $command");
|
|
my $result = $control->$command(\%options);
|
|
|
|
if (defined $result) {
|
|
if (ref $result) {
|
|
print(Dumper($result));
|
|
} else {
|
|
print("$result\n");
|
|
}
|
|
}
|
|
|
|
$control->close();
|
|
exit(0);
|
|
}
|
|
|
|
if (!$id) {
|
|
print(STDERR "Please give a valid monitor id or use --protocol for direct mode\n");
|
|
pod2usage(-exitstatus => -1);
|
|
}
|
|
|
|
($id) = $id =~ /^(\w+)$/;
|
|
logInit($id?(id=>'zmcontrol_'.$id):());
|
|
|
|
my $sock_file = $Config{ZM_PATH_SOCKS}.'/zmcontrol-'.$id.'.sock';
|
|
Debug("zmcontrol: arg string: $arg_string sock file $sock_file");
|
|
|
|
socket(CLIENT, PF_UNIX, SOCK_STREAM, 0) or Fatal("Can't open socket: $!");
|
|
|
|
my $saddr = sockaddr_un($sock_file);
|
|
|
|
if ($options{command}) {
|
|
# Have a command, so we are the client, connect to the server and send it.
|
|
|
|
my $tries = 10;
|
|
my $server_up;
|
|
while ( $tries and ! ( $server_up = connect(CLIENT, $saddr) ) ) {
|
|
Debug("Failed to connect to zmcontrol server at $sock_file");
|
|
runCommand("zmdc.pl start zmcontrol.pl --id $id");
|
|
sleep 1;
|
|
$tries -= 1;
|
|
}
|
|
if ( $server_up ) {
|
|
# The server is there, connect to it
|
|
#print( "Writing commands\n" );
|
|
CLIENT->autoflush();
|
|
|
|
if ( $options{command} ) {
|
|
my $message = jsonEncode(\%options);
|
|
print(CLIENT $message);
|
|
}
|
|
shutdown(CLIENT, 1);
|
|
} else {
|
|
Error("Unable to connect to zmcontrol server at $sock_file");
|
|
}
|
|
} else {
|
|
# We are the server
|
|
require ZoneMinder::Monitor;
|
|
|
|
my $monitor = ZoneMinder::Monitor->find_one(Id=>$id);
|
|
Fatal("Unable to load control data for monitor $id") if !$monitor;
|
|
|
|
my $control = $monitor->Control();
|
|
|
|
my $protocol = $control->{Protocol};
|
|
if (!$protocol) {
|
|
Fatal('No protocol is set in monitor. Please edit the monitor, edit control type, select the control capability and fill in the Protocol field');
|
|
}
|
|
|
|
Info("Starting control server $id/$protocol");
|
|
close(CLIENT);
|
|
|
|
my $zm_terminate = 0;
|
|
sub TermHandler {
|
|
Info('Received TERM, exiting');
|
|
$zm_terminate = 1;
|
|
}
|
|
$SIG{TERM} = \&TermHandler;
|
|
$SIG{INT} = \&TermHandler;
|
|
|
|
Info("Control server $id/$protocol starting at " .strftime('%y/%m/%d %H:%M:%S', localtime()));
|
|
|
|
$0 = $0.' --id '.$id;
|
|
|
|
my $control_key = $control->getKey();
|
|
$control->loadMonitor();
|
|
if (!$control->open()) {
|
|
Warning("Failed to open control");
|
|
}
|
|
|
|
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 (!$zm_terminate) {
|
|
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>;
|
|
close(CLIENT);
|
|
|
|
next if !$message;
|
|
|
|
my $params = jsonDecode($message);
|
|
Debug(Dumper($params));
|
|
|
|
my $command = $params->{command};
|
|
if ( $command eq 'quit' ) {
|
|
last;
|
|
} elsif ( $command ) {
|
|
$control->$command($params);
|
|
} else {
|
|
Warning("No command in $message");
|
|
}
|
|
} else {
|
|
Fatal('Bogus descriptor');
|
|
}
|
|
} elsif ( $nfound < 0 ) {
|
|
if ( $! == EINTR ) {
|
|
# Likely just SIGHUP
|
|
Debug("Can't select: $!");
|
|
} elsif ( $! == EPIPE ) {
|
|
Error("Can't select: $!");
|
|
} else {
|
|
Fatal("Can't select: $!");
|
|
}
|
|
} else {
|
|
Debug('Select timed out');
|
|
}
|
|
} # end while !$zm_terminate
|
|
Info("Control server $id/$protocol exiting");
|
|
unlink($sock_file);
|
|
$control->close();
|
|
} # end if !server up
|
|
|
|
exit(0);
|
|
|
|
1;
|
|
__END__
|
|
|
|
=head1 NAME
|
|
|
|
zmcontrol.pl - ZoneMinder control script
|
|
|
|
=head1 SYNOPSIS
|
|
|
|
zmcontrol.pl --id {monitor_id} [--command={command}] [various options]
|
|
zmcontrol.pl --protocol {module} --address {addr} [--device {dev}] --command {cmd}
|
|
|
|
=head1 DESCRIPTION
|
|
|
|
ZoneMinder camera control script. Operates in two modes:
|
|
|
|
B<Monitor mode> (--id): Connects to the control daemon for the given monitor,
|
|
or starts the daemon if run without --command. Requires a running ZoneMinder
|
|
instance with database access.
|
|
|
|
B<Direct protocol mode> (--protocol): Loads a ZoneMinder::Control module
|
|
directly and executes a single command without requiring database access or a
|
|
running ZoneMinder instance. Useful for testing control modules.
|
|
|
|
=head1 OPTIONS
|
|
|
|
--id [ monitor_id ] - Monitor ID (monitor mode)
|
|
--protocol [ name ] - Control module name, e.g. ONVIF (direct mode)
|
|
--address [ addr ] - ControlAddress, e.g. user:pass@host:port (direct mode)
|
|
--device [ dev ] - ControlDevice / profile token (direct mode)
|
|
--command [ cmd ] - Command to execute (e.g. moveConUp, get_config)
|
|
--autostop -
|
|
--xcoord [ arg ] - X-coord
|
|
--ycoord [ arg ] - Y-coord
|
|
--speed [ arg ] - Speed
|
|
--step [ arg ] -
|
|
--panspeed [ arg ] -
|
|
--panstep [ arg ] -
|
|
--tiltspeed [ arg ] -
|
|
--tiltstep [ arg ] -
|
|
--preset [ arg ] -
|
|
|
|
=head1 EXAMPLES
|
|
|
|
# Direct protocol testing (no database needed)
|
|
zmcontrol.pl --protocol ONVIF --address admin:pass@192.168.1.100 --command get_config
|
|
|
|
# With profile token
|
|
zmcontrol.pl --protocol ONVIF --address admin:pass@cam:8080 --device prof0 --command get_config
|
|
|
|
# Send command to running monitor daemon
|
|
zmcontrol.pl --id 1 --command moveConUp
|
|
|
|
=cut
|