use Test::More; use strict; use warnings; use IO::Socket::INET; use File::Spec::Functions; use t::Util; use Net::Proxy; my @lines = ( "pete_peters sophie les mr_dork vijay\n", "d_fence abacus_remote ups_onlinet d2k_datamover2 ssslog_mgr\n", "Scooby_Doo Velma Freddy Daphne Shaggy\n", "STARTTLS\n", "Brian Florence Dougal Ermintrude Zebedee\n", "ale sherry cider wine whiskey\n", "Arcadio The_Witch_of_Kaan Drumm Rufferto Gravito\n", ); my $tests = @lines; my $all_tests = $tests + 1; plan tests => $all_tests; init_rand(@ARGV); # lock 2 ports my @free = find_free_ports(2); SKIP: { eval { require IO::Socket::SSL; }; skip 'IO::Socket::SSL required to test ssl', $all_tests if $@; skip 'Not enough available ports', $all_tests if @free < 2; no warnings 'once'; $IO::Socket::SSL::DEBUG = $ENV{NET_PROXY_VERBOSITY} || 0; my ( $proxy_port, $server_port ) = @free; my $pid = fork; skip 'proxy fork failed', $tests if !defined $pid; if ( $pid == 0 ) { my $proxy = Net::Proxy->new( { in => { type => 'ssl', port => $proxy_port, timeout => 1, start_cleartext => 1, SSL_cert_file => catfile( 't', 'test.cert' ), SSL_key_file => catfile( 't', 'test.key' ), hook => sub { my ( $dataref, $sock, $connector ) = @_; if ( $$dataref =~ s/^STARTTLS\n// ) { print $sock "OK\n"; $connector->upgrade_SSL($sock); } }, }, out => { type => 'tcp', host => 'localhost', port => $server_port, }, } ); $proxy->register(); Net::Proxy->set_verbosity( $ENV{NET_PROXY_VERBOSITY} || 0 ); Net::Proxy->mainloop(1); exit; } else { SKIP: { # wait for the proxy to set up sleep 1; # start a server my $listener = listen_on_port($server_port) or skip "Couldn't start the server: $!", $tests; # start a client my $client = connect_to_port($proxy_port) or skip "Couldn't start the client: $!", $tests; my $server = $listener->accept() or skip "Proxy didn't connect: $!", $tests; # remember which was the original client my $o_client = $client; # exchange the data for my $line (@lines) { ( $client, $server ) = random_swap( $client, $server ); if( $line eq "STARTTLS\n" ) { print $o_client $line; is( <$o_client>, "OK\n", "STARTTLS acknowledged" ); IO::Socket::SSL->start_SSL($o_client); } else { print $client $line; is( <$server>, $line, "Line received" ); } } $client->close(); is_closed( $server, 'peer' ); $server->close(); } } }