# Copyright 2002-2012 Interchange Development Group and others # # This program is free software; you can redistribute it and/or modify # it under the terms of the GNU General Public License as published by # the Free Software Foundation; either version 2 of the License, or # (at your option) any later version. See the LICENSE file for details. UserTag email Order to subject reply from extra UserTag email hasEndTag UserTag email addAttr UserTag email Interpolate UserTag email Routine <{MailOrderTo}; $from =~ s/,.*//; } # Use local copy to avoid mangling with caller's data $cc = $opt->{cc}; $bcc = $opt->{bcc}; # See if UTF-8 support is required $utf8 = $::Variable->{MV_UTF8} || $Global::Variable->{MV_UTF8}; # Prevent header injections from spammers' hostile content for ($to, $subject, $reply, $from, $cc, $bcc) { # unfold valid RFC 2822 "2.2.3. Long Header Fields" s/\r?\n([ \t]+)/$1/g; # now remove any invalid extra lines left over s/[\r\n](.*)//s and ::logError("Header injection attempted in email tag: %s", $1); } for (grep /\S/, split /[\r\n]+/, $extra) { # require header conformance with RFC 2822 section 2.2 push (@extra, $_), next if /^[\x21-\x39\x3b-\x7e]+:[\x00-\x09\x0b\x0c\x0e-\x7f]+$/; ::logError("Invalid header given to email tag: %s", $_); } unshift @extra, "From: $from" if $from; # force utf8 email through MIME as attachment unless (($opt->{attach} || $opt->{html}) && $utf8){ $opt->{body_mime} = $opt->{mimetype}; $body = utf8_to_other($body, 'utf8'); } my $sent_with_attach = 0; ATTACH: { #::logDebug("Checking for attachment"); last ATTACH unless $opt->{attach} || $opt->{html}; unless ($Have_mime_lite) { ::logError("email tag: attachment without MIME::Lite installed."); last ATTACH; } my $att1_format; my $att = $opt->{attach}; my @attach; my @extra_headers; # encode values if utf8 is supported if($utf8){ $to = utf8_to_other($to, 'MIME-Header'); $from = utf8_to_other($from, 'MIME-Header'); $subject = utf8_to_other($subject, 'MIME-Header'); $cc = utf8_to_other($cc, 'MIME-Header'); $bcc = utf8_to_other($bcc, 'MIME-Header'); $reply = utf8_to_other($reply, 'MIME-Header'); } my %msg_args = (To => $to, From => $from, Subject => $subject, Type => $opt->{mimetype}, Cc => $cc, Bcc => $bcc, 'Reply-To' => $reply, ); if($opt->{html}) { if ($body =~ /\S/) { $msg_args{Type} ||= 'multipart/alternative'; } else { $msg_args{Type} ||= 'text/html' . ($utf8 ? '; charset=UTF-8' : ''); $msg_args{Data} ||= ($utf8 ? utf8_to_other($opt->{html}, 'utf8') : $opt->{html}); } $att1_format = 'flowed'; } else { $msg_args{Type} ||= 'multipart/mixed'; } my $msg = MIME::Lite->new(%msg_args); for(@extra) { m{(.*?):\s+(.*)}; my $name = $1 or next; next if lc($name) eq 'from'; my $content = $2 or next; $name =~ s/[-_]+/-/g; $name =~ s/\b(\w)/\U$1/g; $msg->add($name, ($utf8 ? utf8_to_other($content, 'utf8') : $content)) if $name && $content; } if ($body =~ /\S/) { $opt->{body_mime} ||= 'text/plain' . ($utf8 ? '; charset=UTF-8' : ''); $opt->{body_encoding} ||= 'quoted-printable'; $msg->attach( Type => $opt->{body_mime}, Encoding => $opt->{body_encoding}, Data => ($utf8 ? utf8_to_other($body, 'utf8') : $body), Disposition => $opt->{body_disposition} || 'inline', Format => $opt->{body_format} || $att1_format, ); } if(! ref($att) ) { my $fn = $att; $att = [ { path => $fn } ]; } elsif(ref($att) eq 'HASH') { $att = [ $att ]; } elsif(ref($att) eq 'ARRAY') { # turn array of file names into array of hash references my $new_att = []; for (@$att) { if (ref($_)) { push (@$new_att, $_); } else { push (@$new_att, {path => $_}); } } $att = $new_att; } $att ||= []; if($opt->{html} && $body =~ /\S/) { unshift @$att, {type => 'text/html' .($utf8 ? '; charset=UTF-8': ''), data => ($utf8 ? utf8_to_other($opt->{html}, 'utf8') : $opt->{html}), disposition => 'inline', }; } my %encoding_types = ( 'text/plain' => ($utf8 ? 'quoted-printable' : '8bit'), 'text/html' => 'quoted-printable', 'text/html; charset=UTF-8' => 'quoted-printable', ); for my $ref (@$att) { next unless $ref; next unless $ref->{path} || $ref->{data}; unless ($ref->{filename}) { my $fn = $ref->{path}; $fn =~ s:.*[\\/]::; $ref->{filename} = $fn; } $ref->{type} ||= 'AUTO'; $ref->{disposition} ||= 'attachment'; if(! $ref->{encoding}) { $ref->{encoding} = $encoding_types{$ref->{type}}; } eval { $msg->attach( Type => $ref->{type}, Path => $ref->{path}, ReadNow => 1, Data => $ref->{data}, Filename => $ref->{filename}, Encoding => $ref->{encoding}, Disposition => $ref->{disposition}, ); }; if($@) { ::logError("email tag: failed to attach %s: %s", $ref->{path}, $@); $Tag->error({name => 'email', set => errmsg('Failed to attach %s', $ref->{path})}); return; } } my $body = $msg->body_as_string; my $header = $msg->header_as_string; #::logDebug("[email] Mail: \n$header\n$body"); if($opt->{test}) { return "$header\n$body"; } else { last ATTACH unless $header; my @head = split(/\r?\n/,$header); $ok = send_mail(\@head,$body); $sent_with_attach = 1; } } $reply = '' unless defined $reply; $reply = "Reply-to: $reply\n" if $reply; if ($cc) { push(@extra, "Cc: $cc"); } if ($bcc) { push(@extra, "Bcc: $bcc"); } if ($utf8 && ! $opt->{mimetype}) { push(@extra, 'MIME-Version: 1.0'); push(@extra, 'Content-Type: text/plain; charset=UTF-8'); push(@extra, 'Content-Transfer-Encoding: 8bit'); } $ok = send_mail($to, $subject, $body, $reply, 0, @extra) unless $sent_with_attach; if (!$ok) { logError("Unable to send mail using $Vend::Cfg->{SendMailProgram}\n" . "To '$to'\n" . "From '$from'\n" . "With extra headers '$extra'\n" . "With reply-to '$reply'\n" . "With subject '$subject'\n" . "And body:\n$body"); } return $opt->{hide} ? '' : $ok; } EOR