Spaces:
Running
Running
# Net::SMTP.pm | |
# | |
# Copyright (C) 1995-2004 Graham Barr. All rights reserved. | |
# Copyright (C) 2013-2016, 2020 Steve Hay. All rights reserved. | |
# This module is free software; you can redistribute it and/or modify it under | |
# the same terms as Perl itself, i.e. under the terms of either the GNU General | |
# Public License or the Artistic License, as specified in the F<LICENCE> file. | |
package Net::SMTP; | |
use 5.008001; | |
use strict; | |
use warnings; | |
use Carp; | |
use IO::Socket; | |
use Net::Cmd; | |
use Net::Config; | |
use Socket; | |
our $VERSION = "3.13"; | |
# Code for detecting if we can use SSL | |
my $ssl_class = eval { | |
require IO::Socket::SSL; | |
# first version with default CA on most platforms | |
no warnings 'numeric'; | |
IO::Socket::SSL->VERSION(2.007); | |
} && 'IO::Socket::SSL'; | |
my $nossl_warn = !$ssl_class && | |
'To use SSL please install IO::Socket::SSL with version>=2.007'; | |
# Code for detecting if we can use IPv6 | |
my $family_key = 'Domain'; | |
my $inet6_class = eval { | |
require IO::Socket::IP; | |
no warnings 'numeric'; | |
IO::Socket::IP->VERSION(0.25) || die; | |
$family_key = 'Family'; | |
} && 'IO::Socket::IP' || eval { | |
require IO::Socket::INET6; | |
no warnings 'numeric'; | |
IO::Socket::INET6->VERSION(2.62); | |
} && 'IO::Socket::INET6'; | |
sub can_ssl { $ssl_class }; | |
sub can_inet6 { $inet6_class }; | |
our @ISA = ('Net::Cmd', $inet6_class || 'IO::Socket::INET'); | |
sub new { | |
my $self = shift; | |
my $type = ref($self) || $self; | |
my ($host, %arg); | |
if (@_ % 2) { | |
$host = shift; | |
%arg = @_; | |
} | |
else { | |
%arg = @_; | |
$host = delete $arg{Host}; | |
} | |
if ($arg{SSL}) { | |
# SSL from start | |
die $nossl_warn if !$ssl_class; | |
$arg{Port} ||= 465; | |
} | |
my $hosts = defined $host ? $host : $NetConfig{smtp_hosts}; | |
my $obj; | |
$arg{Timeout} = 120 if ! defined $arg{Timeout}; | |
foreach my $h (@{ref($hosts) ? $hosts : [$hosts]}) { | |
$obj = $type->SUPER::new( | |
PeerAddr => ($host = $h), | |
PeerPort => $arg{Port} || 'smtp(25)', | |
LocalAddr => $arg{LocalAddr}, | |
LocalPort => $arg{LocalPort}, | |
$family_key => $arg{Domain} || $arg{Family}, | |
Proto => 'tcp', | |
Timeout => $arg{Timeout} | |
) | |
and last; | |
} | |
return | |
unless defined $obj; | |
${*$obj}{'net_smtp_arg'} = \%arg; | |
${*$obj}{'net_smtp_host'} = $host; | |
if ($arg{SSL}) { | |
Net::SMTP::_SSL->start_SSL($obj,%arg) | |
or return; | |
} | |
$obj->autoflush(1); | |
$obj->debug(exists $arg{Debug} ? $arg{Debug} : undef); | |
unless ($obj->response() == CMD_OK) { | |
my $err = ref($obj) . ": " . $obj->code . " " . $obj->message; | |
$obj->close(); | |
$@ = $err; | |
return; | |
} | |
${*$obj}{'net_smtp_exact_addr'} = $arg{ExactAddresses}; | |
(${*$obj}{'net_smtp_banner'}) = $obj->message; | |
(${*$obj}{'net_smtp_domain'}) = $obj->message =~ /\A\s*(\S+)/; | |
if (!exists $arg{SendHello} || $arg{SendHello}) { | |
unless ($obj->hello($arg{Hello} || "")) { | |
my $err = ref($obj) . ": " . $obj->code . " " . $obj->message; | |
$obj->close(); | |
$@ = $err; | |
return; | |
} | |
} | |
$obj; | |
} | |
sub host { | |
my $me = shift; | |
${*$me}{'net_smtp_host'}; | |
} | |
## | |
## User interface methods | |
## | |
sub banner { | |
my $me = shift; | |
return ${*$me}{'net_smtp_banner'} || undef; | |
} | |
sub domain { | |
my $me = shift; | |
return ${*$me}{'net_smtp_domain'} || undef; | |
} | |
sub etrn { | |
my $self = shift; | |
defined($self->supports('ETRN', 500, ["Command unknown: 'ETRN'"])) | |
&& $self->_ETRN(@_); | |
} | |
sub auth { | |
my ($self, $username, $password) = @_; | |
eval { | |
require MIME::Base64; | |
require Authen::SASL; | |
} or $self->set_status(500, ["Need MIME::Base64 and Authen::SASL todo auth"]), return 0; | |
my $mechanisms = $self->supports('AUTH', 500, ["Command unknown: 'AUTH'"]); | |
return unless defined $mechanisms; | |
my $sasl; | |
if (ref($username) and UNIVERSAL::isa($username, 'Authen::SASL')) { | |
$sasl = $username; | |
my $requested_mechanisms = $sasl->mechanism(); | |
if (! defined($requested_mechanisms) || $requested_mechanisms eq '') { | |
$sasl->mechanism($mechanisms); | |
} | |
} | |
else { | |
die "auth(username, password)" if not length $username; | |
$sasl = Authen::SASL->new( | |
mechanism => $mechanisms, | |
callback => { | |
user => $username, | |
pass => $password, | |
authname => $username, | |
}, | |
debug => $self->debug | |
); | |
} | |
my $client; | |
my $str; | |
do { | |
if ($client) { | |
# $client mechanism failed, so we need to exclude this mechanism from list | |
my $failed_mechanism = $client->mechanism; | |
return unless defined $failed_mechanism; | |
$self->debug_text("Auth mechanism failed: $failed_mechanism") | |
if $self->debug; | |
$mechanisms =~ s/\b\Q$failed_mechanism\E\b//; | |
return unless $mechanisms =~ /\S/; | |
$sasl->mechanism($mechanisms); | |
} | |
# We should probably allow the user to pass the host, but I don't | |
# currently know and SASL mechanisms that are used by smtp that need it | |
$client = $sasl->client_new('smtp', ${*$self}{'net_smtp_host'}, 0); | |
$str = $client->client_start; | |
} while (!defined $str); | |
# We don't support sasl mechanisms that encrypt the socket traffic. | |
# todo that we would really need to change the ISA hierarchy | |
# so we don't inherit from IO::Socket, but instead hold it in an attribute | |
my @cmd = ("AUTH", $client->mechanism); | |
my $code; | |
push @cmd, MIME::Base64::encode_base64($str, '') | |
if defined $str and length $str; | |
while (($code = $self->command(@cmd)->response()) == CMD_MORE) { | |
my $str2 = MIME::Base64::decode_base64(($self->message)[0]); | |
$self->debug_print(0, "(decoded) " . $str2 . "\n") if $self->debug; | |
$str = $client->client_step($str2); | |
@cmd = ( | |
MIME::Base64::encode_base64($str, '') | |
); | |
$self->debug_print(1, "(decoded) " . $str . "\n") if $self->debug; | |
} | |
$code == CMD_OK; | |
} | |
sub hello { | |
my $me = shift; | |
my $domain = shift || "localhost.localdomain"; | |
my $ok = $me->_EHLO($domain); | |
my @msg = $me->message; | |
if ($ok) { | |
my $h = ${*$me}{'net_smtp_esmtp'} = {}; | |
foreach my $ln (@msg) { | |
$h->{uc $1} = $2 | |
if $ln =~ /([-\w]+)\b[= \t]*([^\n]*)/; | |
} | |
} | |
elsif ($me->status == CMD_ERROR) { | |
@msg = $me->message | |
if $ok = $me->_HELO($domain); | |
} | |
return unless $ok; | |
${*$me}{net_smtp_hello_domain} = $domain; | |
$msg[0] =~ /\A\s*(\S+)/; | |
return ($1 || " "); | |
} | |
sub starttls { | |
my $self = shift; | |
$ssl_class or die $nossl_warn; | |
$self->_STARTTLS or return; | |
Net::SMTP::_SSL->start_SSL($self, | |
%{ ${*$self}{'net_smtp_arg'} }, # (ssl) args given in new | |
@_ # more (ssl) args | |
) or return; | |
# another hello after starttls to read new ESMTP capabilities | |
return $self->hello(${*$self}{net_smtp_hello_domain}); | |
} | |
sub supports { | |
my $self = shift; | |
my $cmd = uc shift; | |
return ${*$self}{'net_smtp_esmtp'}->{$cmd} | |
if exists ${*$self}{'net_smtp_esmtp'}->{$cmd}; | |
$self->set_status(@_) | |
if @_; | |
return; | |
} | |
sub _addr { | |
my $self = shift; | |
my $addr = shift; | |
$addr = "" unless defined $addr; | |
if (${*$self}{'net_smtp_exact_addr'}) { | |
return $1 if $addr =~ /^\s*(<.*>)\s*$/s; | |
} | |
else { | |
return $1 if $addr =~ /(<[^>]*>)/; | |
$addr =~ s/^\s+|\s+$//sg; | |
} | |
"<$addr>"; | |
} | |
sub mail { | |
my $me = shift; | |
my $addr = _addr($me, shift); | |
my $opts = ""; | |
if (@_) { | |
my %opt = @_; | |
my ($k, $v); | |
if (exists ${*$me}{'net_smtp_esmtp'}) { | |
my $esmtp = ${*$me}{'net_smtp_esmtp'}; | |
if (defined($v = delete $opt{Size})) { | |
if (exists $esmtp->{SIZE}) { | |
$opts .= sprintf " SIZE=%d", $v + 0; | |
} | |
else { | |
carp 'Net::SMTP::mail: SIZE option not supported by host'; | |
} | |
} | |
if (defined($v = delete $opt{Return})) { | |
if (exists $esmtp->{DSN}) { | |
$opts .= " RET=" . ((uc($v) eq "FULL") ? "FULL" : "HDRS"); | |
} | |
else { | |
carp 'Net::SMTP::mail: DSN option not supported by host'; | |
} | |
} | |
if (defined($v = delete $opt{Bits})) { | |
if ($v eq "8") { | |
if (exists $esmtp->{'8BITMIME'}) { | |
$opts .= " BODY=8BITMIME"; | |
} | |
else { | |
carp 'Net::SMTP::mail: 8BITMIME option not supported by host'; | |
} | |
} | |
elsif ($v eq "binary") { | |
if (exists $esmtp->{'BINARYMIME'} && exists $esmtp->{'CHUNKING'}) { | |
$opts .= " BODY=BINARYMIME"; | |
${*$me}{'net_smtp_chunking'} = 1; | |
} | |
else { | |
carp 'Net::SMTP::mail: BINARYMIME option not supported by host'; | |
} | |
} | |
elsif (exists $esmtp->{'8BITMIME'} or exists $esmtp->{'BINARYMIME'}) { | |
$opts .= " BODY=7BIT"; | |
} | |
else { | |
carp 'Net::SMTP::mail: 8BITMIME and BINARYMIME options not supported by host'; | |
} | |
} | |
if (defined($v = delete $opt{Transaction})) { | |
if (exists $esmtp->{CHECKPOINT}) { | |
$opts .= " TRANSID=" . _addr($me, $v); | |
} | |
else { | |
carp 'Net::SMTP::mail: CHECKPOINT option not supported by host'; | |
} | |
} | |
if (defined($v = delete $opt{Envelope})) { | |
if (exists $esmtp->{DSN}) { | |
$v =~ s/([^\041-\176]|=|\+)/sprintf "+%02X", ord($1)/sge; | |
$opts .= " ENVID=$v"; | |
} | |
else { | |
carp 'Net::SMTP::mail: DSN option not supported by host'; | |
} | |
} | |
if (defined($v = delete $opt{ENVID})) { | |
# expected to be in a format as required by RFC 3461, xtext-encoded | |
if (exists $esmtp->{DSN}) { | |
$opts .= " ENVID=$v"; | |
} | |
else { | |
carp 'Net::SMTP::mail: DSN option not supported by host'; | |
} | |
} | |
if (defined($v = delete $opt{AUTH})) { | |
# expected to be in a format as required by RFC 2554, | |
# rfc2821-quoted and xtext-encoded, or <> | |
if (exists $esmtp->{AUTH}) { | |
$v = '<>' if !defined($v) || $v eq ''; | |
$opts .= " AUTH=$v"; | |
} | |
else { | |
carp 'Net::SMTP::mail: AUTH option not supported by host'; | |
} | |
} | |
if (defined($v = delete $opt{XVERP})) { | |
if (exists $esmtp->{'XVERP'}) { | |
$opts .= " XVERP"; | |
} | |
else { | |
carp 'Net::SMTP::mail: XVERP option not supported by host'; | |
} | |
} | |
carp 'Net::SMTP::recipient: unknown option(s) ' . join(" ", keys %opt) . ' - ignored' | |
if scalar keys %opt; | |
} | |
else { | |
carp 'Net::SMTP::mail: ESMTP not supported by host - options discarded :-('; | |
} | |
} | |
$me->_MAIL("FROM:" . $addr . $opts); | |
} | |
sub send { my $me = shift; $me->_SEND("FROM:" . _addr($me, $_[0])) } | |
sub send_or_mail { my $me = shift; $me->_SOML("FROM:" . _addr($me, $_[0])) } | |
sub send_and_mail { my $me = shift; $me->_SAML("FROM:" . _addr($me, $_[0])) } | |
sub reset { | |
my $me = shift; | |
$me->dataend() | |
if (exists ${*$me}{'net_smtp_lastch'}); | |
$me->_RSET(); | |
} | |
sub recipient { | |
my $smtp = shift; | |
my $opts = ""; | |
my $skip_bad = 0; | |
if (@_ && ref($_[-1])) { | |
my %opt = %{pop(@_)}; | |
my $v; | |
$skip_bad = delete $opt{'SkipBad'}; | |
if (exists ${*$smtp}{'net_smtp_esmtp'}) { | |
my $esmtp = ${*$smtp}{'net_smtp_esmtp'}; | |
if (defined($v = delete $opt{Notify})) { | |
if (exists $esmtp->{DSN}) { | |
$opts .= " NOTIFY=" . join(",", map { uc $_ } @$v); | |
} | |
else { | |
carp 'Net::SMTP::recipient: DSN option not supported by host'; | |
} | |
} | |
if (defined($v = delete $opt{ORcpt})) { | |
if (exists $esmtp->{DSN}) { | |
$opts .= " ORCPT=" . $v; | |
} | |
else { | |
carp 'Net::SMTP::recipient: DSN option not supported by host'; | |
} | |
} | |
carp 'Net::SMTP::recipient: unknown option(s) ' . join(" ", keys %opt) . ' - ignored' | |
if scalar keys %opt; | |
} | |
elsif (%opt) { | |
carp 'Net::SMTP::recipient: ESMTP not supported by host - options discarded :-('; | |
} | |
} | |
my @ok; | |
foreach my $addr (@_) { | |
if ($smtp->_RCPT("TO:" . _addr($smtp, $addr) . $opts)) { | |
push(@ok, $addr) if $skip_bad; | |
} | |
elsif (!$skip_bad) { | |
return 0; | |
} | |
} | |
return $skip_bad ? @ok : 1; | |
} | |
BEGIN { | |
*to = \&recipient; | |
*cc = \&recipient; | |
*bcc = \&recipient; | |
} | |
sub data { | |
my $me = shift; | |
if (exists ${*$me}{'net_smtp_chunking'}) { | |
carp 'Net::SMTP::data: CHUNKING extension in use, must call bdat instead'; | |
} | |
else { | |
my $ok = $me->_DATA() && $me->datasend(@_); | |
$ok && @_ | |
? $me->dataend | |
: $ok; | |
} | |
} | |
sub bdat { | |
my $me = shift; | |
if (exists ${*$me}{'net_smtp_chunking'}) { | |
my $data = shift; | |
$me->_BDAT(length $data) | |
&& $me->rawdatasend($data) | |
&& $me->response() == CMD_OK; | |
} | |
else { | |
carp 'Net::SMTP::bdat: CHUNKING extension is not in use, call data instead'; | |
} | |
} | |
sub bdatlast { | |
my $me = shift; | |
if (exists ${*$me}{'net_smtp_chunking'}) { | |
my $data = shift; | |
$me->_BDAT(length $data, "LAST") | |
&& $me->rawdatasend($data) | |
&& $me->response() == CMD_OK; | |
} | |
else { | |
carp 'Net::SMTP::bdat: CHUNKING extension is not in use, call data instead'; | |
} | |
} | |
sub datafh { | |
my $me = shift; | |
return unless $me->_DATA(); | |
return $me->tied_fh; | |
} | |
sub expand { | |
my $me = shift; | |
$me->_EXPN(@_) | |
? ($me->message) | |
: (); | |
} | |
sub verify { shift->_VRFY(@_) } | |
sub help { | |
my $me = shift; | |
$me->_HELP(@_) | |
? scalar $me->message | |
: undef; | |
} | |
sub quit { | |
my $me = shift; | |
$me->_QUIT; | |
$me->close; | |
} | |
sub DESTROY { | |
# ignore | |
} | |
## | |
## RFC821 commands | |
## | |
sub _EHLO { shift->command("EHLO", @_)->response() == CMD_OK } | |
sub _HELO { shift->command("HELO", @_)->response() == CMD_OK } | |
sub _MAIL { shift->command("MAIL", @_)->response() == CMD_OK } | |
sub _RCPT { shift->command("RCPT", @_)->response() == CMD_OK } | |
sub _SEND { shift->command("SEND", @_)->response() == CMD_OK } | |
sub _SAML { shift->command("SAML", @_)->response() == CMD_OK } | |
sub _SOML { shift->command("SOML", @_)->response() == CMD_OK } | |
sub _VRFY { shift->command("VRFY", @_)->response() == CMD_OK } | |
sub _EXPN { shift->command("EXPN", @_)->response() == CMD_OK } | |
sub _HELP { shift->command("HELP", @_)->response() == CMD_OK } | |
sub _RSET { shift->command("RSET")->response() == CMD_OK } | |
sub _NOOP { shift->command("NOOP")->response() == CMD_OK } | |
sub _QUIT { shift->command("QUIT")->response() == CMD_OK } | |
sub _DATA { shift->command("DATA")->response() == CMD_MORE } | |
sub _BDAT { shift->command("BDAT", @_) } | |
sub _TURN { shift->unsupported(@_); } | |
sub _ETRN { shift->command("ETRN", @_)->response() == CMD_OK } | |
sub _AUTH { shift->command("AUTH", @_)->response() == CMD_OK } | |
sub _STARTTLS { shift->command("STARTTLS")->response() == CMD_OK } | |
{ | |
package Net::SMTP::_SSL; | |
our @ISA = ( $ssl_class ? ($ssl_class):(), 'Net::SMTP' ); | |
sub starttls { die "SMTP connection is already in SSL mode" } | |
sub start_SSL { | |
my ($class,$smtp,%arg) = @_; | |
delete @arg{ grep { !m{^SSL_} } keys %arg }; | |
( $arg{SSL_verifycn_name} ||= $smtp->host ) | |
=~s{(?<!:):[\w()]+$}{}; # strip port | |
$arg{SSL_hostname} = $arg{SSL_verifycn_name} | |
if ! defined $arg{SSL_hostname} && $class->can_client_sni; | |
$arg{SSL_verifycn_scheme} ||= 'smtp'; | |
my $ok = $class->SUPER::start_SSL($smtp,%arg); | |
$@ = $ssl_class->errstr if !$ok; | |
return $ok; | |
} | |
} | |
1; | |
__END__ | |
=head1 NAME | |
Net::SMTP - Simple Mail Transfer Protocol Client | |
=head1 SYNOPSIS | |
use Net::SMTP; | |
# Constructors | |
$smtp = Net::SMTP->new('mailhost'); | |
$smtp = Net::SMTP->new('mailhost', Timeout => 60); | |
=head1 DESCRIPTION | |
This module implements a client interface to the SMTP and ESMTP | |
protocol, enabling a perl5 application to talk to SMTP servers. This | |
documentation assumes that you are familiar with the concepts of the | |
SMTP protocol described in RFC2821. | |
With L<IO::Socket::SSL> installed it also provides support for implicit and | |
explicit TLS encryption, i.e. SMTPS or SMTP+STARTTLS. | |
The Net::SMTP class is a subclass of Net::Cmd and (depending on avaibility) of | |
IO::Socket::IP, IO::Socket::INET6 or IO::Socket::INET. | |
=head2 Class Methods | |
=over 4 | |
=item C<new([$host][, %options])> | |
This is the constructor for a new Net::SMTP object. C<$host> is the | |
name of the remote host to which an SMTP connection is required. | |
On failure C<undef> will be returned and C<$@> will contain the reason | |
for the failure. | |
C<$host> is optional. If C<$host> is not given then it may instead be | |
passed as the C<Host> option described below. If neither is given then | |
the C<SMTP_Hosts> specified in C<Net::Config> will be used. | |
C<%options> are passed in a hash like fashion, using key and value pairs. | |
Possible options are: | |
B<Hello> - SMTP requires that you identify yourself. This option | |
specifies a string to pass as your mail domain. If not given localhost.localdomain | |
will be used. | |
B<SendHello> - If false then the EHLO (or HELO) command that is normally sent | |
when constructing the object will not be sent. In that case the command will | |
have to be sent manually by calling C<hello()> instead. | |
B<Host> - SMTP host to connect to. It may be a single scalar (hostname[:port]), | |
as defined for the C<PeerAddr> option in L<IO::Socket::INET>, or a reference to | |
an array with hosts to try in turn. The L</host> method will return the value | |
which was used to connect to the host. | |
Format - C<PeerHost> from L<IO::Socket::INET> new method. | |
B<Port> - port to connect to. | |
Default - 25 for plain SMTP and 465 for immediate SSL. | |
B<SSL> - If the connection should be done from start with SSL, contrary to later | |
upgrade with C<starttls>. | |
You can use SSL arguments as documented in L<IO::Socket::SSL>, but it will | |
usually use the right arguments already. | |
B<LocalAddr> and B<LocalPort> - These parameters are passed directly | |
to IO::Socket to allow binding the socket to a specific local address and port. | |
B<Domain> - This parameter is passed directly to IO::Socket and makes it | |
possible to enforce IPv4 connections even if L<IO::Socket::IP> is used as super | |
class. Alternatively B<Family> can be used. | |
B<Timeout> - Maximum time, in seconds, to wait for a response from the | |
SMTP server (default: 120) | |
B<ExactAddresses> - If true then all C<$address> arguments must be as | |
defined by C<addr-spec> in RFC2822. If not given, or false, then | |
Net::SMTP will attempt to extract the address from the value passed. | |
B<Debug> - Enable debugging information | |
Example: | |
$smtp = Net::SMTP->new('mailhost', | |
Hello => 'my.mail.domain', | |
Timeout => 30, | |
Debug => 1, | |
); | |
# the same | |
$smtp = Net::SMTP->new( | |
Host => 'mailhost', | |
Hello => 'my.mail.domain', | |
Timeout => 30, | |
Debug => 1, | |
); | |
# the same with direct SSL | |
$smtp = Net::SMTP->new('mailhost', | |
Hello => 'my.mail.domain', | |
Timeout => 30, | |
Debug => 1, | |
SSL => 1, | |
); | |
# Connect to the default server from Net::config | |
$smtp = Net::SMTP->new( | |
Hello => 'my.mail.domain', | |
Timeout => 30, | |
); | |
=back | |
=head1 Object Methods | |
Unless otherwise stated all methods return either a I<true> or I<false> | |
value, with I<true> meaning that the operation was a success. When a method | |
states that it returns a value, failure will be returned as I<undef> or an | |
empty list. | |
C<Net::SMTP> inherits from C<Net::Cmd> so methods defined in C<Net::Cmd> may | |
be used to send commands to the remote SMTP server in addition to the methods | |
documented here. | |
=over 4 | |
=item C<banner()> | |
Returns the banner message which the server replied with when the | |
initial connection was made. | |
=item C<domain()> | |
Returns the domain that the remote SMTP server identified itself as during | |
connection. | |
=item C<hello($domain)> | |
Tell the remote server the mail domain which you are in using the EHLO | |
command (or HELO if EHLO fails). Since this method is invoked | |
automatically when the Net::SMTP object is constructed the user should | |
normally not have to call it manually. | |
=item C<host()> | |
Returns the value used by the constructor, and passed to IO::Socket::INET, | |
to connect to the host. | |
=item C<etrn($domain)> | |
Request a queue run for the C<$domain> given. | |
=item C<starttls(%sslargs)> | |
Upgrade existing plain connection to SSL. | |
You can use SSL arguments as documented in L<IO::Socket::SSL>, but it will | |
usually use the right arguments already. | |
=item C<auth($username, $password)> | |
=item C<auth($sasl)> | |
Attempt SASL authentication. Requires Authen::SASL module. The first form | |
constructs a new Authen::SASL object using the given username and password; | |
the second form uses the given Authen::SASL object. | |
=item C<mail($address[, %options])> | |
=item C<send($address)> | |
=item C<send_or_mail($address)> | |
=item C<send_and_mail($address)> | |
Send the appropriate command to the server MAIL, SEND, SOML or SAML. C<$address> | |
is the address of the sender. This initiates the sending of a message. The | |
method C<recipient> should be called for each address that the message is to | |
be sent to. | |
The C<mail> method can take some additional ESMTP C<%options> which is passed | |
in hash like fashion, using key and value pairs. Possible options are: | |
Size => <bytes> | |
Return => "FULL" | "HDRS" | |
Bits => "7" | "8" | "binary" | |
Transaction => <ADDRESS> | |
Envelope => <ENVID> # xtext-encodes its argument | |
ENVID => <ENVID> # similar to Envelope, but expects argument encoded | |
XVERP => 1 | |
AUTH => <submitter> # encoded address according to RFC 2554 | |
The C<Return> and C<Envelope> parameters are used for DSN (Delivery | |
Status Notification). | |
The submitter address in C<AUTH> option is expected to be in a format as | |
required by RFC 2554, in an RFC2821-quoted form and xtext-encoded, or <> . | |
=item C<reset()> | |
Reset the status of the server. This may be called after a message has been | |
initiated, but before any data has been sent, to cancel the sending of the | |
message. | |
=item C<recipient($address[, $address[, ...]][, %options])> | |
Notify the server that the current message should be sent to all of the | |
addresses given. Each address is sent as a separate command to the server. | |
Should the sending of any address result in a failure then the process is | |
aborted and a I<false> value is returned. It is up to the user to call | |
C<reset> if they so desire. | |
The C<recipient> method can also pass additional case-sensitive C<%options> as an | |
anonymous hash using key and value pairs. Possible options are: | |
Notify => ['NEVER'] or ['SUCCESS','FAILURE','DELAY'] (see below) | |
ORcpt => <ORCPT> | |
SkipBad => 1 (to ignore bad addresses) | |
If C<SkipBad> is true the C<recipient> will not return an error when a bad | |
address is encountered and it will return an array of addresses that did | |
succeed. | |
$smtp->recipient($recipient1,$recipient2); # Good | |
$smtp->recipient($recipient1,$recipient2, { SkipBad => 1 }); # Good | |
$smtp->recipient($recipient1,$recipient2, { Notify => ['FAILURE','DELAY'], SkipBad => 1 }); # Good | |
@goodrecips=$smtp->recipient(@recipients, { Notify => ['FAILURE'], SkipBad => 1 }); # Good | |
$smtp->recipient("$recipient,$recipient2"); # BAD | |
Notify is used to request Delivery Status Notifications (DSNs), but your | |
SMTP/ESMTP service may not respect this request depending upon its version and | |
your site's SMTP configuration. | |
Leaving out the Notify option usually defaults an SMTP service to its default | |
behavior equivalent to ['FAILURE'] notifications only, but again this may be | |
dependent upon your site's SMTP configuration. | |
The NEVER keyword must appear by itself if used within the Notify option and "requests | |
that a DSN not be returned to the sender under any conditions." | |
{Notify => ['NEVER']} | |
$smtp->recipient(@recipients, { Notify => ['NEVER'], SkipBad => 1 }); # Good | |
You may use any combination of these three values 'SUCCESS','FAILURE','DELAY' in | |
the anonymous array reference as defined by RFC3461 (see | |
L<https://www.ietf.org/rfc/rfc3461.txt> for more information. Note: quotations | |
in this topic from same.). | |
A Notify parameter of 'SUCCESS' or 'FAILURE' "requests that a DSN be issued on | |
successful delivery or delivery failure, respectively." | |
A Notify parameter of 'DELAY' "indicates the sender's willingness to receive | |
delayed DSNs. Delayed DSNs may be issued if delivery of a message has been | |
delayed for an unusual amount of time (as determined by the Message Transfer | |
Agent (MTA) at which the message is delayed), but the final delivery status | |
(whether successful or failure) cannot be determined. The absence of the DELAY | |
keyword in a NOTIFY parameter requests that a "delayed" DSN NOT be issued under | |
any conditions." | |
{Notify => ['SUCCESS','FAILURE','DELAY']} | |
$smtp->recipient(@recipients, { Notify => ['FAILURE','DELAY'], SkipBad => 1 }); # Good | |
ORcpt is also part of the SMTP DSN extension according to RFC3461. | |
It is used to pass along the original recipient that the mail was first | |
sent to. The machine that generates a DSN will use this address to inform | |
the sender, because he can't know if recipients get rewritten by mail servers. | |
It is expected to be in a format as required by RFC3461, xtext-encoded. | |
=item C<to($address[, $address[, ...]])> | |
=item C<cc($address[, $address[, ...]])> | |
=item C<bcc($address[, $address[, ...]])> | |
Synonyms for C<recipient>. | |
=item C<data([$data])> | |
Initiate the sending of the data from the current message. | |
C<$data> may be a reference to a list or a list and must be encoded by the | |
caller to octets of whatever encoding is required, e.g. by using the Encode | |
module's C<encode()> function. | |
If specified the contents of C<$data> and a termination string C<".\r\n"> is | |
sent to the server. The result will be true if the data was accepted. | |
If C<$data> is not specified then the result will indicate that the server | |
wishes the data to be sent. The data must then be sent using the C<datasend> | |
and C<dataend> methods described in L<Net::Cmd>. | |
=item C<bdat($data)> | |
=item C<bdatlast($data)> | |
Use the alternate C<$data> command "BDAT" of the data chunking service extension | |
defined in RFC1830 for efficiently sending large MIME messages. | |
=item C<expand($address)> | |
Request the server to expand the given address Returns an array | |
which contains the text read from the server. | |
=item C<verify($address)> | |
Verify that C<$address> is a legitimate mailing address. | |
Most sites usually disable this feature in their SMTP service configuration. | |
Use "Debug => 1" option under new() to see if disabled. | |
=item C<help([$subject])> | |
Request help text from the server. Returns the text or undef upon failure | |
=item C<quit()> | |
Send the QUIT command to the remote SMTP server and close the socket connection. | |
=item C<can_inet6()> | |
Returns whether we can use IPv6. | |
=item C<can_ssl()> | |
Returns whether we can use SSL. | |
=back | |
=head2 Addresses | |
Net::SMTP attempts to DWIM with addresses that are passed. For | |
example an application might extract The From: line from an email | |
and pass that to mail(). While this may work, it is not recommended. | |
The application should really use a module like L<Mail::Address> | |
to extract the mail address and pass that. | |
If C<ExactAddresses> is passed to the constructor, then addresses | |
should be a valid rfc2821-quoted address, although Net::SMTP will | |
accept the address surrounded by angle brackets. | |
funny user@domain WRONG | |
"funny user"@domain RIGHT, recommended | |
<"funny user"@domain> OK | |
=head1 EXAMPLES | |
This example prints the mail domain name of the SMTP server known as mailhost: | |
#!/usr/local/bin/perl -w | |
use Net::SMTP; | |
$smtp = Net::SMTP->new('mailhost'); | |
print $smtp->domain,"\n"; | |
$smtp->quit; | |
This example sends a small message to the postmaster at the SMTP server | |
known as mailhost: | |
#!/usr/local/bin/perl -w | |
use Net::SMTP; | |
my $smtp = Net::SMTP->new('mailhost'); | |
$smtp->mail($ENV{USER}); | |
if ($smtp->to('postmaster')) { | |
$smtp->data(); | |
$smtp->datasend("To: postmaster\n"); | |
$smtp->datasend("\n"); | |
$smtp->datasend("A simple test message\n"); | |
$smtp->dataend(); | |
} else { | |
print "Error: ", $smtp->message(); | |
} | |
$smtp->quit; | |
=head1 EXPORTS | |
I<None>. | |
=head1 KNOWN BUGS | |
See L<https://rt.cpan.org/Dist/Display.html?Status=Active&Queue=libnet>. | |
=head1 SEE ALSO | |
L<Net::Cmd>, | |
L<IO::Socket::SSL>. | |
=head1 AUTHOR | |
Graham Barr E<lt>L<[email protected]|mailto:[email protected]>E<gt>. | |
Steve Hay E<lt>L<[email protected]|mailto:[email protected]>E<gt> is now maintaining | |
libnet as of version 1.22_02. | |
=head1 COPYRIGHT | |
Copyright (C) 1995-2004 Graham Barr. All rights reserved. | |
Copyright (C) 2013-2016, 2020 Steve Hay. All rights reserved. | |
=head1 LICENCE | |
This module is free software; you can redistribute it and/or modify it under the | |
same terms as Perl itself, i.e. under the terms of either the GNU General Public | |
License or the Artistic License, as specified in the F<LICENCE> file. | |
=head1 VERSION | |
Version 3.13 | |
=head1 DATE | |
23 Dec 2020 | |
=head1 HISTORY | |
See the F<Changes> file. | |
=cut | |