From 599378943bf3cac92fda33fc450d8ac04db41644 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Michal=20Josef=20=C5=A0pa=C4=8Dek?= Date: Fri, 5 May 2023 12:14:28 +0200 Subject: [PATCH 1/4] Fix usage of IPv6 There were a bad check which break IPv6 functionality --- lib/SNMP_Session.pm | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/lib/SNMP_Session.pm b/lib/SNMP_Session.pm index 9d643ac..eab42df 100644 --- a/lib/SNMP_Session.pm +++ b/lib/SNMP_Session.pm @@ -902,7 +902,7 @@ sub open { $local_hostname,$ipv4only) = @_; my($remote_addr,$socket,$sockfamily); - $ipv4only = 1 unless defined $ipv4only; + $ipv4only = 1 if defined $ipv4only; $sockfamily = AF_INET; $community = 'public' unless defined $community; From 258adcff4c5af890d9ecc3ba8534f976d252ed30 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Michal=20Josef=20=C5=A0pa=C4=8Dek?= Date: Fri, 5 May 2023 14:12:44 +0200 Subject: [PATCH 2/4] Remove prefix Socket6 This is preparation for new networking support. All functions are imported from Socket6. Functionality is same. --- lib/SNMP_Session.pm | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/lib/SNMP_Session.pm b/lib/SNMP_Session.pm index eab42df..2692eab 100644 --- a/lib/SNMP_Session.pm +++ b/lib/SNMP_Session.pm @@ -140,8 +140,8 @@ BEGIN { if (eval {local $SIG{__DIE__};require Socket6;} && eval {local $SIG{__DIE__};require IO::Socket::INET6; IO::Socket::INET6->VERSION("1.26");}) { - Socket6->import(qw(inet_pton inet_ntop getaddrinfo)); - $ipv6_addr_len = length(Socket6::pack_sockaddr_in6(161, inet_pton(AF_INET6(), "::1"))); + Socket6->import(qw(inet_pton inet_ntop getaddrinfo unpack_sockaddr_in6)); + $ipv6_addr_len = length(pack_sockaddr_in6(161, inet_pton(AF_INET6(), "::1"))); $SNMP_Session::ipv6available = 1; } eval 'local $SIG{__DIE__};local $SIG{__WARN__};$dont_wait_flags = MSG_DONTWAIT();'; @@ -688,7 +688,7 @@ sub pretty_address { # complaining about AF_INET6 when Socket6 is not available if( (defined $ipv6_addr_len) && (length $addr == $ipv6_addr_len)) { - ($port,$addrunpack) = Socket6::unpack_sockaddr_in6 ($addr); + ($port,$addrunpack) = unpack_sockaddr_in6 ($addr); $addrstr = inet_ntop (AF_INET6(), $addrunpack); } else { ($port,$addrunpack) = unpack_sockaddr_in ($addr); @@ -774,7 +774,7 @@ sub receive_trap ($ ) { my ($iaddr, $port, $af); if ((defined $ipv6_addr_len) && (length $remote_addr == $ipv6_addr_len)) { - ($port, $iaddr) = Socket6::unpack_sockaddr_in6 ($remote_addr); + ($port, $iaddr) = unpack_sockaddr_in6 ($remote_addr); $af = AF_INET6; } else { ($port, $iaddr) = unpack_sockaddr_in ($remote_addr); @@ -1125,8 +1125,8 @@ sub sa_equal_p ($$$) { ($p2,$a2) = unpack_sockaddr_in ($sa2); } elsif($this->{'sockfamily'} == AF_INET6()) { # IPv6 addresses - ($p1,$a1) = Socket6::unpack_sockaddr_in6 ($sa1); - ($p2,$a2) = Socket6::unpack_sockaddr_in6 ($sa2); + ($p1,$a1) = unpack_sockaddr_in6 ($sa1); + ($p2,$a2) = unpack_sockaddr_in6 ($sa2); } else { return 0; } From aa697eaf88370e6ba5a4ed18c38cdc869db0649e Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Michal=20Josef=20=C5=A0pa=C4=8Dek?= Date: Fri, 5 May 2023 14:12:57 +0200 Subject: [PATCH 3/4] Remove obsolete comment --- lib/SNMP_Session.pm | 3 --- 1 file changed, 3 deletions(-) diff --git a/lib/SNMP_Session.pm b/lib/SNMP_Session.pm index 2692eab..c13f63a 100644 --- a/lib/SNMP_Session.pm +++ b/lib/SNMP_Session.pm @@ -684,9 +684,6 @@ sub pretty_address { my($addr) = shift; my($port, $addrunpack, $addrstr); - # Disable strict subs to stop old versions of perl from - # complaining about AF_INET6 when Socket6 is not available - if( (defined $ipv6_addr_len) && (length $addr == $ipv6_addr_len)) { ($port,$addrunpack) = unpack_sockaddr_in6 ($addr); $addrstr = inet_ntop (AF_INET6(), $addrunpack); From 7de9937d03f3566825a992ae20522c9b7cacce71 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Michal=20Josef=20=C5=A0pa=C4=8Dek?= Date: Fri, 5 May 2023 14:14:34 +0200 Subject: [PATCH 4/4] Add backward compatile support for IPv6 We have Socket with IPv6 functionality. We have IO::Socket::IP with IPv4/6 functionality. Commit is adding new networking code before legacy code. --- lib/SNMP_Session.pm | 61 ++++++++++++++++++++++++++++++++++----------- 1 file changed, 47 insertions(+), 14 deletions(-) diff --git a/lib/SNMP_Session.pm b/lib/SNMP_Session.pm index c13f63a..2e0b163 100644 --- a/lib/SNMP_Session.pm +++ b/lib/SNMP_Session.pm @@ -136,13 +136,19 @@ my $dont_wait_flags; BEGIN { $ipv6_addr_len = undef; $SNMP_Session::ipv6available = 0; + $SNMP_Session::old_ipv6_networking = 0; $dont_wait_flags = 0; - if (eval {local $SIG{__DIE__};require Socket6;} && + if (eval {local $SIG{__DIE__};Socket->VERSION("2.000");}) { + Socket->import(qw(inet_pton inet_ntop getaddrinfo unpack_sockaddr_in6)); + $ipv6_addr_len = length(pack_sockaddr_in6(161, inet_pton(AF_INET6(), "::1"))); + $SNMP_Session::ipv6available = 1; + } elsif (eval {local $SIG{__DIE__};require Socket6;} && eval {local $SIG{__DIE__};require IO::Socket::INET6; IO::Socket::INET6->VERSION("1.26");}) { Socket6->import(qw(inet_pton inet_ntop getaddrinfo unpack_sockaddr_in6)); $ipv6_addr_len = length(pack_sockaddr_in6(161, inet_pton(AF_INET6(), "::1"))); $SNMP_Session::ipv6available = 1; + $SNMP_Session::old_ipv6_networking = 1; } eval 'local $SIG{__DIE__};local $SIG{__WARN__};$dont_wait_flags = MSG_DONTWAIT();'; } @@ -846,8 +852,16 @@ use Carp; BEGIN { if($SNMP_Session::ipv6available) { - import IO::Socket::INET6; - Socket6->import(qw(inet_pton inet_ntop getaddrinfo)); + if (! $SNMP_Session::old_ipv6_networking + && eval{local $SIG{__DIE__}; require IO::Socket::IP; IO::Socket::IP->VERSION('0.25');}) { + + import IO::Socket::IP; + Socket->import(qw(inet_pton inet_ntop getaddrinfo)); + } elsif (eval{local $SIG{__DIE__}; require IO::Socket::INET6;}) { + import IO::Socket::INET6; + Socket6->import(qw(inet_pton inet_ntop getaddrinfo)); + $SNMP_Session::old_ipv6_networking = 1; + } } } @@ -937,11 +951,22 @@ sub open { $remote_hostname = $1; } my (@res, $socktype_tmp, $proto_tmp, $canonname_tmp); - @res = getaddrinfo($remote_hostname, $port, AF_UNSPEC, SOCK_DGRAM); - ($sockfamily, $socktype_tmp, $proto_tmp, $remote_addr, $canonname_tmp) = @res; - if (scalar(@res) < 5) { - return $this->error_return ("can't resolve \"$remote_hostname\" to IPv6 address"); - } + if ($SNMP_Session::old_ipv6_networking) { + @res = getaddrinfo($remote_hostname, $port, AF_UNSPEC, SOCK_DGRAM); + ($sockfamily, $socktype_tmp, $proto_tmp, $remote_addr, $canonname_tmp) = @res; + if (scalar(@res) < 5) { + return $this->error_return ("can't resolve \"$remote_hostname\" to IPv6 address"); + } + } else { + my ($err, $result) = getaddrinfo($remote_hostname, $port, + {'family' => AF_UNSPEC, 'socktype' => SOCK_DGRAM}); + ($sockfamily, $socktype_tmp, $proto_tmp, $remote_addr, $canonname_tmp) + = ($result->{'family'}, $result->{'socktype'}, $result->{'protocol'}, + $result->{'addr'}, $result->{'canonname'}); + if ($err) { + return $this->error_return ("can't resolve \"$remote_hostname\" to IPv6 address"); + } + } } else { $sockfamily = AF_INET6; } @@ -955,12 +980,20 @@ sub open { LocalPort => $local_port) || return $this->error_return ("creating socket: $!"); } else { - $socket = IO::Socket::INET6->new(Domain => AF_INET6, - Proto => 17, - Type => SOCK_DGRAM, - LocalAddr => $local_hostname, - LocalPort => $local_port) - || return $this->error_return ("creating socket: $!"); + my %net_setting = ( + Domain => AF_INET6, + Proto => 17, + Type => SOCK_DGRAM, + LocalAddr => $local_hostname, + LocalPort => $local_port, + ); + if ($SNMP_Session::old_ipv6_networking) { + $socket = IO::Socket::INET6->new(%net_setting) + || return $this->error_return ("creating socket: $!"); + } else { + $socket = IO::Socket::IP->new(%net_setting) + || return $this->error_return ("creating socket: $!"); + } $the_socket{$sockfamily} = $socket if $SNMP_Session::recycle_socket; }