Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Send Test Mail Feature #2023

Merged
merged 32 commits into from
May 31, 2020
Merged
Show file tree
Hide file tree
Changes from 23 commits
Commits
Show all changes
32 commits
Select commit Hold shift + click to select a range
8984554
First working version
askvortsov1 Feb 24, 2020
4483c0c
Added explanatory comment to TemporarySettingsRepository
askvortsov1 Feb 24, 2020
b3e3ae8
Added translation support
askvortsov1 Feb 24, 2020
edb081b
StyleCI Fixes
askvortsov1 Feb 24, 2020
7a13d76
Moved sending test email to API
askvortsov1 Feb 28, 2020
c6586b5
Renamed mail settings endpoint to /mail/settings for consistency
askvortsov1 Feb 28, 2020
ec6aed3
Use email of active user instead of arbitrary email when sending test…
askvortsov1 Feb 28, 2020
19fdf7e
Disable send test mail button while sending to prevent double sending.
askvortsov1 Feb 28, 2020
827e111
Typehint container instead of app.
askvortsov1 Feb 28, 2020
ff0c3a0
Removed unused import from admin routes
askvortsov1 Feb 28, 2020
e71f4c8
Use mithril props for disabling 'send test mail' button while waiting…
askvortsov1 Feb 29, 2020
f2d418c
Set send test mail controller to use saved settings instead of testin…
askvortsov1 Mar 6, 2020
d47732b
Prettify code
askvortsov1 Apr 21, 2020
7344e87
Changed mail settings to new URL
askvortsov1 Apr 21, 2020
166fa9d
Rearranged signature of construct method to follow alphabetical order
askvortsov1 Apr 21, 2020
3b7c64a
De-propify sendingTest on MailPage
askvortsov1 May 12, 2020
fb51415
Removed redundant type: button
askvortsov1 May 12, 2020
22ff44b
settings no longer necessary when sending test mail
askvortsov1 May 12, 2020
6911395
Remove redundant error handling code
askvortsov1 May 12, 2020
e5c21be
Roll back mail settings api endpoint route change, this should be in …
askvortsov1 May 15, 2020
f659234
Simplified JSX a little bit
askvortsov1 May 15, 2020
230a73e
Formatting updates
askvortsov1 May 15, 2020
7986925
One more formatting update
askvortsov1 May 15, 2020
3c97653
Revert "One more formatting update"
askvortsov1 May 23, 2020
dd8f401
Revert "Formatting updates"
askvortsov1 May 23, 2020
3a0a134
Revert "Simplified JSX a little bit"
askvortsov1 May 23, 2020
8a67471
Use JSX for that one FieldSet
askvortsov1 May 23, 2020
98c8ac6
Update conditionals
askvortsov1 May 23, 2020
543d628
Use /mail/settings route after all.
askvortsov1 May 23, 2020
95a3bcd
Revert unnecessary changes
askvortsov1 May 31, 2020
ac6e691
Dismiss alerts so they don't stack up
askvortsov1 May 31, 2020
c7bb324
More prettify
askvortsov1 May 31, 2020
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
145 changes: 88 additions & 57 deletions js/src/admin/components/MailPage.js
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,7 @@ export default class MailPage extends Page {
super.init();

this.saving = false;
this.sendingTest = false;
this.refresh();
}

Expand Down Expand Up @@ -68,65 +69,69 @@ export default class MailPage extends Page {
<h2>{app.translator.trans('core.admin.email.heading')}</h2>
<div className="helpText">{app.translator.trans('core.admin.email.text')}</div>

{FieldSet.component({
label: app.translator.trans('core.admin.email.addresses_heading'),
className: 'MailPage-MailSettings',
children: [
<FieldSet label={app.translator.trans('core.admin.email.addresses_heading')} className="MailPage-MailSettings">
<div className="MailPage-MailSettings-input">
<label>
{app.translator.trans('core.admin.email.from_label')}
<input className="FormControl" value={this.values.mail_from() || ''} oninput={m.withAttr('value', this.values.mail_from)} />
</label>
</div>
</FieldSet>

<FieldSet label={app.translator.trans('core.admin.email.driver_heading')} className="MailPage-MailSettings">
<div className="MailPage-MailSettings-input">
<label>
{app.translator.trans('core.admin.email.driver_label')}
<Select
value={this.values.mail_driver()}
options={Object.keys(this.driverFields).reduce((memo, val) => ({ ...memo, [val]: val }), {})}
onchange={this.values.mail_driver}
/>
</label>
</div>
</FieldSet>

{this.status.sending
askvortsov1 marked this conversation as resolved.
Show resolved Hide resolved
? ''
: Alert.component({
children: app.translator.trans('core.admin.email.not_sending_message'),
dismissible: false,
})}

{fieldKeys.length > 0 ? (
askvortsov1 marked this conversation as resolved.
Show resolved Hide resolved
<FieldSet label={app.translator.trans(`core.admin.email.${this.values.mail_driver()}_heading`)} className="MailPage-MailSettings">
<div className="MailPage-MailSettings-input">
<label>
{app.translator.trans('core.admin.email.from_label')}
<input className="FormControl" value={this.values.mail_from() || ''} oninput={m.withAttr('value', this.values.mail_from)} />
</label>
</div>,
],
})}

{FieldSet.component({
label: app.translator.trans('core.admin.email.driver_heading'),
className: 'MailPage-MailSettings',
children: [
<div className="MailPage-MailSettings-input">
<label>
{app.translator.trans('core.admin.email.driver_label')}
<Select
value={this.values.mail_driver()}
options={Object.keys(this.driverFields).reduce((memo, val) => ({ ...memo, [val]: val }), {})}
onchange={this.values.mail_driver}
/>
</label>
</div>,
],
})}

{this.status.sending ||
Alert.component({
children: app.translator.trans('core.admin.email.not_sending_message'),
dismissible: false,
{fieldKeys.map((field) => [
<label>
{app.translator.trans(`core.admin.email.${field}_label`)}
{this.renderField(field)}
</label>,
this.status.errors[field] && <p className="ValidationError">{this.status.errors[field]}</p>,
])}
</div>
</FieldSet>
) : (
''
)}

<FieldSet>
{Button.component({
type: 'submit',
className: 'Button Button--primary',
children: app.translator.trans('core.admin.email.submit_button'),
disabled: !this.changed(),
})}

{fieldKeys.length > 0 &&
FieldSet.component({
label: app.translator.trans(`core.admin.email.${this.values.mail_driver()}_heading`),
className: 'MailPage-MailSettings',
children: [
<div className="MailPage-MailSettings-input">
{fieldKeys.map((field) => [
<label>
{app.translator.trans(`core.admin.email.${field}_label`)}
{this.renderField(field)}
</label>,
this.status.errors[field] && <p className="ValidationError">{this.status.errors[field]}</p>,
])}
</div>,
],
</FieldSet>

<FieldSet label={app.translator.trans('core.admin.email.send_test_mail_heading')} className="MailPage-MailSettings">
<div className="helpText">{app.translator.trans('core.admin.email.send_test_mail_text', { email: app.session.user.email() })}</div>
{Button.component({
className: 'Button Button--primary',
children: app.translator.trans('core.admin.email.send_test_mail_button'),
disabled: this.sendingTest || this.changed(),
onclick: () => this.sendTestEmail(),
})}

{Button.component({
type: 'submit',
className: 'Button Button--primary',
children: app.translator.trans('core.admin.email.submit_button'),
disabled: !this.changed(),
})}
</FieldSet>
</form>
</div>
</div>
Expand All @@ -149,10 +154,36 @@ export default class MailPage extends Page {
return this.fields.some((key) => this.values[key]() !== app.data.settings[key]);
}

sendTestEmail() {
if (this.saving || this.sendingTest) return;

this.sendingTest = true;

app
.request({
method: 'POST',
url: app.forum.attribute('apiUrl') + '/mail/test',
})
.then((response) => {
this.sendingTest = false;
app.alerts.show(
new Alert({
type: 'success',
children: app.translator.trans('core.admin.email.send_test_mail_success'),
})
);
})
.catch((error) => {
this.sendingTest = false;
m.redraw();
throw error;
});
}

onsubmit(e) {
e.preventDefault();

if (this.saving) return;
if (this.saving || this.sendingTest) return;

this.saving = true;
app.alerts.dismiss(this.successAlert);
Expand Down
53 changes: 53 additions & 0 deletions src/Api/Controller/SendTestMailController.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,53 @@
<?php

/*
* This file is part of Flarum.
*
* For detailed copyright and license information, please view the
* LICENSE file that was distributed with this source code.
*/

namespace Flarum\Api\Controller;

use Flarum\User\AssertPermissionTrait;
use Illuminate\Container\Container;
use Illuminate\Contracts\Mail\Mailer;
use Illuminate\Mail\Message;
use Laminas\Diactoros\Response\EmptyResponse;
use Psr\Http\Message\ResponseInterface;
use Psr\Http\Message\ServerRequestInterface;
use Psr\Http\Server\RequestHandlerInterface;
use Symfony\Component\Translation\TranslatorInterface;

class SendTestMailController implements RequestHandlerInterface
{
use AssertPermissionTrait;

protected $container;

protected $mailer;

protected $translator;

public function __construct(Container $container, Mailer $mailer, TranslatorInterface $translator)
{
$this->container = $container;
$this->mailer = $mailer;
$this->translator = $translator;
}

public function handle(ServerRequestInterface $request): ResponseInterface
{
$actor = $request->getAttribute('actor');
$this->assertAdmin($actor);

$body = $this->translator->trans('core.email.send_test.body', ['{username}' => $actor->username]);

$this->mailer->raw($body, function (Message $message) use ($actor) {
$message->to($actor->email);
$message->subject($this->translator->trans('core.email.send_test.subject'));
});

return new EmptyResponse();
}
}
7 changes: 7 additions & 0 deletions src/Api/routes.php
Original file line number Diff line number Diff line change
Expand Up @@ -313,4 +313,11 @@
'mailSettings.index',
$route->toController(Controller\ShowMailSettingsController::class)
);

// Send test mail post
$map->post(
'/mail/test',
'mailTest',
$route->toController(Controller\SendTestMailController::class)
);
};