[ADD] mail_post_defer: deferred message posting with queue

- Faster because the email sending doesn't block the UI.
- Safer because users can undo mails while they're still not sent.

@moduon MT-1579 MT-2480
pull/1213/head
Jairo Llopis 2023-03-09 09:12:14 +00:00
parent c7ed17a466
commit 11d1699f9c
No known key found for this signature in database
GPG Key ID: E47E3BE44B940490
18 changed files with 425 additions and 0 deletions

View File

@ -0,0 +1,35 @@
**This file is going to be generated by oca-gen-addon-readme.**
*Manual changes will be overwritten.*
Please provide content in the ``readme`` directory:
* **DESCRIPTION.rst** (required)
* INSTALL.rst (optional)
* CONFIGURE.rst (optional)
* **USAGE.rst** (optional, highly recommended)
* DEVELOP.rst (optional)
* ROADMAP.rst (optional)
* HISTORY.rst (optional, recommended)
* **CONTRIBUTORS.rst** (optional, highly recommended)
* CREDITS.rst (optional)
Content of this README will also be drawn from the addon manifest,
from keys such as name, authors, maintainers, development_status,
and license.
A good, one sentence summary in the manifest is also highly recommended.
Automatic changelog generation
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
`HISTORY.rst` can be auto generated using `towncrier <https://pypi.org/project/towncrier>`_.
Just put towncrier compatible changelog fragments into `readme/newsfragments`
and the changelog file will be automatically generated and updated when a new fragment is added.
Please refer to `towncrier` documentation to know more.
NOTE: the changelog will be automatically generated when using `/ocabot merge $option`.
If you need to run it manually, refer to `OCA/maintainer-tools README <https://github.com/OCA/maintainer-tools>`_.

View File

@ -0,0 +1,2 @@
from . import models
from .hooks import post_init_hook

View File

@ -0,0 +1,25 @@
# Copyright 2022-2023 Moduon Team S.L. <info@moduon.team>
# License LGPL-3.0 or later (https://www.gnu.org/licenses/lgpl).
{
"name": "Deferred Message Posting",
"summary": "Faster and cancellable outgoing messages",
"version": "15.0.1.0.0",
"development_status": "Alpha",
"category": "Productivity/Discuss",
"website": "https://github.com/OCA/social",
"author": "Moduon, Odoo Community Association (OCA)",
"maintainers": ["Yajo"],
"license": "LGPL-3",
"depends": [
"mail",
],
"post_init_hook": "post_init_hook",
"assets": {
"web.assets_backend": [
"mail_post_defer/static/src/**/*.js",
],
"web.assets_qweb": [
"mail_post_defer/static/src/**/*.xml",
],
},
}

View File

@ -0,0 +1,23 @@
# Copyright 2022-2023 Moduon Team S.L. <info@moduon.team>
# License LGPL-3.0 or later (https://www.gnu.org/licenses/lgpl).
import logging
from odoo import SUPERUSER_ID, api
_logger = logging.getLogger(__name__)
def post_init_hook(cr, registry):
"""Increase cadence of mail queue cron."""
env = api.Environment(cr, SUPERUSER_ID, {})
try:
cron = env.ref("mail.ir_cron_mail_scheduler_action")
except ValueError:
_logger.warning(
"Couldn't find the standard mail scheduler cron. "
"Maybe no mails will be ever sent!"
)
else:
_logger.info("Setting mail queue cron cadence to 1 minute")
cron.interval_number = 1
cron.interval_type = "minutes"

View File

@ -0,0 +1,2 @@
from . import mail_message
from . import mail_thread

View File

@ -0,0 +1,18 @@
# Copyright 2022-2023 Moduon Team S.L. <info@moduon.team>
# License LGPL-3.0 or later (https://www.gnu.org/licenses/lgpl).
from odoo import models
class MailMessage(models.Model):
_inherit = "mail.message"
def _cleanup_side_records(self):
"""Delete pending outgoing mails."""
self.mail_ids.filtered(lambda mail: mail.state == "outgoing").unlink()
return super()._cleanup_side_records()
def _update_content(self, body, attachment_ids):
"""Let checker know about empty body."""
_self = self.with_context(deleting=body == "")
return super(MailMessage, _self)._update_content(body, attachment_ids)

View File

@ -0,0 +1,43 @@
# Copyright 2022-2023 Moduon Team S.L. <info@moduon.team>
# License LGPL-3.0 or later (https://www.gnu.org/licenses/lgpl).
from datetime import timedelta
from odoo import fields, models
class MailThread(models.AbstractModel):
_inherit = "mail.thread"
def message_post(self, **kwargs):
"""Post messages using queue by default."""
_self = self
force_send = self.env.context.get("mail_notify_force_send") or kwargs.get(
"force_send", False
)
kwargs.setdefault("force_send", force_send)
if not force_send:
# If deferring message, give the user some minimal time to revert it
_self = self.with_context(mail_defer_seconds=30)
return super(MailThread, _self).message_post(**kwargs)
def _notify_by_email_add_values(self, base_mail_values):
"""Defer emails by default."""
result = super()._notify_by_email_add_values(base_mail_values)
defer_seconds = self.env.context.get("mail_defer_seconds")
if defer_seconds:
result.setdefault(
"scheduled_date",
fields.Datetime.now() + timedelta(seconds=defer_seconds),
)
return result
def _check_can_update_message_content(self, message):
"""Allow deleting unsent mails."""
if (
self.env.context.get("deleting")
and set(message.notification_ids.mapped("notification_status")) == {"ready"}
and set(message.mail_ids.mapped("state")) == {"outgoing"}
):
return
return super()._check_can_update_message_content(message)

View File

@ -0,0 +1,15 @@
You need to do nothing. The module is configured appropriately out of the box.
The mail queue processing is made by a cron job. This is normal Odoo behavior,
not specific to this module. However, since you will start using that queue for
every message posted by any user in any thread, this module configures that job
to execute every minute by default.
You can still change that cadence after installing the module (although it is
not recommended). To do so:
#. Log in with an administrator user.
#. Activate developer mode.
#. Go to *Settings > Technical > Automation > Scheduled Actions*.
#. Edit the action named "Mail: Email Queue Manager".
#. Lower down the frequency in the field *Execute Every*. Recommended: 1 minute.

View File

@ -0,0 +1 @@
* Jairo Llopis (https://www.moduon.team/)

View File

@ -0,0 +1,9 @@
This module enhances mail threads by using the mail queue by default.
Without this module, Odoo attempts to notify recipients of your message immediately.
If your mail server is slow or you have many followers, this can mean a lot of time.
Install this module and make Odoo more snappy!
All emails will be kept in the outgoing queue by at least 30 seconds,
giving you some time to re-think what you wrote. During that time,
you can still delete the message and start again.

View File

@ -0,0 +1,3 @@
* Add minimal deferring time configuration if it ever becomes necessary. See
https://github.com/OCA/social/pull/1001#issuecomment-1461581573 for the
rationale behind current hardcoded value of 30 seconds.

View File

@ -0,0 +1,15 @@
To use this module, you need to:
#. Go to the form view of any record that has a mail thread. It can be a partner, for example.
#. Post a message.
The mail is now in the outgoing mail queue. It will be there for at least 30
seconds. It will be really sent the next time the "Mail: Email Queue Manager"
cron job is executed.
While the message has not been yet sent:
#. Hover over the little envelope. You will see a paper airplane icon,
indicating it is still outgoing.
#. Hover over the message and click on the little trash icon to delete it.
Mails will not be sent.

Binary file not shown.

After

Width:  |  Height:  |  Size: 26 KiB

View File

@ -0,0 +1,48 @@
/** @odoo-module **/
import {
registerFieldPatchModel,
registerInstancePatchModel,
} from "@mail/model/model_core";
import {attr} from "@mail/model/model_field";
registerInstancePatchModel("mail.message", "mail_post_defer.message", {
xmlDependencies: ["/mail_post_defer/static/src/xml/message.xml"],
/**
* Allow deleting deferred messages
*
* @param {Boolean} editing Set `true` to know if you can edit the message
* @returns {Boolean}
*/
_computeCanBeDeleted(editing) {
return (
this._super() ||
(!editing &&
this.notifications.filter(
(current) => current.notification_status !== "ready"
).length === 0)
);
},
/**
* Allow editing messages.
*
* Upstream Odoo allows editing any message that can be deleted. We do the
* same here. However, if the message is a public message that is deferred,
* it can be edited but not deleted.
*
* @returns {Boolean}
*/
_computeCanBeEdited() {
return this._computeCanBeDeleted(true);
},
});
registerFieldPatchModel("mail.message", "mail_post_defer.message", {
/**
* Whether this message can be edited.
*/
canBeEdited: attr({
compute: "_computeCanBeEdited",
}),
});

View File

@ -0,0 +1,13 @@
<?xml version="1.0" encoding="utf-8" ?>
<!-- Copyright 2023 Moduon Team S.L. <info@moduon.team>
License LGPL-3.0 or later (https://www.gnu.org/licenses/lgpl). -->
<template>
<t t-inherit="mail.MessageActionList" t-inherit-mode="extension" owl="1">
<xpath
expr="//*[@t-if='messageActionList.message.canBeDeleted'][hasclass('o_MessageActionList_actionEdit')]"
position="attributes"
>
<attribute name="t-if">messageActionList.message.canBeEdited</attribute>
</xpath>
</t>
</template>

View File

@ -0,0 +1,2 @@
from . import test_install
from . import test_mail

View File

@ -0,0 +1,12 @@
# Copyright 2022-2023 Moduon Team S.L. <info@moduon.team>
# License LGPL-3.0 or later (https://www.gnu.org/licenses/lgpl).
from odoo.tests.common import TransactionCase
class InstallationCase(TransactionCase):
def test_cron_cadence(self):
"""Test that the post_init_hook was properly executed."""
cron = self.env.ref("mail.ir_cron_mail_scheduler_action")
cadence = cron.interval_number, cron.interval_type
self.assertEqual(cadence, (1, "minutes"))

View File

@ -0,0 +1,159 @@
# Copyright 2022-2023 Moduon Team S.L. <info@moduon.team>
# License LGPL-3.0 or later (https://www.gnu.org/licenses/lgpl).
import freezegun
from odoo.exceptions import UserError
from odoo.addons.mail.tests.common import MailCommon
@freezegun.freeze_time("2023-01-02 10:00:00")
class MessagePostCase(MailCommon):
@classmethod
def setUpClass(cls):
super().setUpClass()
cls._create_portal_user()
# Notify employee by email
cls.user_employee.notification_type = "email"
def test_standard(self):
"""A normal call just uses the queue by default."""
with self.mock_mail_gateway():
self.partner_portal.message_post(
body="test body",
subject="test subject",
message_type="comment",
partner_ids=self.partner_employee.ids,
)
self.assertMailMail(
self.partner_employee,
"outgoing",
author=self.env.user.partner_id,
content="test body",
fields_values={"scheduled_date": "2023-01-02 10:00:30"},
)
def test_forced_arg(self):
"""A forced send via method argument is sent directly."""
with self.mock_mail_gateway():
self.partner_portal.message_post(
body="test body",
subject="test subject",
message_type="comment",
partner_ids=self.partner_employee.ids,
force_send=True,
)
self.assertMailMail(
self.partner_employee,
"sent",
author=self.env.user.partner_id,
content="test body",
fields_values={"scheduled_date": False},
)
def test_forced_context(self):
"""A forced send via context is sent directly."""
with self.mock_mail_gateway():
self.partner_portal.with_context(mail_notify_force_send=True).message_post(
body="test body",
subject="test subject",
message_type="comment",
partner_ids=self.partner_employee.ids,
)
self.assertMailMail(
self.partner_employee,
"sent",
author=self.env.user.partner_id,
content="test body",
fields_values={"scheduled_date": False},
)
def test_no_msg_edit(self):
"""Cannot update messages.
This is normal upstream Odoo behavior. It is not a feature of this
module, but it is important to make sure this protection is still
respected, because we disable it for queued message deletion.
A non-malicious end user won't get to this code because the edit button
is hidden. Still, the server-side protection is important.
If, at some point, this module is improved to support this use case,
then this test should change; and that would be a good thing probably.
"""
with self.mock_mail_gateway():
msg = self.partner_portal.message_post(
body="test body",
subject="test subject",
message_type="comment",
partner_ids=self.partner_employee.ids,
subtype_xmlid="mail.mt_comment",
)
# Emulate user clicking on edit button and going through the
# `/mail/message/update_content` controller
with self.assertRaises(UserError):
msg._update_content("new body", [])
self.assertMailMail(
self.partner_employee,
"outgoing",
author=self.env.user.partner_id,
content="test body",
fields_values={"scheduled_date": "2023-01-02 10:00:30"},
)
def test_queued_msg_delete(self):
"""A user can delete a message before it's sent."""
with self.mock_mail_gateway():
msg = self.partner_portal.message_post(
body="test body",
subject="test subject",
message_type="comment",
partner_ids=self.partner_employee.ids,
subtype_xmlid="mail.mt_comment",
)
# Emulate user clicking on delete button and going through the
# `/mail/message/update_content` controller
msg._update_content("", [])
self.assertNoMail(
self.partner_employee,
author=self.env.user.partner_id,
)
# One minute later, the cron has no mails to send
with freezegun.freeze_time("2023-01-02 10:01:00"):
self.env["mail.mail"].process_email_queue()
self.assertNoMail(
self.partner_employee,
author=self.env.user.partner_id,
)
def test_no_sent_msg_delete(self):
"""A user cannot delete a message after it's sent.
Usually, the trash button will be hidden in UI if the message is sent.
However, the server-side protection is still important, because there
can be a race condition when the mail is sent in the background but
the user didn't refresh the view.
"""
with self.mock_mail_gateway():
msg = self.partner_portal.message_post(
body="test body",
subject="test subject",
message_type="comment",
partner_ids=self.partner_employee.ids,
subtype_xmlid="mail.mt_comment",
)
# One minute later, the cron sends the mail
with freezegun.freeze_time("2023-01-02 10:01:00"):
self.env["mail.mail"].process_email_queue()
self.assertMailMail(
self.partner_employee,
"sent",
author=self.env.user.partner_id,
content="test body",
fields_values={"scheduled_date": "2023-01-02 10:00:30"},
)
# Emulate user clicking on delete button and going through the
# `/mail/message/update_content` controller
with self.assertRaises(UserError):
msg._update_content("", [])