Skip to content

Commit

Permalink
CRM-21521: fix nested multipart handling
Browse files Browse the repository at this point in the history
When for example a multipart-report email body's first part is
multipart-related, extract the text correctly instead of crashing.

This logic should handle all kinds of weird multipart nesting, up
to MIME_MAX_RECURSION levels deep.
  • Loading branch information
ejegg committed Jan 8, 2018
1 parent 06640b2 commit 01467ae
Show file tree
Hide file tree
Showing 3 changed files with 434 additions and 49 deletions.
155 changes: 106 additions & 49 deletions CRM/Utils/Mail/EmailProcessor.php
Original file line number Diff line number Diff line change
Expand Up @@ -41,6 +41,8 @@
*/
class CRM_Utils_Mail_EmailProcessor {

const MIME_MAX_RECURSION = 10;

/**
* Process the default mailbox (ie. that is used by civiMail for the bounce)
*
Expand Down Expand Up @@ -286,55 +288,7 @@ public static function _process($civiMail, $dao, $is_create_activities) {
$text = $mail->body->text;
}
elseif ($mail->body instanceof ezcMailMultipart) {
if ($mail->body instanceof ezcMailMultipartReport) {
$part = $mail->body->getMachinePart();
if ($part instanceof ezcMailDeliveryStatus) {
foreach ($part->recipients as $rec) {
if (isset($rec["Diagnostic-Code"])) {
$text = $rec["Diagnostic-Code"];
break;
}
elseif (isset($rec["Description"])) {
$text = $rec["Description"];
break;
}
// no diagnostic info present - try getting the human readable part
elseif (isset($rec["Status"])) {
$text = $rec["Status"];
$textpart = $mail->body->getReadablePart();
if ($textpart != NULL and isset($textpart->text)) {
$text .= " " . $textpart->text;
}
else {
$text .= " Delivery failed but no diagnostic code or description.";
}
break;
}
}
}
elseif ($part != NULL and isset($part->text)) {
$text = $part->text;
}
elseif (($part = $mail->body->getReadablePart()) != NULL) {
$text = $part->text;
}
}
elseif ($mail->body instanceof ezcMailMultipartRelated) {
foreach ($mail->body->getRelatedParts() as $part) {
if (isset($part->subType) and $part->subType == 'plain') {
$text = $part->text;
break;
}
}
}
else {
foreach ($mail->body->getParts() as $part) {
if (isset($part->subType) and $part->subType == 'plain') {
$text = $part->text;
break;
}
}
}
$text = self::getTextFromMultipart($mail->body);
}

if (
Expand Down Expand Up @@ -475,4 +429,107 @@ public static function _process($civiMail, $dao, $is_create_activities) {
}
}

/**
* @param \ezcMailMultipart $multipart
* @param int $recursionLevel
*
* @return array
*/
protected static function getTextFromMultipart($multipart, $recursionLevel = 0) {
if ($recursionLevel >= self::MIME_MAX_RECURSION) {
return NULL;
}
$recursionLevel += 1;
$text = NULL;
if ($multipart instanceof ezcMailMultipartReport) {
$text = self::getTextFromMulipartReport($multipart, $recursionLevel);
}
elseif ($multipart instanceof ezcMailMultipartRelated) {
$text = self::getTextFromMultipartRelated($multipart, $recursionLevel);
}
else {
foreach ($multipart->getParts() as $part) {
if (isset($part->subType) and $part->subType === 'plain') {
$text = $part->text;
}
elseif ($part instanceof ezcMailMultipart) {
$text = self::getTextFromMultipart($part, $recursionLevel);
}
if ($text) {
break;
}
}
}
return $text;
}

/**
* @param \ezcMailMultipartRelated $related
* @param int $recursionLevel
*
* @return array
*/
protected static function getTextFromMultipartRelated($related, $recursionLevel) {
$text = NULL;
foreach ($related->getRelatedParts() as $part) {
if (isset($part->subType) and $part->subType === 'plain') {
$text = $part->text;
}
elseif ($part instanceof ezcMailMultipart) {
$text = self::getTextFromMultipart($part, $recursionLevel);
}
if ($text) {
break;
}
}
return $text;
}

/**
* @param \ezcMailMultipartReport $multipart
* @param $recursionLevel
*
* @return array
*/
protected static function getTextFromMulipartReport($multipart, $recursionLevel) {
$text = NULL;
$part = $multipart->getMachinePart();
if ($part instanceof ezcMailDeliveryStatus) {
foreach ($part->recipients as $rec) {
if (isset($rec["Diagnostic-Code"])) {
$text = $rec["Diagnostic-Code"];
break;
}
elseif (isset($rec["Description"])) {
$text = $rec["Description"];
break;
}
// no diagnostic info present - try getting the human readable part
elseif (isset($rec["Status"])) {
$text = $rec["Status"];
$textpart = $multipart->getReadablePart();
if ($textpart !== NULL and isset($textpart->text)) {
$text .= " " . $textpart->text;
}
else {
$text .= " Delivery failed but no diagnostic code or description.";
}
break;
}
}
}
elseif ($part !== NULL and isset($part->text)) {
$text = $part->text;
}
elseif (($part = $multipart->getReadablePart()) !== NULL) {
if (isset($part->text)) {
$text = $part->text;
}
elseif ($part instanceof ezcMailMultipart) {
$text = self::getTextFromMultipart($part, $recursionLevel);
}
}
return $text;
}

}
15 changes: 15 additions & 0 deletions tests/phpunit/CRM/Utils/Mail/EmailProcessorTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -69,6 +69,21 @@ public function testProcessingMultipartRelatedEmail() {
$this->checkMailingBounces(1);
}

/**
* Tests that a nested multipart email does not cause pain & misery & fatal errors.
*
* Sample anonymized from an email that broke bounce processing at Wikimedia
*/
public function testProcessingNestedMultipartEmail() {
$this->setUpMailing();
$mail = 'test_nested_message.eml';

copy(__DIR__ . '/data/bounces/' . $mail, __DIR__ . '/data/mail/' . $mail);
$this->callAPISuccess('job', 'fetch_bounces', array());
$this->assertFalse(file_exists(__DIR__ . '/data/mail/' . $mail));
$this->checkMailingBounces(1);
}

/**
* Test that a deleted email does not cause a hard fail.
*
Expand Down
Loading

0 comments on commit 01467ae

Please sign in to comment.