#!perl use 5.008001; use strict; use warnings; use Config; use File::Temp 'tempfile'; use Net::SMTP; use Test::More; my $debug = 0; # Net::SMTP Debug => .. my $parent = 0; plan skip_all => "no SSL support found in Net::SMTP" if ! Net::SMTP->can_ssl; plan skip_all => "fork not supported on this platform" unless $Config::Config{d_fork} || $Config::Config{d_pseudofork} || (($^O eq 'MSWin32' || $^O eq 'NetWare') and $Config::Config{useithreads} and $Config::Config{ccflags} =~ /-DPERL_IMPLICIT_SYS/); my $srv = IO::Socket::INET->new( LocalAddr => '127.0.0.1', Listen => 10 ); plan skip_all => "cannot create listener on localhost: $!" if ! $srv; my $saddr = $srv->sockhost.':'.$srv->sockport; plan tests => 2; require IO::Socket::SSL::Utils; my ($ca,$key) = IO::Socket::SSL::Utils::CERT_create( CA => 1 ); my ($fh,$cafile) = tempfile(); print $fh IO::Socket::SSL::Utils::PEM_cert2string($ca); close($fh); $parent = $$; END { unlink($cafile) if $$ == $parent } my ($cert) = IO::Socket::SSL::Utils::CERT_create( subject => { CN => 'smtp.example.com' }, issuer_cert => $ca, issuer_key => $key, key => $key ); test(1); # direct ssl test(0); # starttls sub test { my $ssl = shift; defined( my $pid = fork()) or die "fork failed: $!"; exit(smtp_server($ssl)) if ! $pid; smtp_client($ssl); wait; } sub smtp_client { my $ssl = shift; my %sslopt = ( SSL_verifycn_name => 'smtp.example.com', SSL_ca_file => $cafile ); $sslopt{SSL} = 1 if $ssl; my $cl = Net::SMTP->new($saddr, %sslopt, Debug => $debug); note("created Net::SMTP object"); if (!$cl) { fail( ($ssl ? "SSL ":"" )."SMTP connect failed"); } elsif ($ssl) { $cl->quit; pass("SSL SMTP connect success"); } elsif ( ! $cl->starttls ) { no warnings 'once'; fail("starttls failed: $IO::Socket::SSL::SSL_ERROR"); } else { $cl->quit; pass("starttls success"); } } sub smtp_server { my $ssl = shift; my $cl = $srv->accept or die "accept failed: $!"; my %sslargs = ( SSL_server => 1, SSL_cert => $cert, SSL_key => $key, ); if ( $ssl ) { if ( ! IO::Socket::SSL->start_SSL($cl, %sslargs)) { diag("initial ssl handshake with client failed"); return; } } print $cl "220 welcome\r\n"; while (<$cl>) { my ($cmd,$arg) = m{^(\S+)(?: +(.*))?\r\n} or die $_; $cmd = uc($cmd); if ($cmd eq 'QUIT' ) { print $cl "250 bye\r\n"; last; } elsif ( $cmd eq 'HELO' ) { print $cl "250 localhost\r\n"; } elsif ( $cmd eq 'EHLO' ) { print $cl "250-localhost\r\n". ( $ssl ? "" : "250-STARTTLS\r\n" ). "250 HELP\r\n"; } elsif ( ! $ssl and $cmd eq 'STARTTLS' ) { print $cl "250 starting ssl\r\n"; if ( ! IO::Socket::SSL->start_SSL($cl, %sslargs)) { diag("initial ssl handshake with client failed"); return; } $ssl = 1; } else { diag("received unknown command: $cmd"); print "500 unknown cmd\r\n"; } } note("SMTP dialog done"); }