You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

277 lines
9.8 KiB

<?php
/**
* Copyright 2004-2017 Horde LLC (http://www.horde.org/)
*
* See the enclosed file COPYING for license information (LGPL). If you
* did not receive this file, see http://www.horde.org/licenses/lgpl21.
*
* @category Horde
* @copyright 2004-2017 Horde LLC
* @license http://www.horde.org/licenses/lgpl21 LGPL 2.1
* @package Mime
*/
/**
* Message Disposition Notifications (RFC 3798).
*
* @author Michael Slusarz <slusarz@horde.org>
* @category Horde
* @copyright 2004-2017 Horde LLC
* @license http://www.horde.org/licenses/lgpl21 LGPL 2.1
* @package Mime
*/
class Horde_Mime_Mdn
{
/* RFC 3798 header for requesting a MDN. */
const MDN_HEADER = 'Disposition-Notification-To';
/**
* The Horde_Mime_Headers object.
*
* @var Horde_Mime_Headers
*/
protected $_headers;
/**
* The text of the original message.
*
* @var string
*/
protected $_msgtext = false;
/**
* Constructor.
*
* @param Horde_Mime_Headers $mime_headers A headers object.
*/
public function __construct(Horde_Mime_Headers $headers)
{
$this->_headers = $headers;
}
/**
* Returns the address(es) to return the MDN to.
*
* @return string The address(es) to send the MDN to. Returns null if no
* MDN is requested.
*/
public function getMdnReturnAddr()
{
/* RFC 3798 [2.1] requires the Disposition-Notification-To header
* for an MDN to be created. */
return ($hdr = $this->_headers[self::MDN_HEADER])
? strval($hdr)
: null;
}
/**
* Is user input required to send the MDN?
* Explicit confirmation is needed in some cases to prevent mail loops
* and the use of MDNs for mail bombing.
*
* @return boolean Is explicit user input required to send the MDN?
*/
public function userConfirmationNeeded()
{
$return_path = $this->_headers['Return-Path'];
/* RFC 3798 [2.1]: Explicit confirmation is needed if there is no
* Return-Path in the header. Also, "if the message contains more
* than one Return-Path header, the implementation may [] treat the
* situation as a failure of the comparison." */
if (!$return_path || (count($return_path->value) > 1)) {
return true;
}
/* RFC 3798 [2.1]: Explicit confirmation is needed if there is more
* than one distinct address in the Disposition-Notification-To
* header. */
$addr_ob = ($hdr = $this->_headers[self::MDN_HEADER])
? $hdr->getAddressList(true)
: array();
switch (count($addr_ob)) {
case 0:
return false;
case 1:
// No-op
break;
default:
return true;
}
/* RFC 3798 [2.1] states that "MDNs SHOULD NOT be sent automatically
* if the address in the Disposition-Notification-To header differs
* from the address in the Return-Path header." This comparison is
* case-sensitive for the mailbox part and case-insensitive for the
* host part. */
$ret_ob = new Horde_Mail_Rfc822_Address($return_path->value);
return (!$ret_ob->valid || !$addr_ob->match($ret_ob));
}
/**
* When generating the MDN, should we return the enitre text of the
* original message? The default is no - we only return the headers of
* the original message. If the text is passed in via this method, we
* will return the entire message.
*
* @param string $text The text of the original message.
*/
public function originalMessageText($text)
{
$this->_msgtext = $text;
}
/**
* Generate the MDN according to the specifications listed in RFC
* 3798 [3].
*
* @param boolean $action Was this MDN type a result of a manual
* action on part of the user?
* @param boolean $sending Was this MDN sent as a result of a manual
* action on part of the user?
* @param string $type The type of action performed by the user.
* Per RFC 3798 [3.2.6.2] the following types are
* valid:
* - deleted
* - displayed
* @param string $name The name of the local server.
* @param Horde_Mail_Transport $mailer Mail transport object.
* @param array $opts Additional options:
* - charset: (string) Default charset.
* DEFAULT: NONE
* - from_addr: (string) From address.
* DEFAULT: NONE
* @param array $mod The list of modifications. Per RFC 3798
* [3.2.6.3] the following modifications are
* valid:
* - error
* @param array $err If $mod is 'error', the additional
* information to provide. Key is the type of
* modification, value is the text.
*/
public function generate($action, $sending, $type, $name, $mailer,
array $opts = array(), array $mod = array(),
array $err = array())
{
$opts = array_merge(array(
'charset' => null,
'from_addr' => null
), $opts);
if (!($hdr = $this->_headers[self::MDN_HEADER])) {
throw new RuntimeException(
'Need at least one address to send MDN to.'
);
}
$to = $hdr->getAddressList(true);
$ua = Horde_Mime_Headers_UserAgent::create();
if ($orig_recip = $this->_headers['Original-Recipient']) {
$orig_recip = $orig_recip->value_single;
}
/* Set up the mail headers. */
$msg_headers = new Horde_Mime_Headers();
$msg_headers->addHeaderOb(Horde_Mime_Headers_MessageId::create());
$msg_headers->addHeaderOb($ua);
/* RFC 3834 [5.2] */
$msg_headers->addHeader('Auto-Submitted', 'auto-replied');
$msg_headers->addHeaderOb(Horde_Mime_Headers_Date::create());
if ($opts['from_addr']) {
$msg_headers->addHeader('From', $opts['from_addr']);
}
$msg_headers->addHeader('To', $to);
$msg_headers->addHeader('Subject', Horde_Mime_Translation::t("Disposition Notification"));
/* MDNs are a subtype of 'multipart/report'. */
$msg = new Horde_Mime_Part();
$msg->setType('multipart/report');
$msg->setContentTypeParameter('report-type', 'disposition-notification');
/* The first part is a human readable message. */
$part_one = new Horde_Mime_Part();
$part_one->setType('text/plain');
$part_one->setCharset($opts['charset']);
if ($type == 'displayed') {
$contents = sprintf(
Horde_Mime_Translation::t("The message sent on %s to %s with subject \"%s\" has been displayed.\n\nThis is no guarantee that the message has been read or understood."),
$this->_headers['Date'],
$this->_headers['To'],
$this->_headers['Subject']
);
$flowed = new Horde_Text_Flowed($contents, $opts['charset']);
$flowed->setDelSp(true);
$part_one->setContentTypeParameter('format', 'flowed');
$part_one->setContentTypeParameter('DelSp', 'Yes');
$part_one->setContents($flowed->toFlowed());
}
// TODO: Messages for other notification types.
$msg[] = $part_one;
/* The second part is a machine-parseable description. */
$part_two = new Horde_Mime_Part();
$part_two->setType('message/disposition-notification');
$part_two_h = new Horde_Mime_Headers();
$part_two_h->addHeader('Reporting-UA', $name . '; ' . $ua);
if (!empty($orig_recip)) {
$part_two_h->addHeader('Original-Recipient', 'rfc822;' . $orig_recip);
}
if ($opts['from_addr']) {
$part_two_h->addHeader('Final-Recipient', 'rfc822;' . $opts['from_addr']);
}
if ($msg_id = $this->_headers['Message-ID']) {
$part_two_h->addHeader('Original-Message-ID', strval($msg_id));
}
/* Create the Disposition field now (RFC 3798 [3.2.6]). */
$dispo = (($action) ? 'manual-action' : 'automatic-action') .
'/' .
(($sending) ? 'MDN-sent-manually' : 'MDN-sent-automatically') .
'; ' .
$type;
if (!empty($mod)) {
$dispo .= '/' . implode(', ', $mod);
}
$part_two_h->addHeader('Disposition', $dispo);
if (in_array('error', $mod) && isset($err['error'])) {
$part_two_h->addHeader('Error', $err['error']);
}
$part_two->setContents(trim($part_two_h->toString()) . "\n");
$msg[] = $part_two;
/* The third part is the text of the original message. RFC 3798 [3]
* allows us to return only a portion of the entire message - this
* is left up to the user. */
$part_three = new Horde_Mime_Part();
$part_three->setType('message/rfc822');
$part_three_text = array(trim($this->_headers->toString()) . "\n");
if (!empty($this->_msgtext)) {
$part_three_text[] = "\n" . $this->_msgtext;
}
$part_three->setContents($part_three_text);
$msg[] = $part_three;
return $msg->send($to, $msg_headers, $mailer);
}
/**
* Add a MDN (read receipt) request header.
*
* @param mixed $to The address(es) the receipt should be mailed to.
*/
public function addMdnRequestHeaders($to)
{
/* This is the RFC 3798 way of requesting a receipt. */
$this->_headers->addHeader(self::MDN_HEADER, $to);
}
}