1
0
Fork 0
mirror of https://we.phorge.it/source/phorge.git synced 2025-01-24 21:48:21 +01:00

Clean up and expose filtered recipient lists for PhabricatorMetaMTAMail

Summary: Provide a public interface to get all the filtered recipients of an email. The intent is to pass this along to Notifications so it can mark notifications as read if the user is also receiving an email, possibly based on some preference (see T1403). This also simplifies the enormous sendNow() method a little bit.

Test Plan: Added unit tests, and sent a few mails that should cover most/all of these cases. They appeared to produce the correct recipients.

Reviewers: btrahan, vrana, nh

Reviewed By: btrahan

CC: aran

Maniphest Tasks: T1403

Differential Revision: https://secure.phabricator.com/D3778
This commit is contained in:
epriestley 2012-10-22 16:25:43 -07:00
parent c679b78270
commit 90ccfe4da1
2 changed files with 262 additions and 78 deletions

View file

@ -18,6 +18,8 @@
/** /**
* See #394445 for an explanation of why this thing even exists. * See #394445 for an explanation of why this thing even exists.
*
* @task recipients Managing Recipients
*/ */
final class PhabricatorMetaMTAMail extends PhabricatorMetaMTADAO { final class PhabricatorMetaMTAMail extends PhabricatorMetaMTADAO {
@ -61,8 +63,8 @@ final class PhabricatorMetaMTAMail extends PhabricatorMetaMTADAO {
return $this; return $this;
} }
protected function getParam($param) { protected function getParam($param, $default = null) {
return idx($this->parameters, $param); return idx($this->parameters, $param, $default);
} }
/** /**
@ -310,6 +312,7 @@ final class PhabricatorMetaMTAMail extends PhabricatorMetaMTADAO {
return PhabricatorEnv::newObjectFromConfig('metamta.mail-adapter'); return PhabricatorEnv::newObjectFromConfig('metamta.mail-adapter');
} }
/** /**
* Attempt to deliver an email immediately, in this process. * Attempt to deliver an email immediately, in this process.
* *
@ -341,15 +344,26 @@ final class PhabricatorMetaMTAMail extends PhabricatorMetaMTADAO {
try { try {
$parameters = $this->parameters; $parameters = $this->parameters;
$phids = array(); $phids = array();
foreach ($parameters as $key => $value) {
switch ($key) {
case 'to':
$parameters[$key] = $this->buildToList();
break;
case 'cc':
$parameters[$key] = $this->buildCCList();
break;
}
}
foreach ($parameters as $key => $value) { foreach ($parameters as $key => $value) {
switch ($key) { switch ($key) {
case 'from': case 'from':
$value = array($value);
/* fallthrough */
case 'to': case 'to':
case 'cc': case 'cc':
if (!is_array($value)) { foreach ($value as $phid) {
$value = array($value);
}
foreach (array_filter($value) as $phid) {
$type = phid_get_type($phid); $type = phid_get_type($phid);
$phids[$phid] = $type; $phids[$phid] = $type;
} }
@ -359,8 +373,6 @@ final class PhabricatorMetaMTAMail extends PhabricatorMetaMTADAO {
$this->loadEmailAndNameDataFromPHIDs($phids); $this->loadEmailAndNameDataFromPHIDs($phids);
$exclude = array_fill_keys($this->getExcludeMailRecipientPHIDs(), true);
$params = $this->parameters; $params = $this->parameters;
$default = PhabricatorEnv::getEnvConfig('metamta.default-address'); $default = PhabricatorEnv::getEnvConfig('metamta.default-address');
if (empty($params['from'])) { if (empty($params['from'])) {
@ -368,21 +380,6 @@ final class PhabricatorMetaMTAMail extends PhabricatorMetaMTADAO {
} else { } else {
$from = $params['from']; $from = $params['from'];
// If the user has set their preferences to not send them email about
// things they do, exclude them from being on To or Cc.
$from_user = id(new PhabricatorUser())->loadOneWhere(
'phid = %s',
$from);
if ($from_user) {
$pref_key = PhabricatorUserPreferences::PREFERENCE_NO_SELF_MAIL;
$exclude_self = $from_user
->loadPreferences()
->getPreference($pref_key);
if ($exclude_self) {
$exclude[$from] = true;
}
}
if (!PhabricatorEnv::getEnvConfig('metamta.can-send-as-user')) { if (!PhabricatorEnv::getEnvConfig('metamta.can-send-as-user')) {
if (empty($params['reply-to'])) { if (empty($params['reply-to'])) {
$params['reply-to'] = $phids[$from]['email']; $params['reply-to'] = $phids[$from]['email'];
@ -415,7 +412,7 @@ final class PhabricatorMetaMTAMail extends PhabricatorMetaMTADAO {
$mailer->addReplyTo($value, $reply_to_name); $mailer->addReplyTo($value, $reply_to_name);
break; break;
case 'to': case 'to':
$to_emails = $this->filterSendable($value, $phids, $exclude); $to_emails = $this->filterSendable($value, $phids);
if ($to_emails) { if ($to_emails) {
$add_to = array_merge($add_to, $to_emails); $add_to = array_merge($add_to, $to_emails);
} }
@ -424,7 +421,7 @@ final class PhabricatorMetaMTAMail extends PhabricatorMetaMTADAO {
$add_to = array_merge($add_to, $value); $add_to = array_merge($add_to, $value);
break; break;
case 'cc': case 'cc':
$cc_emails = $this->filterSendable($value, $phids, $exclude); $cc_emails = $this->filterSendable($value, $phids);
if ($cc_emails) { if ($cc_emails) {
$add_cc = $cc_emails; $add_cc = $cc_emails;
} }
@ -568,6 +565,14 @@ final class PhabricatorMetaMTAMail extends PhabricatorMetaMTADAO {
} }
} }
if (!$add_to && !$add_cc) {
$this->setStatus(self::STATUS_VOID);
$this->setMessage(
"Message has no valid recipients: all To/CC are disabled or ".
"configured not to receive this mail.");
return $this->save();
}
$mailer->addHeader('X-Phabricator-Sent-This-Message', 'Yes'); $mailer->addHeader('X-Phabricator-Sent-This-Message', 'Yes');
$mailer->addHeader('X-Mail-Transport-Agent', 'MetaMTA'); $mailer->addHeader('X-Mail-Transport-Agent', 'MetaMTA');
@ -577,61 +582,13 @@ final class PhabricatorMetaMTAMail extends PhabricatorMetaMTADAO {
// If the message has mailtags, filter out any recipients who don't want // If the message has mailtags, filter out any recipients who don't want
// to receive this type of mail. // to receive this type of mail.
$mailtags = $this->getParam('mailtags'); $mailtags = $this->getParam('mailtags');
if ($mailtags && ($add_to || $add_cc)) { if ($mailtags) {
$tag_header = array(); $tag_header = array();
foreach ($mailtags as $mailtag) { foreach ($mailtags as $mailtag) {
$tag_header[] = '<'.$mailtag.'>'; $tag_header[] = '<'.$mailtag.'>';
} }
$tag_header = implode(', ', $tag_header); $tag_header = implode(', ', $tag_header);
$mailer->addHeader('X-Phabricator-Mail-Tags', $tag_header); $mailer->addHeader('X-Phabricator-Mail-Tags', $tag_header);
$exclude = array();
$all_recipients = array_merge(
array_keys($add_to),
array_keys($add_cc));
$all_prefs = id(new PhabricatorUserPreferences())->loadAllWhere(
'userPHID in (%Ls)',
$all_recipients);
$all_prefs = mpull($all_prefs, null, 'getUserPHID');
foreach ($all_recipients as $recipient) {
$prefs = idx($all_prefs, $recipient);
if (!$prefs) {
continue;
}
$user_mailtags = $prefs->getPreference(
PhabricatorUserPreferences::PREFERENCE_MAILTAGS,
array());
// The user must have elected to receive mail for at least one
// of the mailtags.
$send = false;
foreach ($mailtags as $tag) {
if (idx($user_mailtags, $tag, true)) {
$send = true;
break;
}
}
if (!$send) {
$exclude[$recipient] = true;
}
}
$add_to = array_diff_key($add_to, $exclude);
$add_cc = array_diff_key($add_cc, $exclude);
}
if (!$add_to && !$add_cc) {
$this->setStatus(self::STATUS_VOID);
$this->setMessage(
"Message has no valid recipients: all To/CC are disabled or ".
"configured not to receive this mail.");
return $this->save();
} }
// Some mailers require a valid "To:" in order to deliver mail. If we // Some mailers require a valid "To:" in order to deliver mail. If we
@ -819,12 +776,9 @@ final class PhabricatorMetaMTAMail extends PhabricatorMetaMTADAO {
return $name; return $name;
} }
private function filterSendable($value, $phids, $exclude) { private function filterSendable($value, $phids) {
$result = array(); $result = array();
foreach ($value as $phid) { foreach ($value as $phid) {
if (isset($exclude[$phid])) {
continue;
}
if (isset($phids[$phid]) && $phids[$phid]['mailable']) { if (isset($phids[$phid]) && $phids[$phid]['mailable']) {
$result[$phid] = $phids[$phid]['email']; $result[$phid] = $phids[$phid]['email'];
} }
@ -836,4 +790,148 @@ final class PhabricatorMetaMTAMail extends PhabricatorMetaMTADAO {
return PhabricatorEnv::getEnvConfig('metamta.one-mail-per-recipient'); return PhabricatorEnv::getEnvConfig('metamta.one-mail-per-recipient');
} }
/* -( Managing Recipients )------------------------------------------------ */
/**
* Get all of the recipients for this mail, after preference filters are
* applied. This list has all objects to whom delivery will be attempted, but
* does not exclude recipeints two whom delivery may be impossible.
*
* @return list<phid> A list of all recipients to whom delivery will be
* attempted.
* @task recipients
*/
public function buildRecipientList() {
return $this->resolveRecipients(
array_merge(
$this->getRawToPHIDs(),
$this->getRawCCPHIDs()));
}
/**
* Filter out duplicate, invalid, or excluded recipients from a PHID list.
*
* @param list<phid> Unfiltered recipients.
* @return list<phid> Filtered recipients.
*
* @task recipients
*/
private function resolveRecipients(array $phids) {
$phids = array_combine($phids, $phids);
// Exclude PHIDs explicitly marked for exclusion. We use this to prevent
// recipients of an accidental "Reply All" from receiving the followup
// mail from Phabricator.
$exclude = $this->getExcludeMailRecipientPHIDs();
$exclude = array_fill_keys($exclude, true);
$phids = array_diff_key($phids, $exclude);
// If the actor is a recipient and has configured their preferences not to
// send them mail about their own actions, drop them from the recipient
// list.
$from = $this->getParam('from');
if (isset($phids[$from])) {
$from_user = id(new PhabricatorUser())->loadOneWhere(
'phid = %s',
$from);
if ($from_user) {
$pref_key = PhabricatorUserPreferences::PREFERENCE_NO_SELF_MAIL;
$exclude_self = $from_user
->loadPreferences()
->getPreference($pref_key);
if ($exclude_self) {
unset($phids[$from]);
}
}
}
// Exclude all recipients who have set preferences to not receive this type
// of email (for example, a user who says they don't want emails about task
// CC changes).
$tags = $this->getParam('mailtags');
if ($tags && $phids) {
$all_prefs = id(new PhabricatorUserPreferences())->loadAllWhere(
'userPHID in (%Ls)',
$phids);
$all_prefs = mpull($all_prefs, null, 'getUserPHID');
foreach ($phids as $phid) {
$prefs = idx($all_prefs, $phid);
if (!$prefs) {
continue;
}
$user_mailtags = $prefs->getPreference(
PhabricatorUserPreferences::PREFERENCE_MAILTAGS,
array());
// The user must have elected to receive mail for at least one
// of the mailtags.
$send = false;
foreach ($tags as $tag) {
if (idx($user_mailtags, $tag, true)) {
$send = true;
break;
}
}
if (!$send) {
unset($phids[$phid]);
}
}
}
return array_keys($phids);
}
/**
* @task recipients
*/
private function buildToList() {
return $this->resolveRecipients($this->getRawToPHIDs());
}
/**
* @task recipients
*/
private function buildCCList() {
return $this->resolveRecipients($this->getRawCCPHIDs());
}
/**
* @task recipients
*/
private function getRawToPHIDs() {
$to = $this->getParam('to', array());
return $this->filterRawPHIDList($to);
}
/**
* @task recipients
*/
private function getRawCCPHIDs() {
$cc = $this->getParam('cc', array());
return $this->filterRawPHIDList($cc);
}
/**
* @task recipients
*/
private function filterRawPHIDList(array $list) {
$list = array_filter($list);
$list = array_unique($list);
return array_values($list);
}
} }

View file

@ -18,6 +18,92 @@
final class PhabricatorMetaMTAMailTestCase extends PhabricatorTestCase { final class PhabricatorMetaMTAMailTestCase extends PhabricatorTestCase {
protected function getPhabricatorTestCaseConfiguration() {
return array(
self::PHABRICATOR_TESTCONFIG_BUILD_STORAGE_FIXTURES => true,
);
}
public function testRecipients() {
$user = $this->generateNewTestUser();
$phid = $user->getPHID();
$prefs = $user->loadPreferences();
$mailer = new PhabricatorMailImplementationTestAdapter();
$mail = new PhabricatorMetaMTAMail();
$mail->addTos(array($phid));
$this->assertEqual(
true,
in_array($phid, $mail->buildRecipientList()),
'"To" is a recipient.');
// Test that the "No Self Mail" preference works correctly.
$mail->setFrom($phid);
$this->assertEqual(
true,
in_array($phid, $mail->buildRecipientList()),
'"From" does not exclude recipients by default.');
$prefs->setPreference(
PhabricatorUserPreferences::PREFERENCE_NO_SELF_MAIL,
true);
$prefs->save();
$this->assertEqual(
false,
in_array($phid, $mail->buildRecipientList()),
'"From" excludes recipients with no-self-mail set.');
$prefs->unsetPreference(
PhabricatorUserPreferences::PREFERENCE_NO_SELF_MAIL);
$prefs->save();
$this->assertEqual(
true,
in_array($phid, $mail->buildRecipientList()),
'"From" does not exclude recipients by default.');
// Test that explicit exclusion works correctly.
$mail->setExcludeMailRecipientPHIDs(array($phid));
$this->assertEqual(
false,
in_array($phid, $mail->buildRecipientList()),
'Explicit exclude excludes recipients.');
$mail->setExcludeMailRecipientPHIDs(array());
// Test that mail tag preferences exclude recipients.
$prefs->setPreference(
PhabricatorUserPreferences::PREFERENCE_MAILTAGS,
array(
'test-tag' => false,
));
$prefs->save();
$mail->setMailTags(array('test-tag'));
$this->assertEqual(
false,
in_array($phid, $mail->buildRecipientList()),
'Tag preference excludes recipients.');
$prefs->unsetPreference(PhabricatorUserPreferences::PREFERENCE_MAILTAGS);
$prefs->save();
$this->assertEqual(
true,
in_array($phid, $mail->buildRecipientList()),
'Recipients restored after tag preference removed.');
}
public function testThreadIDHeaders() { public function testThreadIDHeaders() {
$this->runThreadIDHeadersWithConfiguration(true, true); $this->runThreadIDHeadersWithConfiguration(true, true);
$this->runThreadIDHeadersWithConfiguration(true, false); $this->runThreadIDHeadersWithConfiguration(true, false);