2011-01-25 17:40:21 -08:00
|
|
|
<?php
|
|
|
|
|
|
|
|
/*
|
Add email preferences to receive fewer less-important notifications
Summary:
A few similar requests have come in across several tools and use cases that I
think this does a reasonable job of resolving.
We currently send one email for each update an object receives, but these aren't
always appreciated:
- Asana does post-commit review via Differential, so the "committed" mails are
useless.
- Quora wants to make project category edits to bugs without spamming people
attached to them.
- Some users in general are very sensitive to email volumes, and this gives us
a good way to reduce the volumes without incurring the complexity of
delayed-send-batching.
The technical mechanism is basically:
- Mail may optionally have "mail tags", which indicate content in the mail
(e.g., "maniphest-priority, maniphest-cc, maniphest-comment" for a mail which
contains a priority change, a CC change, and a comment).
- If a mail has tags, remove any recipients who have opted out of all the
tags.
- Some tags can't be opted out of via the UI, so this ensures that important
email is still delivered (e.g., cc + assign + comment is always delivered
because you can't opt out of "assign" or "comment").
Test Plan:
- Disabled all mail tags in the web UI.
- Used test console to send myself mail with an opt-outable tag, it was
immediately dropped.
- Used test console to send myself mail with an opt-outable tag and a custom
tag, it was delivered.
- Made Differential updates affecting CCs with and without comments, got
appropriate delivery.
- Made Maniphest updates affecting project, priority and CCs with and without
comments, got appropriate delivery.
- Verified mail headers in all cases.
Reviewers: btrahan
Reviewed By: btrahan
CC: aran, epriestley, moskov
Maniphest Tasks: T616, T855
Differential Revision: https://secure.phabricator.com/D1635
2012-02-17 22:57:07 -08:00
|
|
|
* Copyright 2012 Facebook, Inc.
|
2011-01-25 17:40:21 -08:00
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
|
|
|
|
2012-03-09 15:46:25 -08:00
|
|
|
final class PhabricatorMetaMTASendController
|
|
|
|
extends PhabricatorMetaMTAController {
|
2011-01-25 17:40:21 -08:00
|
|
|
|
|
|
|
public function processRequest() {
|
|
|
|
|
|
|
|
$request = $this->getRequest();
|
|
|
|
|
|
|
|
if ($request->isFormPost()) {
|
2011-11-08 14:14:00 -08:00
|
|
|
|
2011-01-25 17:40:21 -08:00
|
|
|
$mail = new PhabricatorMetaMTAMail();
|
|
|
|
$mail->addTos($request->getArr('to'));
|
|
|
|
$mail->addCCs($request->getArr('cc'));
|
|
|
|
$mail->setSubject($request->getStr('subject'));
|
|
|
|
$mail->setBody($request->getStr('body'));
|
|
|
|
|
2011-11-08 14:14:00 -08:00
|
|
|
$files = $request->getArr('files');
|
|
|
|
if ($files) {
|
|
|
|
foreach ($files as $phid) {
|
|
|
|
$file = id(new PhabricatorFile())->loadOneWhere('phid = %s', $phid);
|
2011-11-08 15:15:44 -08:00
|
|
|
$mail->addAttachment(new PhabricatorMetaMTAAttachment(
|
2011-11-08 14:14:00 -08:00
|
|
|
$file->loadFileData(),
|
|
|
|
$file->getName(),
|
2011-11-08 15:15:44 -08:00
|
|
|
$file->getMimeType()
|
|
|
|
));
|
2011-11-08 14:14:00 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-01-26 16:02:12 -08:00
|
|
|
$mail->setFrom($request->getUser()->getPHID());
|
2011-01-25 17:40:21 -08:00
|
|
|
$mail->setSimulatedFailureCount($request->getInt('failures'));
|
|
|
|
$mail->setIsHTML($request->getInt('html'));
|
2011-10-23 12:07:37 -07:00
|
|
|
$mail->setIsBulk($request->getInt('bulk'));
|
Add email preferences to receive fewer less-important notifications
Summary:
A few similar requests have come in across several tools and use cases that I
think this does a reasonable job of resolving.
We currently send one email for each update an object receives, but these aren't
always appreciated:
- Asana does post-commit review via Differential, so the "committed" mails are
useless.
- Quora wants to make project category edits to bugs without spamming people
attached to them.
- Some users in general are very sensitive to email volumes, and this gives us
a good way to reduce the volumes without incurring the complexity of
delayed-send-batching.
The technical mechanism is basically:
- Mail may optionally have "mail tags", which indicate content in the mail
(e.g., "maniphest-priority, maniphest-cc, maniphest-comment" for a mail which
contains a priority change, a CC change, and a comment).
- If a mail has tags, remove any recipients who have opted out of all the
tags.
- Some tags can't be opted out of via the UI, so this ensures that important
email is still delivered (e.g., cc + assign + comment is always delivered
because you can't opt out of "assign" or "comment").
Test Plan:
- Disabled all mail tags in the web UI.
- Used test console to send myself mail with an opt-outable tag, it was
immediately dropped.
- Used test console to send myself mail with an opt-outable tag and a custom
tag, it was delivered.
- Made Differential updates affecting CCs with and without comments, got
appropriate delivery.
- Made Maniphest updates affecting project, priority and CCs with and without
comments, got appropriate delivery.
- Verified mail headers in all cases.
Reviewers: btrahan
Reviewed By: btrahan
CC: aran, epriestley, moskov
Maniphest Tasks: T616, T855
Differential Revision: https://secure.phabricator.com/D1635
2012-02-17 22:57:07 -08:00
|
|
|
$mail->setMailTags($request->getStrList('mailtags'));
|
2011-01-25 17:40:21 -08:00
|
|
|
$mail->save();
|
2011-04-30 22:51:25 -07:00
|
|
|
if ($request->getInt('immediately')) {
|
|
|
|
$mail->sendNow();
|
2011-01-25 17:40:21 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
return id(new AphrontRedirectResponse())
|
2012-08-13 12:37:06 -07:00
|
|
|
->setURI($this->getApplicationURI('/view/'.$mail->getID().'/'));
|
2011-01-25 17:40:21 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
$failure_caption =
|
|
|
|
"Enter a number to simulate that many consecutive send failures before ".
|
|
|
|
"really attempting to deliver via the underlying MTA.";
|
|
|
|
|
2011-06-12 20:16:34 -07:00
|
|
|
$doclink_href = PhabricatorEnv::getDoclink(
|
|
|
|
'article/Configuring_Outbound_Email.html');
|
|
|
|
|
|
|
|
$doclink = phutil_render_tag(
|
|
|
|
'a',
|
|
|
|
array(
|
|
|
|
'href' => $doclink_href,
|
|
|
|
'target' => '_blank',
|
|
|
|
),
|
|
|
|
'Configuring Outbound Email');
|
|
|
|
$instructions =
|
|
|
|
'<p class="aphront-form-instructions">This form will send a normal '.
|
|
|
|
'email using the settings you have configured for Phabricator. For more '.
|
|
|
|
'information, see '.$doclink.'.</p>';
|
|
|
|
|
|
|
|
$adapter = PhabricatorEnv::getEnvConfig('metamta.mail-adapter');
|
|
|
|
$warning = null;
|
|
|
|
if ($adapter == 'PhabricatorMailImplementationTestAdapter') {
|
|
|
|
$warning = new AphrontErrorView();
|
|
|
|
$warning->setTitle('Email is Disabled');
|
|
|
|
$warning->setSeverity(AphrontErrorView::SEVERITY_WARNING);
|
|
|
|
$warning->appendChild(
|
|
|
|
'<p>This installation of Phabricator is currently set to use '.
|
|
|
|
'<tt>PhabricatorMailImplementationTestAdapter</tt> to deliver '.
|
|
|
|
'outbound email. This completely disables outbound email! All '.
|
|
|
|
'outbound email will be thrown in a deep, dark hole until you '.
|
|
|
|
'configure a real adapter.</p>');
|
|
|
|
}
|
2011-01-25 17:40:21 -08:00
|
|
|
|
2011-11-08 14:14:00 -08:00
|
|
|
$panel_id = celerity_generate_unique_node_id();
|
|
|
|
|
2011-01-25 17:40:21 -08:00
|
|
|
$form = new AphrontFormView();
|
2011-01-30 18:52:29 -08:00
|
|
|
$form->setUser($request->getUser());
|
2011-01-25 17:40:21 -08:00
|
|
|
$form
|
2011-06-12 20:16:34 -07:00
|
|
|
->appendChild($instructions)
|
2011-01-25 17:40:21 -08:00
|
|
|
->appendChild(
|
2011-06-12 20:16:34 -07:00
|
|
|
id(new AphrontFormStaticControl())
|
2012-08-13 12:37:06 -07:00
|
|
|
->setLabel('Adapter')
|
2011-06-12 20:16:34 -07:00
|
|
|
->setValue($adapter))
|
2011-01-25 17:40:21 -08:00
|
|
|
->appendChild(
|
|
|
|
id(new AphrontFormTokenizerControl())
|
|
|
|
->setLabel('To')
|
|
|
|
->setName('to')
|
2011-01-26 10:40:38 -08:00
|
|
|
->setDatasource('/typeahead/common/mailable/'))
|
2011-01-25 17:40:21 -08:00
|
|
|
->appendChild(
|
|
|
|
id(new AphrontFormTokenizerControl())
|
|
|
|
->setLabel('CC')
|
|
|
|
->setName('cc')
|
2011-01-26 10:40:38 -08:00
|
|
|
->setDatasource('/typeahead/common/mailable/'))
|
2011-01-25 17:40:21 -08:00
|
|
|
->appendChild(
|
|
|
|
id(new AphrontFormTextControl())
|
|
|
|
->setLabel('Subject')
|
|
|
|
->setName('subject'))
|
|
|
|
->appendChild(
|
|
|
|
id(new AphrontFormTextAreaControl())
|
|
|
|
->setLabel('Body')
|
|
|
|
->setName('body'))
|
Add email preferences to receive fewer less-important notifications
Summary:
A few similar requests have come in across several tools and use cases that I
think this does a reasonable job of resolving.
We currently send one email for each update an object receives, but these aren't
always appreciated:
- Asana does post-commit review via Differential, so the "committed" mails are
useless.
- Quora wants to make project category edits to bugs without spamming people
attached to them.
- Some users in general are very sensitive to email volumes, and this gives us
a good way to reduce the volumes without incurring the complexity of
delayed-send-batching.
The technical mechanism is basically:
- Mail may optionally have "mail tags", which indicate content in the mail
(e.g., "maniphest-priority, maniphest-cc, maniphest-comment" for a mail which
contains a priority change, a CC change, and a comment).
- If a mail has tags, remove any recipients who have opted out of all the
tags.
- Some tags can't be opted out of via the UI, so this ensures that important
email is still delivered (e.g., cc + assign + comment is always delivered
because you can't opt out of "assign" or "comment").
Test Plan:
- Disabled all mail tags in the web UI.
- Used test console to send myself mail with an opt-outable tag, it was
immediately dropped.
- Used test console to send myself mail with an opt-outable tag and a custom
tag, it was delivered.
- Made Differential updates affecting CCs with and without comments, got
appropriate delivery.
- Made Maniphest updates affecting project, priority and CCs with and without
comments, got appropriate delivery.
- Verified mail headers in all cases.
Reviewers: btrahan
Reviewed By: btrahan
CC: aran, epriestley, moskov
Maniphest Tasks: T616, T855
Differential Revision: https://secure.phabricator.com/D1635
2012-02-17 22:57:07 -08:00
|
|
|
->appendChild(
|
|
|
|
id(new AphrontFormTextControl())
|
|
|
|
->setLabel('Mail Tags')
|
|
|
|
->setName('mailtags')
|
|
|
|
->setCaption(
|
|
|
|
'Example: <tt>differential-cc, differential-comment</tt>'))
|
2011-11-08 14:14:00 -08:00
|
|
|
->appendChild(
|
|
|
|
id(new AphrontFormDragAndDropUploadControl())
|
|
|
|
->setLabel('Attach Files')
|
|
|
|
->setName('files')
|
|
|
|
->setDragAndDropTarget($panel_id)
|
|
|
|
->setActivatedClass('aphront-panel-view-drag-and-drop'))
|
2011-01-25 17:40:21 -08:00
|
|
|
->appendChild(
|
|
|
|
id(new AphrontFormTextControl())
|
|
|
|
->setLabel('Simulate Failures')
|
|
|
|
->setName('failures')
|
|
|
|
->setCaption($failure_caption))
|
|
|
|
->appendChild(
|
|
|
|
id(new AphrontFormCheckboxControl())
|
|
|
|
->setLabel('HTML')
|
|
|
|
->addCheckbox('html', '1', 'Send as HTML email.'))
|
2011-10-23 12:07:37 -07:00
|
|
|
->appendChild(
|
|
|
|
id(new AphrontFormCheckboxControl())
|
|
|
|
->setLabel('Bulk')
|
|
|
|
->addCheckbox('bulk', '1', 'Send with bulk email headers.'))
|
2011-01-25 17:40:21 -08:00
|
|
|
->appendChild(
|
|
|
|
id(new AphrontFormCheckboxControl())
|
|
|
|
->setLabel('Send Now')
|
|
|
|
->addCheckbox(
|
|
|
|
'immediately',
|
|
|
|
'1',
|
|
|
|
'Send immediately, not via MetaMTA background script.'))
|
|
|
|
->appendChild(
|
|
|
|
id(new AphrontFormSubmitControl())
|
|
|
|
->setValue('Send Mail'));
|
|
|
|
|
|
|
|
$panel = new AphrontPanelView();
|
|
|
|
$panel->setHeader('Send Email');
|
|
|
|
$panel->appendChild($form);
|
2011-11-08 14:14:00 -08:00
|
|
|
$panel->setID($panel_id);
|
2012-08-13 12:37:06 -07:00
|
|
|
$panel->setWidth(AphrontPanelView::WIDTH_FORM);
|
2011-01-25 17:40:21 -08:00
|
|
|
|
2012-08-13 12:37:06 -07:00
|
|
|
$nav = $this->buildSideNavView();
|
|
|
|
$nav->selectFilter('send');
|
|
|
|
$nav->appendChild(
|
2011-06-12 20:16:34 -07:00
|
|
|
array(
|
|
|
|
$warning,
|
|
|
|
$panel,
|
2012-08-13 12:37:06 -07:00
|
|
|
));
|
|
|
|
|
|
|
|
return $this->buildApplicationPage(
|
|
|
|
$nav,
|
2011-01-25 17:40:21 -08:00
|
|
|
array(
|
2012-08-13 12:37:06 -07:00
|
|
|
'title' => 'Send Test',
|
2011-01-25 17:40:21 -08:00
|
|
|
));
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|