From 93cd51bc6221ffc846c69135f2746120683b44c5 Mon Sep 17 00:00:00 2001 From: alvar <8402811+oxzi@users.noreply.github.com> Date: Mon, 18 Mar 2024 09:05:04 +0000 Subject: check_ircd: IPv6 support (#1995) The prior bindRemote function was restricted to IPv4 by explicitly using address family specific functions as pack_sockaddr_in or unparametrized gethostbyname. Since Perl 5.14, released in early 2013, there is IO::Socket::IP, which supports lookups and connects for a dual stacked network. By switching the connection establishment code, the plugin is now able to establish connections to IPv6-only hosts. Furthermore, additional -4 and -6 flags were introduced to enforce a specific address family. > $ netstat -p tcp -ln | grep 6667 > tcp6 0 0 *.6667 *.* LISTEN > $ ./plugins-scripts/check_ircd -H localhost > IRCD ok - Current Local Users: 4 > $ ./plugins-scripts/check_ircd -H localhost -4 > IRCD UNKNOWN: Could not start socket (Connection refused) > $ ./plugins-scripts/check_ircd -H localhost -6 > IRCD ok - Current Local Users: 4 --- plugins-scripts/check_ircd.pl | 61 ++++++++++++++++++------------------------- 1 file changed, 26 insertions(+), 35 deletions(-) diff --git a/plugins-scripts/check_ircd.pl b/plugins-scripts/check_ircd.pl index 4822fe68..15a70802 100755 --- a/plugins-scripts/check_ircd.pl +++ b/plugins-scripts/check_ircd.pl @@ -40,15 +40,16 @@ # ----------------------------------------------------------------[ Require ]-- -require 5.004; +require 5.14.0; # -------------------------------------------------------------------[ Uses ]-- -use Socket; use strict; +use IO::Socket::IP; use Getopt::Long; -use vars qw($opt_V $opt_h $opt_t $opt_p $opt_H $opt_w $opt_c $verbose); +use vars qw($opt_V $opt_h $opt_t $opt_p $opt_H $opt_w $opt_c $opt_4 $opt_6 $verbose); use vars qw($PROGNAME); +use vars qw($ClientSocket); use FindBin; use lib "$FindBin::Bin"; use utils qw($TIMEOUT %ERRORS &print_revision &support &usage); @@ -58,7 +59,6 @@ use utils qw($TIMEOUT %ERRORS &print_revision &support &usage); sub print_help (); sub print_usage (); sub connection ($$$$); -sub bindRemote ($$); # -------------------------------------------------------------[ Environment ]-- @@ -104,7 +104,7 @@ sub connection ($$$$) $answer = "Server $in_remotehost has less than 0 users! Something is Really WRONG!\n"; } - print ClientSocket "quit\n"; + print $ClientSocket "quit\n"; print $answer; exit $ERRORS{$state}; } @@ -112,7 +112,7 @@ sub connection ($$$$) # ------------------------------------------------------------[ print_usage ]-- sub print_usage () { - print "Usage: $PROGNAME -H [-w ] [-c ] [-p ]\n"; + print "Usage: $PROGNAME -H [-w ] [-c ] [-p ] [ -4|-6 ]\n"; } # -------------------------------------------------------------[ print_help ]-- @@ -135,33 +135,15 @@ Perl Check IRCD plugin for monitoring Number of connected users which generates a critical state (Default: 100) -p, --port=INTEGER Port that the ircd daemon is running on (Default: 6667) +-4, --use-ipv4 + Use IPv4 connection +-6, --use-ipv6 + Use IPv6 connection -v, --verbose Print extra debugging information "; } -# -------------------------------------------------------------[ bindRemote ]-- - -sub bindRemote ($$) -{ - my ($in_remotehost, $in_remoteport) = @_; - my $proto = getprotobyname('tcp'); - my $that; - my ($name, $aliases,$type,$len,$thataddr) = gethostbyname($in_remotehost); - - if (!socket(ClientSocket,AF_INET, SOCK_STREAM, $proto)) { - print "IRCD UNKNOWN: Could not start socket ($!)\n"; - exit $ERRORS{"UNKNOWN"}; - } - $that = pack_sockaddr_in ($in_remoteport, $thataddr); - if (!connect(ClientSocket, $that)) { - print "IRCD UNKNOWN: Could not connect socket ($!)\n"; - exit $ERRORS{"UNKNOWN"}; - } - select(ClientSocket); $| = 1; select(STDOUT); - return \*ClientSocket; -} - # ===================================================================[ MAIN ]== MAIN: @@ -177,6 +159,8 @@ MAIN: "w=i" => \$opt_w, "warning=i" => \$opt_w, "c=i" => \$opt_c, "critical=i" => \$opt_c, "p=i" => \$opt_p, "port=i" => \$opt_p, + "4" => \$opt_4, "use-ipv4" => \$opt_4, + "6" => \$opt_6, "use-ipv6" => \$opt_6, "H=s" => \$opt_H, "hostname=s" => \$opt_H); if ($opt_V) { @@ -187,7 +171,7 @@ MAIN: if ($opt_h) {print_help(); exit $ERRORS{'UNKNOWN'};} ($opt_H) || ($opt_H = shift @ARGV) || usage("Host name/address not specified\n"); - my $remotehost = $1 if ($opt_H =~ /([-.A-Za-z0-9]+)/); + my $remotehost = $1 if ($opt_H =~ /([-.:%A-Za-z0-9]+)/); ($remotehost) || usage("Invalid host: $opt_H\n"); ($opt_w) || ($opt_w = shift @ARGV) || ($opt_w = 50); @@ -212,21 +196,28 @@ MAIN: alarm($TIMEOUT); - my ($name, $alias, $proto) = getprotobyname('tcp'); - print "MAIN(debug): binding to remote host: $remotehost -> $remoteport\n" if $verbose; - my $ClientSocket = &bindRemote($remotehost,$remoteport); + $ClientSocket = IO::Socket::IP->new( + PeerHost => $remotehost, + PeerService => $remoteport, + Family => $opt_4 ? AF_INET : $opt_6 ? AF_INET6 : undef, + Type => SOCK_STREAM, + ); + if (!$ClientSocket) { + print "IRCD UNKNOWN: Could not start socket ($!)\n"; + exit $ERRORS{"UNKNOWN"}; + } - print ClientSocket "NICK $NICK\nUSER $USER_INFO\n"; + print $ClientSocket "NICK $NICK\nUSER $USER_INFO\n"; - while () { + while (<$ClientSocket>) { print "MAIN(debug): default var = $_\n" if $verbose; # DALnet,LagNet,UnderNet etc. Require this! # Replies with a PONG when presented with a PING query. # If a server doesn't require it, it will be ignored. - if (m/^PING (.*)/) {print ClientSocket "PONG $1\n";} + if (m/^PING (.*)/) {print $ClientSocket "PONG $1\n";} alarm(0); -- cgit v1.2.3-74-g34f1