Commit f4c7bf2d authored by Max Kanat-Alexander's avatar Max Kanat-Alexander

Bug 437076: Allow email_in to accept multipart/alternative HTML email with

attachments r=glob, a=mkanat
parent f539ec23
......@@ -322,12 +322,6 @@ sub OPTIONAL_MODULES {
# Inbound Email
{
package => 'Email-MIME-Attachment-Stripper',
module => 'Email::MIME::Attachment::Stripper',
version => 0,
feature => ['inbound_email'],
},
{
package => 'Email-Reply',
module => 'Email::Reply',
version => 0,
......
......@@ -38,7 +38,6 @@ use Data::Dumper;
use Email::Address;
use Email::Reply qw(reply);
use Email::MIME;
use Email::MIME::Attachment::Stripper;
use Getopt::Long qw(:config bundling);
use Pod::Usage;
use Encode;
......@@ -64,6 +63,14 @@ use Bugzilla::Hook;
# in a message. RFC-compliant mailers use this.
use constant SIGNATURE_DELIMITER => '-- ';
# These MIME types represent a "body" of an email if they have an
# "inline" Content-Disposition (or no content disposition).
use constant BODY_TYPES => qw(
text/plain
text/html
multipart/alternative
);
# $input_email is a global so that it can be used in die_handler.
our ($input_email, %switch);
......@@ -95,9 +102,6 @@ sub parse_mail {
}
my ($body, $attachments) = get_body_and_attachments($input_email);
if (@$attachments) {
$fields{'attachments'} = $attachments;
}
debug_print("Body:\n" . $body, 3);
......@@ -155,6 +159,11 @@ sub parse_mail {
debug_print("Parsed Fields:\n" . Dumper(\%fields), 2);
debug_print("Attachments:\n" . Dumper($attachments), 3);
if (@$attachments) {
$fields{'attachments'} = $attachments;
}
return \%fields;
}
......@@ -239,15 +248,17 @@ sub handle_attachments {
$dbh->bz_start_transaction();
my ($update_comment, $update_bug);
foreach my $attachment (@$attachments) {
my $data = delete $attachment->{payload};
debug_print("Inserting Attachment: " . Dumper($attachment), 2);
$attachment->{content_type} ||= 'application/octet-stream';
debug_print("Inserting Attachment: " . Dumper($attachment), 3);
my $type = $attachment->content_type || 'application/octet-stream';
# MUAs add stuff like "name=" to content-type that we really don't
# want.
$type =~ s/;.*//;
my $obj = Bugzilla::Attachment->create({
bug => $bug,
description => $attachment->{filename},
filename => $attachment->{filename},
mimetype => $attachment->{content_type},
data => $data,
description => $attachment->filename(1),
filename => $attachment->filename(1),
mimetype => $type,
data => $attachment->body,
});
# If we added a comment, and our comment does not already have a type,
# and this is our first attachment, then we make the comment an
......@@ -285,21 +296,36 @@ sub get_body_and_attachments {
my ($email) = @_;
my $ct = $email->content_type || 'text/plain';
debug_print("Splitting Body and Attachments [Type: $ct]...");
debug_print("Splitting Body and Attachments [Type: $ct]...", 2);
my ($bodies, $attachments) = split_body_and_attachments($email);
debug_print(scalar(@$bodies) . " body part(s) and " . scalar(@$attachments)
. " attachment part(s).");
debug_print('Bodies: ' . Dumper($bodies), 3);
# Get the first part of the email that contains a text body,
# and make all the other pieces into attachments. (This handles
# people or MUAs who accidentally attach text files as an "inline"
# attachment.)
my $body;
my $attachments = [];
if ($ct =~ /^multipart\/(alternative|signed)/i) {
$body = get_text_alternative($email);
while (@$bodies) {
my $possible = shift @$bodies;
$body = get_text_alternative($possible);
if (defined $body) {
unshift(@$attachments, @$bodies);
last;
}
}
else {
my $stripper = new Email::MIME::Attachment::Stripper(
$email, force_filename => 1);
my $message = $stripper->message;
$body = get_text_alternative($message);
$attachments = [$stripper->attachments];
if (!defined $body) {
# Note that this only happens if the email does not contain any
# text/plain parts. If the email has an empty text/plain part,
# you're fine, and this message does NOT get thrown.
ThrowUserError('email_no_text_plain');
}
debug_print("Picked Body:\n$body", 2);
return ($body, $attachments);
}
......@@ -315,8 +341,8 @@ sub get_text_alternative {
if ($ct =~ /charset="?([^;"]+)/) {
$charset= $1;
}
debug_print("Part Content-Type: $ct", 2);
debug_print("Part Character Encoding: $charset", 2);
debug_print("Alternative Part Content-Type: $ct", 2);
debug_print("Alternative Part Character Encoding: $charset", 2);
if (!$ct || $ct =~ /^text\/plain/i) {
$body = $part->body;
if (Bugzilla->params->{'utf8'} && !utf8::is_utf8($body)) {
......@@ -326,13 +352,6 @@ sub get_text_alternative {
}
}
if (!defined $body) {
# Note that this only happens if the email does not contain any
# text/plain parts. If the email has an empty text/plain part,
# you're fine, and this message does NOT get thrown.
ThrowUserError('email_no_text_plain');
}
return $body;
}
......@@ -357,6 +376,38 @@ sub html_strip {
return $var;
}
sub split_body_and_attachments {
my ($email) = @_;
my (@body, @attachments);
foreach my $part ($email->parts) {
my $ct = lc($part->content_type || 'text/plain');
my $disposition = lc($part->header('Content-Disposition') || 'inline');
# Remove the charset, etc. from the content-type, we don't care here.
$ct =~ s/;.*//;
debug_print("Part Content-Type: [$ct]", 2);
debug_print("Part Disposition: [$disposition]", 2);
if ($disposition eq 'inline' and grep($_ eq $ct, BODY_TYPES)) {
push(@body, $part);
next;
}
if (scalar($part->parts) == 1) {
push(@attachments, $part);
next;
}
# If this part has sub-parts, analyze them similarly to how we
# did above and return the relevant pieces.
my ($add_body, $add_attachments) = split_body_and_attachments($part);
push(@body, @$add_body);
push(@attachments, @$add_attachments);
}
return (\@body, \@attachments);
}
sub die_handler {
my ($msg) = @_;
......@@ -574,9 +625,4 @@ The email interface only accepts emails that are correctly formatted
per RFC2822. If you send it an incorrectly formatted message, it
may behave in an unpredictable fashion.
You cannot send an HTML mail along with attachments. If you do, Bugzilla
will reject your email, saying that it doesn't contain any text. This
is a bug in L<Email::MIME::Attachment::Stripper> that we can't work
around.
You cannot modify Flags through the email interface.
......@@ -444,8 +444,8 @@
Email address confirmation failed.
[% ELSIF error == "email_no_text_plain" %]
Your message did not contain any text.[% terms.Bugzilla %] does not
accept HTML-only email, or HTML email with attachments.
Your message did not contain any text. [% terms.Bugzilla %] does not
accept HTML-only email.
[% ELSIF error == "empty_group_description" %]
[% title = "The group description can not be empty" %]
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment