summaryrefslogtreecommitdiff
path: root/regress/sys/net/pf_divert/Client.pm
diff options
context:
space:
mode:
authorAlexander Bluhm <bluhm@cvs.openbsd.org>2013-06-03 05:06:39 +0000
committerAlexander Bluhm <bluhm@cvs.openbsd.org>2013-06-03 05:06:39 +0000
commit2ff2ceee1c6b079f25d797b92b8012338950712d (patch)
treef8d32197b18decf805b1347e3675791fb2cbefa9 /regress/sys/net/pf_divert/Client.pm
parent021e76db6fd34fa2ba4b21c5fce3d5341fe2563e (diff)
Add a regression test suite for the pf divert-to and divert-reply
feature. It requires two machines, the local host is running the regression test, the remote host gets pf divert rules installed. The diverted TCP connections are running between these hosts in both directions. The remote host is controlled via ssh.
Diffstat (limited to 'regress/sys/net/pf_divert/Client.pm')
-rw-r--r--regress/sys/net/pf_divert/Client.pm117
1 files changed, 117 insertions, 0 deletions
diff --git a/regress/sys/net/pf_divert/Client.pm b/regress/sys/net/pf_divert/Client.pm
new file mode 100644
index 00000000000..f9f4160e23e
--- /dev/null
+++ b/regress/sys/net/pf_divert/Client.pm
@@ -0,0 +1,117 @@
+# $OpenBSD: Client.pm,v 1.1.1.1 2013/06/03 05:06:38 bluhm Exp $
+
+# Copyright (c) 2010-2013 Alexander Bluhm <bluhm@openbsd.org>
+#
+# Permission to use, copy, modify, and distribute this software for any
+# purpose with or without fee is hereby granted, provided that the above
+# copyright notice and this permission notice appear in all copies.
+#
+# THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
+# WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
+# MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
+# ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
+# WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
+# ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
+# OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
+
+use strict;
+use warnings;
+
+package Client;
+use parent 'Proc';
+use Carp;
+use Socket qw(IPPROTO_TCP TCP_NODELAY);
+use Socket6;
+use IO::Socket;
+use IO::Socket::INET6;
+use constant SO_BINDANY => 0x1000;
+
+sub new {
+ my $class = shift;
+ my %args = @_;
+ $args{logfile} ||= "client.log";
+ $args{up} ||= "Connected";
+ $args{down} ||= $args{alarm} ? "Alarm" :
+ "Shutdown|Broken pipe|Connection reset by peer";
+ my $self = Proc::new($class, %args);
+ $self->{protocol} ||= "tcp";
+ $self->{connectdomain}
+ or croak "$class connect domain not given";
+ $self->{connectaddr}
+ or croak "$class connect addr not given";
+ $self->{connectport}
+ or croak "$class connect port not given";
+
+ my $cs;
+ if ($self->{bindany}) {
+ $cs = IO::Socket::INET6->new(
+ Proto => $self->{protocol},
+ Domain => $self->{connectdomain},
+ Blocking => ($self->{nonblocking} ? 0 : 1),
+ ) or die ref($self), " socket connect failed: $!";
+ $cs->setsockopt(SOL_SOCKET, SO_BINDANY, 1)
+ or die ref($self), " setsockopt SO_BINDANY failed: $!";
+ my @rres = getaddrinfo($self->{bindaddr}, $self->{bindport}||0,
+ $self->{connectdomain}, SOCK_STREAM, 0, AI_PASSIVE);
+ $cs->bind($rres[3])
+ or die ref($self), " bind failed: $!";
+ } elsif ($self->{bindaddr} || $self->{bindport}) {
+ $cs = IO::Socket::INET6->new(
+ Proto => $self->{protocol},
+ Domain => $self->{connectdomain},
+ Blocking => ($self->{nonblocking} ? 0 : 1),
+ LocalAddr => $self->{bindaddr},
+ LocalPort => $self->{bindport},
+ ) or die ref($self), " socket connect failed: $!";
+ }
+ if ($cs) {
+ $self->{bindaddr} = $cs->sockhost();
+ $self->{bindport} = $cs->sockport();
+ $self->{cs} = $cs;
+ }
+
+ return $self;
+}
+
+sub child {
+ my $self = shift;
+
+ my $cs = $self->{cs} || IO::Socket::INET6->new(
+ Proto => $self->{protocol},
+ Domain => $self->{connectdomain},
+ Blocking => ($self->{nonblocking} ? 0 : 1),
+ ) or die ref($self), " socket connect failed: $!";
+ if ($self->{oobinline}) {
+ setsockopt($cs, SOL_SOCKET, SO_OOBINLINE, pack('i', 1))
+ or die ref($self), " set oobinline connect failed: $!";
+ }
+ if ($self->{sndbuf}) {
+ setsockopt($cs, SOL_SOCKET, SO_SNDBUF,
+ pack('i', $self->{sndbuf}))
+ or die ref($self), " set sndbuf connect failed: $!";
+ }
+ if ($self->{rcvbuf}) {
+ setsockopt($cs, SOL_SOCKET, SO_RCVBUF,
+ pack('i', $self->{rcvbuf}))
+ or die ref($self), " set rcvbuf connect failed: $!";
+ }
+ if ($self->{protocol} eq "tcp") {
+ setsockopt($cs, IPPROTO_TCP, TCP_NODELAY, pack('i', 1))
+ or die ref($self), " set nodelay connect failed: $!";
+ }
+ my @rres = getaddrinfo($self->{connectaddr}, $self->{connectport},
+ $self->{connectdomain}, SOCK_STREAM);
+ $cs->connect($rres[3])
+ or die ref($self), " connect failed: $!";
+ print STDERR "connect sock: ",$cs->sockhost()," ",$cs->sockport(),"\n";
+ print STDERR "connect peer: ",$cs->peerhost()," ",$cs->peerport(),"\n";
+ $self->{bindaddr} = $cs->sockhost();
+ $self->{bindport} = $cs->sockport();
+
+ open(STDOUT, '>&', $cs)
+ or die ref($self), " dup STDOUT failed: $!";
+ open(STDIN, '<&', $cs)
+ or die ref($self), " dup STDIN failed: $!";
+}
+
+1;