mirror of
https://we.phorge.it/source/phorge.git
synced 2025-04-05 17:08:27 +02:00
Build separate mail for each recipient, honoring recipient access levels
Summary: Ref T6367. Removes `multiplexMail()`! We can't pass a single body into a function which splits it anymore: we need to split recipients first, then build bodies for each recipient list. This lets us build separate bodies for each recipient's individual translation/access levels. The new logic does this: - First, split recipients into groups called "targets". - Each target corresponds to one actual mail we're going to build. - Each target has a viewer (whose translation / access levels will be used to generate the mail). - Each target has a to/cc list (the users who we'll ultimately send the mail to). - For each target, build a custom mail body based on the viewer's access levels and settings (language prefs not actually implemented). - Then, deliver the mail. Test Plan: - Read new config help. Then did a bunch of testing, primarily with `bin/mail list-outbound` and `bin/mail show-outbound` (to review generated mail), `bin/phd debug taskmaster` (to run daemons freely) and `bin/worker execute --id <id>` (to repeatedly test a specific piece of code after identifying an issue). With `one-mail-per-recipient` on (default): - Sent mail to multiple users. - Verified mail showed up in `mail list-outbound`. - Examined mail with `mail show-outbound`. - Added a project that a subscriber could not see. - Verified it was not present in `X-Phabricator-Projects`. - Verified it was rendered as "Restricted Project" for the non-permissioned viewer. - Added a subscriber, then changed the object policy so they could not see it and sent mail. - Verified I received mail but the other user did not. - Enabled public replies and verified mail generated with public addresses. - Disabld public replies and verified mail generated with private addresses. With `one-mail-per-recipient` off: - Verified that one mail is sent to all recipients. - Verified users who can not see the object are still filtered. - Verified that partially-visible projects are completely visible in the mail (this violates policies, as documented, as the best available compromise). - Enabled public replies and verified the mail generated with "Reply To". Reviewers: btrahan Reviewed By: btrahan Subscribers: carlsverre, epriestley Maniphest Tasks: T6367 Differential Revision: https://secure.phabricator.com/D13131
This commit is contained in:
parent
a9ceebbdb1
commit
6db97bde12
10 changed files with 408 additions and 237 deletions
|
@ -2021,6 +2021,7 @@ phutil_register_library_map(array(
|
||||||
'PhabricatorMailReceiverTestCase' => 'applications/metamta/receiver/__tests__/PhabricatorMailReceiverTestCase.php',
|
'PhabricatorMailReceiverTestCase' => 'applications/metamta/receiver/__tests__/PhabricatorMailReceiverTestCase.php',
|
||||||
'PhabricatorMailReplyHandler' => 'applications/metamta/replyhandler/PhabricatorMailReplyHandler.php',
|
'PhabricatorMailReplyHandler' => 'applications/metamta/replyhandler/PhabricatorMailReplyHandler.php',
|
||||||
'PhabricatorMailSetupCheck' => 'applications/config/check/PhabricatorMailSetupCheck.php',
|
'PhabricatorMailSetupCheck' => 'applications/config/check/PhabricatorMailSetupCheck.php',
|
||||||
|
'PhabricatorMailTarget' => 'applications/metamta/replyhandler/PhabricatorMailTarget.php',
|
||||||
'PhabricatorMailgunConfigOptions' => 'applications/config/option/PhabricatorMailgunConfigOptions.php',
|
'PhabricatorMailgunConfigOptions' => 'applications/config/option/PhabricatorMailgunConfigOptions.php',
|
||||||
'PhabricatorMainMenuSearchView' => 'view/page/menu/PhabricatorMainMenuSearchView.php',
|
'PhabricatorMainMenuSearchView' => 'view/page/menu/PhabricatorMainMenuSearchView.php',
|
||||||
'PhabricatorMainMenuView' => 'view/page/menu/PhabricatorMainMenuView.php',
|
'PhabricatorMainMenuView' => 'view/page/menu/PhabricatorMainMenuView.php',
|
||||||
|
@ -5429,6 +5430,7 @@ phutil_register_library_map(array(
|
||||||
'PhabricatorMailManagementWorkflow' => 'PhabricatorManagementWorkflow',
|
'PhabricatorMailManagementWorkflow' => 'PhabricatorManagementWorkflow',
|
||||||
'PhabricatorMailReceiverTestCase' => 'PhabricatorTestCase',
|
'PhabricatorMailReceiverTestCase' => 'PhabricatorTestCase',
|
||||||
'PhabricatorMailSetupCheck' => 'PhabricatorSetupCheck',
|
'PhabricatorMailSetupCheck' => 'PhabricatorSetupCheck',
|
||||||
|
'PhabricatorMailTarget' => 'Phobject',
|
||||||
'PhabricatorMailgunConfigOptions' => 'PhabricatorApplicationConfigOptions',
|
'PhabricatorMailgunConfigOptions' => 'PhabricatorApplicationConfigOptions',
|
||||||
'PhabricatorMainMenuSearchView' => 'AphrontView',
|
'PhabricatorMainMenuSearchView' => 'AphrontView',
|
||||||
'PhabricatorMainMenuView' => 'AphrontView',
|
'PhabricatorMainMenuView' => 'AphrontView',
|
||||||
|
|
|
@ -54,6 +54,8 @@ alincoln", "To: usgrant", "To: htaft"). The major advantages and disadvantages
|
||||||
of each approach are:
|
of each approach are:
|
||||||
|
|
||||||
- One mail to everyone:
|
- One mail to everyone:
|
||||||
|
- This violates policy controls. The body of the mail is generated without
|
||||||
|
respect for object policies.
|
||||||
- Recipients can see To/Cc at a glance.
|
- Recipients can see To/Cc at a glance.
|
||||||
- If you use mailing lists, you won't get duplicate mail if you're
|
- If you use mailing lists, you won't get duplicate mail if you're
|
||||||
a normal recipient and also Cc'd on a mailing list.
|
a normal recipient and also Cc'd on a mailing list.
|
||||||
|
@ -65,6 +67,7 @@ of each approach are:
|
||||||
- Not supported with a private reply-to address.
|
- Not supported with a private reply-to address.
|
||||||
- Mails are sent in the server default translation.
|
- Mails are sent in the server default translation.
|
||||||
- One mail to each user:
|
- One mail to each user:
|
||||||
|
- Policy controls work correctly and are enforced per-user.
|
||||||
- Recipients need to look in the mail body to see To/Cc.
|
- Recipients need to look in the mail body to see To/Cc.
|
||||||
- If you use mailing lists, recipients may sometimes get duplicate
|
- If you use mailing lists, recipients may sometimes get duplicate
|
||||||
mail.
|
mail.
|
||||||
|
@ -74,8 +77,6 @@ of each approach are:
|
||||||
- Required if private reply-to addresses are configured.
|
- Required if private reply-to addresses are configured.
|
||||||
- Mails are sent in the language of user preference.
|
- Mails are sent in the language of user preference.
|
||||||
|
|
||||||
In the code, splitting one outbound email into one-per-recipient is sometimes
|
|
||||||
referred to as "multiplexing".
|
|
||||||
EODOC
|
EODOC
|
||||||
));
|
));
|
||||||
|
|
||||||
|
@ -222,6 +223,7 @@ EODOC
|
||||||
'metamta.one-mail-per-recipient',
|
'metamta.one-mail-per-recipient',
|
||||||
'bool',
|
'bool',
|
||||||
true)
|
true)
|
||||||
|
->setLocked(true)
|
||||||
->setBoolOptions(
|
->setBoolOptions(
|
||||||
array(
|
array(
|
||||||
pht('Send Mail To Each Recipient'),
|
pht('Send Mail To Each Recipient'),
|
||||||
|
|
|
@ -21,9 +21,8 @@ final class ConpherenceReplyHandler extends PhabricatorMailReplyHandler {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getPrivateReplyHandlerEmailAddress(
|
public function getPrivateReplyHandlerEmailAddress(PhabricatorUser $user) {
|
||||||
PhabricatorObjectHandle $handle) {
|
return $this->getDefaultPrivateReplyHandlerEmailAddress($user, 'Z');
|
||||||
return $this->getDefaultPrivateReplyHandlerEmailAddress($handle, 'Z');
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getPublicReplyHandlerEmailAddress() {
|
public function getPublicReplyHandlerEmailAddress() {
|
||||||
|
|
|
@ -47,7 +47,7 @@ abstract class PhabricatorMailReplyHandler {
|
||||||
|
|
||||||
abstract public function validateMailReceiver($mail_receiver);
|
abstract public function validateMailReceiver($mail_receiver);
|
||||||
abstract public function getPrivateReplyHandlerEmailAddress(
|
abstract public function getPrivateReplyHandlerEmailAddress(
|
||||||
PhabricatorObjectHandle $handle);
|
PhabricatorUser $user);
|
||||||
|
|
||||||
public function getReplyHandlerDomain() {
|
public function getReplyHandlerDomain() {
|
||||||
return PhabricatorEnv::getEnvConfig('metamta.reply-handler-domain');
|
return PhabricatorEnv::getEnvConfig('metamta.reply-handler-domain');
|
||||||
|
@ -117,151 +117,6 @@ abstract class PhabricatorMailReplyHandler {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
final public function getRecipientsSummary(
|
|
||||||
array $to_handles,
|
|
||||||
array $cc_handles) {
|
|
||||||
assert_instances_of($to_handles, 'PhabricatorObjectHandle');
|
|
||||||
assert_instances_of($cc_handles, 'PhabricatorObjectHandle');
|
|
||||||
|
|
||||||
$body = '';
|
|
||||||
|
|
||||||
if (PhabricatorEnv::getEnvConfig('metamta.recipients.show-hints')) {
|
|
||||||
if ($to_handles) {
|
|
||||||
$body .= "To: ".implode(', ', mpull($to_handles, 'getName'))."\n";
|
|
||||||
}
|
|
||||||
if ($cc_handles) {
|
|
||||||
$body .= "Cc: ".implode(', ', mpull($cc_handles, 'getName'))."\n";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return $body;
|
|
||||||
}
|
|
||||||
|
|
||||||
final public function getRecipientsSummaryHTML(
|
|
||||||
array $to_handles,
|
|
||||||
array $cc_handles) {
|
|
||||||
assert_instances_of($to_handles, 'PhabricatorObjectHandle');
|
|
||||||
assert_instances_of($cc_handles, 'PhabricatorObjectHandle');
|
|
||||||
|
|
||||||
if (PhabricatorEnv::getEnvConfig('metamta.recipients.show-hints')) {
|
|
||||||
$body = array();
|
|
||||||
if ($to_handles) {
|
|
||||||
$body[] = phutil_tag('strong', array(), 'To: ');
|
|
||||||
$body[] = phutil_implode_html(', ', mpull($to_handles, 'getName'));
|
|
||||||
$body[] = phutil_tag('br');
|
|
||||||
}
|
|
||||||
if ($cc_handles) {
|
|
||||||
$body[] = phutil_tag('strong', array(), 'Cc: ');
|
|
||||||
$body[] = phutil_implode_html(', ', mpull($cc_handles, 'getName'));
|
|
||||||
$body[] = phutil_tag('br');
|
|
||||||
}
|
|
||||||
return phutil_tag('div', array(), $body);
|
|
||||||
} else {
|
|
||||||
return '';
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
final public function multiplexMail(
|
|
||||||
PhabricatorMetaMTAMail $mail_template,
|
|
||||||
array $to_handles,
|
|
||||||
array $cc_handles) {
|
|
||||||
assert_instances_of($to_handles, 'PhabricatorObjectHandle');
|
|
||||||
assert_instances_of($cc_handles, 'PhabricatorObjectHandle');
|
|
||||||
|
|
||||||
$result = array();
|
|
||||||
|
|
||||||
// If MetaMTA is configured to always multiplex, skip the single-email
|
|
||||||
// case.
|
|
||||||
if (!PhabricatorMetaMTAMail::shouldMultiplexAllMail()) {
|
|
||||||
// If private replies are not supported, simply send one email to all
|
|
||||||
// recipients and CCs. This covers cases where we have no reply handler,
|
|
||||||
// or we have a public reply handler.
|
|
||||||
if (!$this->supportsPrivateReplies()) {
|
|
||||||
$mail = clone $mail_template;
|
|
||||||
$mail->addTos(mpull($to_handles, 'getPHID'));
|
|
||||||
$mail->addCCs(mpull($cc_handles, 'getPHID'));
|
|
||||||
|
|
||||||
if ($this->supportsPublicReplies()) {
|
|
||||||
$reply_to = $this->getPublicReplyHandlerEmailAddress();
|
|
||||||
$mail->setReplyTo($reply_to);
|
|
||||||
}
|
|
||||||
|
|
||||||
$result[] = $mail;
|
|
||||||
|
|
||||||
return $result;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// TODO: This is pretty messy. We should really be doing all of this
|
|
||||||
// multiplexing in the task queue, but that requires significant rewriting
|
|
||||||
// in the general case. ApplicationTransactions can do it fairly easily,
|
|
||||||
// but other mail sites currently can not, so we need to support this
|
|
||||||
// junky version until they catch up and we can swap things over.
|
|
||||||
|
|
||||||
$to_handles = $this->expandRecipientHandles($to_handles);
|
|
||||||
$cc_handles = $this->expandRecipientHandles($cc_handles);
|
|
||||||
|
|
||||||
$tos = mpull($to_handles, null, 'getPHID');
|
|
||||||
$ccs = mpull($cc_handles, null, 'getPHID');
|
|
||||||
|
|
||||||
// Merge all the recipients together. TODO: We could keep the CCs as real
|
|
||||||
// CCs and send to a "noreply@domain.com" type address, but keep it simple
|
|
||||||
// for now.
|
|
||||||
$recipients = $tos + $ccs;
|
|
||||||
|
|
||||||
// When multiplexing mail, explicitly include To/Cc information in the
|
|
||||||
// message body and headers.
|
|
||||||
|
|
||||||
$mail_template = clone $mail_template;
|
|
||||||
|
|
||||||
$mail_template->addPHIDHeaders('X-Phabricator-To', array_keys($tos));
|
|
||||||
$mail_template->addPHIDHeaders('X-Phabricator-Cc', array_keys($ccs));
|
|
||||||
|
|
||||||
$body = $mail_template->getBody();
|
|
||||||
$body .= "\n";
|
|
||||||
$body .= $this->getRecipientsSummary($to_handles, $cc_handles);
|
|
||||||
|
|
||||||
$html_body = $mail_template->getHTMLBody();
|
|
||||||
if (strlen($html_body)) {
|
|
||||||
$html_body .= hsprintf('%s',
|
|
||||||
$this->getRecipientsSummaryHTML($to_handles, $cc_handles));
|
|
||||||
}
|
|
||||||
|
|
||||||
foreach ($recipients as $phid => $recipient) {
|
|
||||||
|
|
||||||
$mail = clone $mail_template;
|
|
||||||
if (isset($to_handles[$phid])) {
|
|
||||||
$mail->addTos(array($phid));
|
|
||||||
} else if (isset($cc_handles[$phid])) {
|
|
||||||
$mail->addCCs(array($phid));
|
|
||||||
} else {
|
|
||||||
// not good - they should be a to or a cc
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
$mail->setBody($body);
|
|
||||||
$mail->setHTMLBody($html_body);
|
|
||||||
|
|
||||||
$reply_to = null;
|
|
||||||
if (!$reply_to && $this->supportsPrivateReplies()) {
|
|
||||||
$reply_to = $this->getPrivateReplyHandlerEmailAddress($recipient);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!$reply_to && $this->supportsPublicReplies()) {
|
|
||||||
$reply_to = $this->getPublicReplyHandlerEmailAddress();
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($reply_to) {
|
|
||||||
$mail->setReplyTo($reply_to);
|
|
||||||
}
|
|
||||||
|
|
||||||
$result[] = $mail;
|
|
||||||
}
|
|
||||||
|
|
||||||
return $result;
|
|
||||||
}
|
|
||||||
|
|
||||||
protected function getDefaultPublicReplyHandlerEmailAddress($prefix) {
|
protected function getDefaultPublicReplyHandlerEmailAddress($prefix) {
|
||||||
|
|
||||||
$receiver = $this->getMailReceiver();
|
$receiver = $this->getMailReceiver();
|
||||||
|
@ -288,31 +143,15 @@ abstract class PhabricatorMailReplyHandler {
|
||||||
}
|
}
|
||||||
|
|
||||||
protected function getDefaultPrivateReplyHandlerEmailAddress(
|
protected function getDefaultPrivateReplyHandlerEmailAddress(
|
||||||
PhabricatorObjectHandle $handle,
|
PhabricatorUser $user,
|
||||||
$prefix) {
|
$prefix) {
|
||||||
|
|
||||||
if ($handle->getType() != PhabricatorPeopleUserPHIDType::TYPECONST) {
|
|
||||||
// You must be a real user to get a private reply handler address.
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
$user = id(new PhabricatorPeopleQuery())
|
|
||||||
->setViewer(PhabricatorUser::getOmnipotentUser())
|
|
||||||
->withPHIDs(array($handle->getPHID()))
|
|
||||||
->executeOne();
|
|
||||||
|
|
||||||
if (!$user) {
|
|
||||||
// This may happen if a user was subscribed to something, and was then
|
|
||||||
// deleted.
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
$receiver = $this->getMailReceiver();
|
$receiver = $this->getMailReceiver();
|
||||||
$receiver_id = $receiver->getID();
|
$receiver_id = $receiver->getID();
|
||||||
$user_id = $user->getID();
|
$user_id = $user->getID();
|
||||||
$hash = PhabricatorObjectMailReceiver::computeMailHash(
|
$hash = PhabricatorObjectMailReceiver::computeMailHash(
|
||||||
$receiver->getMailKey(),
|
$receiver->getMailKey(),
|
||||||
$handle->getPHID());
|
$user->getPHID());
|
||||||
$domain = $this->getReplyHandlerDomain();
|
$domain = $this->getReplyHandlerDomain();
|
||||||
|
|
||||||
$address = "{$prefix}{$receiver_id}+{$user_id}+{$hash}@{$domain}";
|
$address = "{$prefix}{$receiver_id}+{$user_id}+{$hash}@{$domain}";
|
||||||
|
@ -368,21 +207,188 @@ abstract class PhabricatorMailReplyHandler {
|
||||||
return rtrim($output);
|
return rtrim($output);
|
||||||
}
|
}
|
||||||
|
|
||||||
private function expandRecipientHandles(array $handles) {
|
|
||||||
if (!$handles) {
|
/**
|
||||||
|
* Produce a list of mail targets for a given to/cc list.
|
||||||
|
*
|
||||||
|
* Each target should be sent a separate email, and contains the information
|
||||||
|
* required to generate it with appropriate permissions and configuration.
|
||||||
|
*
|
||||||
|
* @param list<phid> List of "To" PHIDs.
|
||||||
|
* @param list<phid> List of "CC" PHIDs.
|
||||||
|
* @return list<PhabricatorMailTarget> List of targets.
|
||||||
|
*/
|
||||||
|
final public function getMailTargets(array $raw_to, array $raw_cc) {
|
||||||
|
list($to, $cc) = $this->expandRecipientPHIDs($raw_to, $raw_cc);
|
||||||
|
list($to, $cc) = $this->loadRecipientUsers($to, $cc);
|
||||||
|
list($to, $cc) = $this->filterRecipientUsers($to, $cc);
|
||||||
|
|
||||||
|
if (!$to && !$cc) {
|
||||||
return array();
|
return array();
|
||||||
}
|
}
|
||||||
|
|
||||||
$phids = mpull($handles, 'getPHID');
|
$template = id(new PhabricatorMailTarget())
|
||||||
$results = id(new PhabricatorMetaMTAMemberQuery())
|
->setRawToPHIDs($raw_to)
|
||||||
->setViewer(PhabricatorUser::getOmnipotentUser())
|
->setRawCCPHIDs($raw_cc);
|
||||||
->withPHIDs($phids)
|
|
||||||
->executeExpansion();
|
|
||||||
|
|
||||||
return id(new PhabricatorHandleQuery())
|
// Set the public reply address as the default, if one exists. We
|
||||||
->setViewer(PhabricatorUser::getOmnipotentUser())
|
// might replace this with a private address later.
|
||||||
->withPHIDs($results)
|
if ($this->supportsPublicReplies()) {
|
||||||
->execute();
|
$reply_to = $this->getPublicReplyHandlerEmailAddress();
|
||||||
|
if ($reply_to) {
|
||||||
|
$template->setReplyTo($reply_to);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$supports_private_replies = $this->supportsPrivateReplies();
|
||||||
|
$mail_all = !PhabricatorEnv::getEnvConfig('metamta.one-mail-per-recipient');
|
||||||
|
$targets = array();
|
||||||
|
if ($mail_all) {
|
||||||
|
$target = id(clone $template)
|
||||||
|
->setViewer(PhabricatorUser::getOmnipotentUser())
|
||||||
|
->setToMap($to)
|
||||||
|
->setCCMap($cc);
|
||||||
|
|
||||||
|
$targets[] = $target;
|
||||||
|
} else {
|
||||||
|
$map = $to + $cc;
|
||||||
|
|
||||||
|
foreach ($map as $phid => $user) {
|
||||||
|
$target = id(clone $template)
|
||||||
|
->setViewer($user)
|
||||||
|
->setToMap(array($phid => $user))
|
||||||
|
->setCCMap(array());
|
||||||
|
|
||||||
|
if ($supports_private_replies) {
|
||||||
|
$reply_to = $this->getPrivateReplyHandlerEmailAddress($user);
|
||||||
|
if ($reply_to) {
|
||||||
|
$target->setReplyTo($reply_to);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$targets[] = $target;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $targets;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Expand lists of recipient PHIDs.
|
||||||
|
*
|
||||||
|
* This takes any compound recipients (like projects) and looks up all their
|
||||||
|
* members.
|
||||||
|
*
|
||||||
|
* @param list<phid> List of To PHIDs.
|
||||||
|
* @param list<phid> List of CC PHIDs.
|
||||||
|
* @return pair<list<phid>, list<phid>> Expanded PHID lists.
|
||||||
|
*/
|
||||||
|
private function expandRecipientPHIDs(array $to, array $cc) {
|
||||||
|
$to_result = array();
|
||||||
|
$cc_result = array();
|
||||||
|
|
||||||
|
$all_phids = array_merge($to, $cc);
|
||||||
|
if ($all_phids) {
|
||||||
|
$map = id(new PhabricatorMetaMTAMemberQuery())
|
||||||
|
->setViewer(PhabricatorUser::getOmnipotentUser())
|
||||||
|
->withPHIDs($all_phids)
|
||||||
|
->execute();
|
||||||
|
foreach ($to as $phid) {
|
||||||
|
foreach ($map[$phid] as $expanded) {
|
||||||
|
$to_result[$expanded] = $expanded;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
foreach ($cc as $phid) {
|
||||||
|
foreach ($map[$phid] as $expanded) {
|
||||||
|
$cc_result[$expanded] = $expanded;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Remove recipients from "CC" if they're also present in "To".
|
||||||
|
$cc_result = array_diff_key($cc_result, $to_result);
|
||||||
|
|
||||||
|
return array(array_values($to_result), array_values($cc_result));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Load @{class:PhabricatorUser} objects for each recipient.
|
||||||
|
*
|
||||||
|
* Invalid recipients are dropped from the results.
|
||||||
|
*
|
||||||
|
* @param list<phid> List of To PHIDs.
|
||||||
|
* @param list<phid> List of CC PHIDs.
|
||||||
|
* @return pair<wild, wild> Maps from PHIDs to users.
|
||||||
|
*/
|
||||||
|
private function loadRecipientUsers(array $to, array $cc) {
|
||||||
|
$to_result = array();
|
||||||
|
$cc_result = array();
|
||||||
|
|
||||||
|
$all_phids = array_merge($to, $cc);
|
||||||
|
if ($all_phids) {
|
||||||
|
$users = id(new PhabricatorPeopleQuery())
|
||||||
|
->setViewer(PhabricatorUser::getOmnipotentUser())
|
||||||
|
->withPHIDs($all_phids)
|
||||||
|
->execute();
|
||||||
|
$users = mpull($users, null, 'getPHID');
|
||||||
|
|
||||||
|
foreach ($to as $phid) {
|
||||||
|
if (isset($users[$phid])) {
|
||||||
|
$to_result[$phid] = $users[$phid];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
foreach ($cc as $phid) {
|
||||||
|
if (isset($users[$phid])) {
|
||||||
|
$cc_result[$phid] = $users[$phid];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return array($to_result, $cc_result);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Remove recipients who do not have permission to view the mail receiver.
|
||||||
|
*
|
||||||
|
* @param map<string, PhabricatorUser> Map of "To" users.
|
||||||
|
* @param map<string, PhabricatorUser> Map of "CC" users.
|
||||||
|
* @return pair<wild, wild> Filtered user maps.
|
||||||
|
*/
|
||||||
|
private function filterRecipientUsers(array $to, array $cc) {
|
||||||
|
$to_result = array();
|
||||||
|
$cc_result = array();
|
||||||
|
|
||||||
|
$all_users = $to + $cc;
|
||||||
|
if ($all_users) {
|
||||||
|
$can_see = array();
|
||||||
|
$object = $this->getMailReceiver();
|
||||||
|
foreach ($all_users as $phid => $user) {
|
||||||
|
$visible = PhabricatorPolicyFilter::hasCapability(
|
||||||
|
$user,
|
||||||
|
$object,
|
||||||
|
PhabricatorPolicyCapability::CAN_VIEW);
|
||||||
|
if ($visible) {
|
||||||
|
$can_see[$phid] = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach ($to as $phid => $user) {
|
||||||
|
if (!empty($can_see[$phid])) {
|
||||||
|
$to_result[$phid] = $all_users[$phid];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach ($cc as $phid => $user) {
|
||||||
|
if (!empty($can_see[$phid])) {
|
||||||
|
$cc_result[$phid] = $all_users[$phid];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return array($to_result, $cc_result);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
146
src/applications/metamta/replyhandler/PhabricatorMailTarget.php
Normal file
146
src/applications/metamta/replyhandler/PhabricatorMailTarget.php
Normal file
|
@ -0,0 +1,146 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
final class PhabricatorMailTarget extends Phobject {
|
||||||
|
|
||||||
|
private $viewer;
|
||||||
|
private $replyTo;
|
||||||
|
private $toMap = array();
|
||||||
|
private $ccMap = array();
|
||||||
|
private $rawToPHIDs;
|
||||||
|
private $rawCCPHIDs;
|
||||||
|
|
||||||
|
public function setRawToPHIDs(array $to_phids) {
|
||||||
|
$this->rawToPHIDs = $to_phids;
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function setRawCCPHIDs(array $cc_phids) {
|
||||||
|
$this->rawCCPHIDs = $cc_phids;
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function setCCMap(array $cc_map) {
|
||||||
|
$this->ccMap = $cc_map;
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getCCMap() {
|
||||||
|
return $this->ccMap;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function setToMap(array $to_map) {
|
||||||
|
$this->toMap = $to_map;
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getToMap() {
|
||||||
|
return $this->toMap;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function setReplyTo($reply_to) {
|
||||||
|
$this->replyTo = $reply_to;
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getReplyTo() {
|
||||||
|
return $this->replyTo;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function setViewer($viewer) {
|
||||||
|
$this->viewer = $viewer;
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getViewer() {
|
||||||
|
return $this->viewer;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function sendMail(PhabricatorMetaMTAMail $mail) {
|
||||||
|
$viewer = $this->getViewer();
|
||||||
|
|
||||||
|
$mail->addPHIDHeaders('X-Phabricator-To', $this->rawToPHIDs);
|
||||||
|
$mail->addPHIDHeaders('X-Phabricator-Cc', $this->rawCCPHIDs);
|
||||||
|
|
||||||
|
$to_handles = $viewer->loadHandles($this->rawToPHIDs);
|
||||||
|
$cc_handles = $viewer->loadHandles($this->rawCCPHIDs);
|
||||||
|
|
||||||
|
$body = $mail->getBody();
|
||||||
|
$body .= "\n";
|
||||||
|
$body .= $this->getRecipientsSummary($to_handles, $cc_handles);
|
||||||
|
$mail->setBody($body);
|
||||||
|
|
||||||
|
$html_body = $mail->getHTMLBody();
|
||||||
|
if (strlen($html_body)) {
|
||||||
|
$html_body .= hsprintf(
|
||||||
|
'%s',
|
||||||
|
$this->getRecipientsSummaryHTML($to_handles, $cc_handles));
|
||||||
|
}
|
||||||
|
$mail->setHTMLBody($html_body);
|
||||||
|
|
||||||
|
$reply_to = $this->getReplyTo();
|
||||||
|
if ($reply_to) {
|
||||||
|
$mail->setReplyTo($reply_to);
|
||||||
|
}
|
||||||
|
|
||||||
|
$to = array_keys($this->getToMap());
|
||||||
|
if ($to) {
|
||||||
|
$mail->addTos($to);
|
||||||
|
}
|
||||||
|
|
||||||
|
$cc = array_keys($this->getCCMap());
|
||||||
|
if ($cc) {
|
||||||
|
$mail->addCCs($cc);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $mail->save();
|
||||||
|
}
|
||||||
|
|
||||||
|
private function getRecipientsSummary(
|
||||||
|
PhabricatorHandleList $to_handles,
|
||||||
|
PhabricatorHandleList $cc_handles) {
|
||||||
|
|
||||||
|
if (!PhabricatorEnv::getEnvConfig('metamta.recipients.show-hints')) {
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
|
||||||
|
$to_handles = iterator_to_array($to_handles);
|
||||||
|
$cc_handles = iterator_to_array($cc_handles);
|
||||||
|
|
||||||
|
$body = '';
|
||||||
|
if ($to_handles) {
|
||||||
|
$body .= "To: ".implode(', ', mpull($to_handles, 'getName'))."\n";
|
||||||
|
}
|
||||||
|
if ($cc_handles) {
|
||||||
|
$body .= "Cc: ".implode(', ', mpull($cc_handles, 'getName'))."\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
return $body;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function getRecipientsSummaryHTML(
|
||||||
|
PhabricatorHandleList $to_handles,
|
||||||
|
PhabricatorHandleList $cc_handles) {
|
||||||
|
|
||||||
|
if (!PhabricatorEnv::getEnvConfig('metamta.recipients.show-hints')) {
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
|
||||||
|
$to_handles = iterator_to_array($to_handles);
|
||||||
|
$cc_handles = iterator_to_array($cc_handles);
|
||||||
|
|
||||||
|
$body = array();
|
||||||
|
if ($to_handles) {
|
||||||
|
$body[] = phutil_tag('strong', array(), 'To: ');
|
||||||
|
$body[] = phutil_implode_html(', ', mpull($to_handles, 'getName'));
|
||||||
|
$body[] = phutil_tag('br');
|
||||||
|
}
|
||||||
|
if ($cc_handles) {
|
||||||
|
$body[] = phutil_tag('strong', array(), 'Cc: ');
|
||||||
|
$body[] = phutil_implode_html(', ', mpull($cc_handles, 'getName'));
|
||||||
|
$body[] = phutil_tag('br');
|
||||||
|
}
|
||||||
|
return phutil_tag('div', array(), $body);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
}
|
|
@ -11,7 +11,7 @@ final class OwnersPackageReplyHandler extends PhabricatorMailReplyHandler {
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getPrivateReplyHandlerEmailAddress(
|
public function getPrivateReplyHandlerEmailAddress(
|
||||||
PhabricatorObjectHandle $handle) {
|
PhabricatorUser $user) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -8,7 +8,7 @@ final class PhabricatorRepositoryPushReplyHandler
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getPrivateReplyHandlerEmailAddress(
|
public function getPrivateReplyHandlerEmailAddress(
|
||||||
PhabricatorObjectHandle $handle) {
|
PhabricatorUser $user) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -27,6 +27,23 @@ final class PhabricatorRepositoryPushMailWorker
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$targets = id(new PhabricatorRepositoryPushReplyHandler())
|
||||||
|
->setMailReceiver($repository)
|
||||||
|
->getMailTargets($email_phids, array());
|
||||||
|
foreach ($targets as $target) {
|
||||||
|
$this->sendMail($target, $repository, $event);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private function sendMail(
|
||||||
|
PhabricatorMailTarget $target,
|
||||||
|
PhabricatorRepository $repository,
|
||||||
|
PhabricatorRepositoryPushEvent $event) {
|
||||||
|
|
||||||
|
$task_data = $this->getTaskData();
|
||||||
|
$viewer = $target->getViewer();
|
||||||
|
// TODO: Swap locale to viewer locale.
|
||||||
|
|
||||||
$logs = $event->getLogs();
|
$logs = $event->getLogs();
|
||||||
|
|
||||||
list($ref_lines, $ref_list) = $this->renderRefs($logs);
|
list($ref_lines, $ref_list) = $this->renderRefs($logs);
|
||||||
|
@ -103,20 +120,7 @@ final class PhabricatorRepositoryPushMailWorker
|
||||||
->addHeader('Thread-Topic', $subject)
|
->addHeader('Thread-Topic', $subject)
|
||||||
->setIsBulk(true);
|
->setIsBulk(true);
|
||||||
|
|
||||||
$to_handles = id(new PhabricatorHandleQuery())
|
$target->sendMail($mail);
|
||||||
->setViewer($viewer)
|
|
||||||
->withPHIDs($email_phids)
|
|
||||||
->execute();
|
|
||||||
|
|
||||||
$reply_handler = new PhabricatorRepositoryPushReplyHandler();
|
|
||||||
$mails = $reply_handler->multiplexMail(
|
|
||||||
$mail,
|
|
||||||
$to_handles,
|
|
||||||
array());
|
|
||||||
|
|
||||||
foreach ($mails as $mail) {
|
|
||||||
$mail->saveAndSend();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public function renderForDisplay(PhabricatorUser $viewer) {
|
public function renderForDisplay(PhabricatorUser $viewer) {
|
||||||
|
|
|
@ -992,12 +992,10 @@ abstract class PhabricatorApplicationTransactionEditor
|
||||||
// Hook for other edges that may need (re-)loading
|
// Hook for other edges that may need (re-)loading
|
||||||
$object = $this->willPublish($object, $xactions);
|
$object = $this->willPublish($object, $xactions);
|
||||||
|
|
||||||
$this->loadHandles($xactions);
|
$mailed = array();
|
||||||
|
|
||||||
$mail = null;
|
|
||||||
if (!$this->getDisableEmail()) {
|
if (!$this->getDisableEmail()) {
|
||||||
if ($this->shouldSendMail($object, $xactions)) {
|
if ($this->shouldSendMail($object, $xactions)) {
|
||||||
$mail = $this->sendMail($object, $xactions);
|
$mailed = $this->sendMail($object, $xactions);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1009,10 +1007,6 @@ abstract class PhabricatorApplicationTransactionEditor
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($this->shouldPublishFeedStory($object, $xactions)) {
|
if ($this->shouldPublishFeedStory($object, $xactions)) {
|
||||||
$mailed = array();
|
|
||||||
if ($mail) {
|
|
||||||
$mailed = $mail->buildRecipientList();
|
|
||||||
}
|
|
||||||
$this->publishFeedStory(
|
$this->publishFeedStory(
|
||||||
$object,
|
$object,
|
||||||
$xactions,
|
$xactions,
|
||||||
|
@ -2128,30 +2122,60 @@ abstract class PhabricatorApplicationTransactionEditor
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!$any_visible) {
|
if (!$any_visible) {
|
||||||
return;
|
return array();
|
||||||
}
|
}
|
||||||
|
|
||||||
$email_force = array();
|
|
||||||
$email_to = $this->mailToPHIDs;
|
$email_to = $this->mailToPHIDs;
|
||||||
$email_cc = $this->mailCCPHIDs;
|
$email_cc = $this->mailCCPHIDs;
|
||||||
|
|
||||||
$email_cc = array_merge($email_cc, $this->heraldEmailPHIDs);
|
$email_cc = array_merge($email_cc, $this->heraldEmailPHIDs);
|
||||||
$email_force = $this->heraldForcedEmailPHIDs;
|
|
||||||
|
|
||||||
$phids = array_merge($email_to, $email_cc);
|
$targets = $this->buildReplyHandler($object)
|
||||||
$handles = id(new PhabricatorHandleQuery())
|
->getMailTargets($email_to, $email_cc);
|
||||||
->setViewer($this->requireActor())
|
|
||||||
->withPHIDs($phids)
|
|
||||||
->execute();
|
|
||||||
|
|
||||||
$template = $this->buildMailTemplate($object);
|
// Set this explicitly before we start swapping out the effective actor.
|
||||||
|
$this->setActingAsPHID($this->getActingAsPHID());
|
||||||
|
|
||||||
|
|
||||||
|
$mailed = array();
|
||||||
|
foreach ($targets as $target) {
|
||||||
|
$original_actor = $this->getActor();
|
||||||
|
$this->setActor($target->getViewer());
|
||||||
|
// TODO: Swap locale to viewer locale.
|
||||||
|
|
||||||
|
$caught = null;
|
||||||
|
try {
|
||||||
|
// Reload handles for the new viewer.
|
||||||
|
$this->loadHandles($xactions);
|
||||||
|
|
||||||
|
$mail = $this->sendMailToTarget($object, $xactions, $target);
|
||||||
|
} catch (Exception $ex) {
|
||||||
|
$caught = $ex;
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->setActor($original_actor);
|
||||||
|
if ($caught) {
|
||||||
|
throw $ex;
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach ($mail->buildRecipientList() as $phid) {
|
||||||
|
$mailed[$phid] = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return array_keys($mailed);
|
||||||
|
}
|
||||||
|
|
||||||
|
private function sendMailToTarget(
|
||||||
|
PhabricatorLiskDAO $object,
|
||||||
|
array $xactions,
|
||||||
|
PhabricatorMailTarget $target) {
|
||||||
|
|
||||||
|
$mail = $this->buildMailTemplate($object);
|
||||||
$body = $this->buildMailBody($object, $xactions);
|
$body = $this->buildMailBody($object, $xactions);
|
||||||
|
|
||||||
$mail_tags = $this->getMailTags($object, $xactions);
|
$mail_tags = $this->getMailTags($object, $xactions);
|
||||||
$action = $this->getMailAction($object, $xactions);
|
$action = $this->getMailAction($object, $xactions);
|
||||||
|
|
||||||
$reply_handler = $this->buildReplyHandler($object);
|
|
||||||
|
|
||||||
if (PhabricatorEnv::getEnvConfig('metamta.email-preferences')) {
|
if (PhabricatorEnv::getEnvConfig('metamta.email-preferences')) {
|
||||||
$this->addEmailPreferenceSectionToMailBody(
|
$this->addEmailPreferenceSectionToMailBody(
|
||||||
$body,
|
$body,
|
||||||
|
@ -2159,48 +2183,36 @@ abstract class PhabricatorApplicationTransactionEditor
|
||||||
$xactions);
|
$xactions);
|
||||||
}
|
}
|
||||||
|
|
||||||
$template
|
$mail
|
||||||
->setFrom($this->getActingAsPHID())
|
->setFrom($this->getActingAsPHID())
|
||||||
->setSubjectPrefix($this->getMailSubjectPrefix())
|
->setSubjectPrefix($this->getMailSubjectPrefix())
|
||||||
->setVarySubjectPrefix('['.$action.']')
|
->setVarySubjectPrefix('['.$action.']')
|
||||||
->setThreadID($this->getMailThreadID($object), $this->getIsNewObject())
|
->setThreadID($this->getMailThreadID($object), $this->getIsNewObject())
|
||||||
->setRelatedPHID($object->getPHID())
|
->setRelatedPHID($object->getPHID())
|
||||||
->setExcludeMailRecipientPHIDs($this->getExcludeMailRecipientPHIDs())
|
->setExcludeMailRecipientPHIDs($this->getExcludeMailRecipientPHIDs())
|
||||||
->setForceHeraldMailRecipientPHIDs($email_force)
|
->setForceHeraldMailRecipientPHIDs($this->heraldForcedEmailPHIDs)
|
||||||
->setMailTags($mail_tags)
|
->setMailTags($mail_tags)
|
||||||
->setIsBulk(true)
|
->setIsBulk(true)
|
||||||
->setBody($body->render())
|
->setBody($body->render())
|
||||||
->setHTMLBody($body->renderHTML());
|
->setHTMLBody($body->renderHTML());
|
||||||
|
|
||||||
foreach ($body->getAttachments() as $attachment) {
|
foreach ($body->getAttachments() as $attachment) {
|
||||||
$template->addAttachment($attachment);
|
$mail->addAttachment($attachment);
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($this->heraldHeader) {
|
if ($this->heraldHeader) {
|
||||||
$template->addHeader('X-Herald-Rules', $this->heraldHeader);
|
$mail->addHeader('X-Herald-Rules', $this->heraldHeader);
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($object instanceof PhabricatorProjectInterface) {
|
if ($object instanceof PhabricatorProjectInterface) {
|
||||||
$this->addMailProjectMetadata($object, $template);
|
$this->addMailProjectMetadata($object, $mail);
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($this->getParentMessageID()) {
|
if ($this->getParentMessageID()) {
|
||||||
$template->setParentMessageID($this->getParentMessageID());
|
$mail->setParentMessageID($this->getParentMessageID());
|
||||||
}
|
}
|
||||||
|
|
||||||
$mails = $reply_handler->multiplexMail(
|
return $target->sendMail($mail);
|
||||||
$template,
|
|
||||||
array_select_keys($handles, $email_to),
|
|
||||||
array_select_keys($handles, $email_cc));
|
|
||||||
|
|
||||||
foreach ($mails as $mail) {
|
|
||||||
$mail->saveAndSend();
|
|
||||||
}
|
|
||||||
|
|
||||||
$template->addTos($email_to);
|
|
||||||
$template->addCCs($email_cc);
|
|
||||||
|
|
||||||
return $template;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private function addMailProjectMetadata(
|
private function addMailProjectMetadata(
|
||||||
|
|
|
@ -6,9 +6,9 @@ abstract class PhabricatorApplicationTransactionReplyHandler
|
||||||
abstract public function getObjectPrefix();
|
abstract public function getObjectPrefix();
|
||||||
|
|
||||||
public function getPrivateReplyHandlerEmailAddress(
|
public function getPrivateReplyHandlerEmailAddress(
|
||||||
PhabricatorObjectHandle $handle) {
|
PhabricatorUser $user) {
|
||||||
return $this->getDefaultPrivateReplyHandlerEmailAddress(
|
return $this->getDefaultPrivateReplyHandlerEmailAddress(
|
||||||
$handle,
|
$user,
|
||||||
$this->getObjectPrefix());
|
$this->getObjectPrefix());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Add table
Reference in a new issue