Skip to content
Projects
Groups
Snippets
Help
This project
Loading...
Sign in / Register
Toggle navigation
bugzilla
Project
Project
Details
Activity
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Wiki
Wiki
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
etersoft
bugzilla
Commits
f4c7bf2d
Commit
f4c7bf2d
authored
Aug 09, 2011
by
Max Kanat-Alexander
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Bug 437076: Allow email_in to accept multipart/alternative HTML email with
attachments r=glob, a=mkanat
parent
f539ec23
Hide whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
83 additions
and
43 deletions
+83
-43
Requirements.pm
Bugzilla/Install/Requirements.pm
+0
-6
email_in.pl
email_in.pl
+81
-35
user-error.html.tmpl
template/en/default/global/user-error.html.tmpl
+2
-2
No files found.
Bugzilla/Install/Requirements.pm
View file @
f4c7bf2d
...
...
@@ -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
,
...
...
email_in.pl
View file @
f4c7bf2d
...
...
@@ -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.
template/en/default/global/user-error.html.tmpl
View file @
f4c7bf2d
...
...
@@ -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" %]
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment