mirror of https://github.com/OCA/social.git
[ADD][fetchmail_thread_default] Default thread for incoming mails
This addon lets the sysadmin choose a default mail thread sink for incoming mails. You can use it to forward all unbound incoming emails to a `mail.channel` where only certain users are subscribed and can triage them.pull/576/head
parent
a928944318
commit
486384e064
|
@ -0,0 +1,93 @@
|
|||
.. image:: https://img.shields.io/badge/licence-AGPL--3-blue.svg
|
||||
:target: http://www.gnu.org/licenses/agpl-3.0-standalone.html
|
||||
:alt: License: AGPL-3
|
||||
|
||||
===================================
|
||||
Default Thread For Unbounded Emails
|
||||
===================================
|
||||
|
||||
This module extends the functionality of mail fetching to support choosing a
|
||||
mail thread that acts as a mail sink and gathers all mail messages that Odoo
|
||||
does not know where to put.
|
||||
|
||||
Dangling emails are really a problem because if you do not care about them,
|
||||
SPAM can enter your inbox and keep increasing fetchmail process network quota
|
||||
because Odoo would gather them every time it runs the fetchmail process.
|
||||
|
||||
Before this, your only choice was to create a new record for those unbounded
|
||||
emails. That could be useful under some circumstances, like creating a
|
||||
``crm.lead`` for them, but what happens if you do not want to have lots of
|
||||
spammy leads? Or if you do not need Odoo's CRM at all?
|
||||
|
||||
Here we come to the rescue. This simple addons adds almost none dependencies
|
||||
and allows you to direct those mails somewhere you can handle or ignore at
|
||||
wish.
|
||||
|
||||
Configuration
|
||||
=============
|
||||
|
||||
To configure this module, you need to:
|
||||
|
||||
#. Go to *Settings > General Settings > Configure the incoming email gateway*.
|
||||
#. Create or edit a record.
|
||||
#. Configure properly.
|
||||
#. Under *Default mail thread*, choose a model and record.
|
||||
|
||||
Tip: if you do not know what to choose, we suggest you to use a mail
|
||||
channel.
|
||||
|
||||
Usage
|
||||
=====
|
||||
|
||||
To use this module, you need to:
|
||||
|
||||
#. Subscribe to the thread you chose as the *Default mail thread*.
|
||||
#. You will be notified when a new unbound email lands in that thread.
|
||||
#. Do what you want with it.
|
||||
|
||||
.. image:: https://odoo-community.org/website/image/ir.attachment/5784_f2813bd/datas
|
||||
:alt: Try me on Runbot
|
||||
:target: https://runbot.odoo-community.org/runbot/205/9.0
|
||||
|
||||
Bug Tracker
|
||||
===========
|
||||
|
||||
Bugs are tracked on `GitHub Issues
|
||||
<https://github.com/OCA/social/issues>`_. In case of trouble, please
|
||||
check there if your issue has already been reported. If you spotted it first,
|
||||
help us smash it by providing detailed and welcomed feedback.
|
||||
|
||||
Credits
|
||||
=======
|
||||
|
||||
Images
|
||||
------
|
||||
|
||||
* Odoo Community Association: `Icon <https://github.com/OCA/maintainer-tools/blob/master/template/module/static/description/icon.svg>`_.
|
||||
|
||||
Contributors
|
||||
------------
|
||||
|
||||
* Jairo Llopis <jairo.llopis@tecnativa.com>
|
||||
|
||||
Funders
|
||||
-------
|
||||
|
||||
The development of this module has been financially supported by:
|
||||
|
||||
* `Tecnativa <https://www.tecnativa.com>`_.
|
||||
|
||||
Maintainer
|
||||
----------
|
||||
|
||||
.. image:: https://odoo-community.org/logo.png
|
||||
:alt: Odoo Community Association
|
||||
:target: https://odoo-community.org
|
||||
|
||||
This module is maintained by the OCA.
|
||||
|
||||
OCA, or the Odoo Community Association, is a nonprofit organization whose
|
||||
mission is to support the collaborative development of Odoo features and
|
||||
promote its widespread use.
|
||||
|
||||
To contribute to this module, please visit https://odoo-community.org.
|
|
@ -0,0 +1,5 @@
|
|||
# -*- coding: utf-8 -*-
|
||||
# Copyright 2017 Jairo Llopis <jairo.llopis@tecnativa.com>
|
||||
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
|
||||
|
||||
from . import models
|
|
@ -0,0 +1,23 @@
|
|||
# -*- coding: utf-8 -*-
|
||||
# Copyright 2017 Jairo Llopis <jairo.llopis@tecnativa.com>
|
||||
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
|
||||
{
|
||||
"name": "Default Thread For Unbounded Emails",
|
||||
"summary": "Post unkonwn messages to an existing thread",
|
||||
"version": "9.0.1.0.0",
|
||||
"category": "Discuss",
|
||||
"website": "https://www.tecnativa.com/",
|
||||
"author": "Tecnativa, Odoo Community Association (OCA)",
|
||||
"license": "AGPL-3",
|
||||
"application": False,
|
||||
"installable": True,
|
||||
"depends": [
|
||||
"fetchmail",
|
||||
],
|
||||
"data": [
|
||||
"views/fetchmail_server_view.xml",
|
||||
],
|
||||
"demo": [
|
||||
"demo/data.xml",
|
||||
],
|
||||
}
|
|
@ -0,0 +1,24 @@
|
|||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<!-- Copyright 2017 Jairo Llopis <jairo.llopis@tecnativa.com>
|
||||
License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). -->
|
||||
|
||||
<odoo>
|
||||
|
||||
<record id="demo_sink" model="mail.channel">
|
||||
<field name="name">mailsink</field>
|
||||
<field name="email_send" eval="True"/>
|
||||
<field name="description">Unbounded email sink</field>
|
||||
<field name="alias_contact">everyone</field>
|
||||
<field name="public">private</field>
|
||||
<field name="group_ids" eval="[(4, ref('base.group_user'))]"/>
|
||||
</record>
|
||||
|
||||
<record id="demo_server" model="fetchmail.server">
|
||||
<field name="name">Demo server</field>
|
||||
<field name="type">pop</field>
|
||||
<field name="server">pop3.example.com</field>
|
||||
<field name="default_thread_id" eval="'mail.channel,%d' % ref('demo_sink')"/>
|
||||
<!-- <field name="default_thread_id">mail.channel,%(demo_sink)d</field> -->
|
||||
</record>
|
||||
|
||||
</odoo>
|
|
@ -0,0 +1,6 @@
|
|||
# -*- coding: utf-8 -*-
|
||||
# Copyright 2017 Jairo Llopis <jairo.llopis@tecnativa.com>
|
||||
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
|
||||
|
||||
from . import fetchmail_server
|
||||
from . import mail_thread
|
|
@ -0,0 +1,49 @@
|
|||
# -*- coding: utf-8 -*-
|
||||
# Copyright 2017 Jairo Llopis <jairo.llopis@tecnativa.com>
|
||||
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
|
||||
|
||||
from openerp import api, fields, models
|
||||
|
||||
|
||||
class FetchmailServer(models.Model):
|
||||
_inherit = "fetchmail.server"
|
||||
|
||||
default_thread_id = fields.Reference(
|
||||
selection="_get_thread_models",
|
||||
string="Default mail thread",
|
||||
help="Messages with no clear route will be posted as a new message "
|
||||
"to this thread.",
|
||||
)
|
||||
|
||||
@api.model
|
||||
def _get_thread_models(self):
|
||||
"""Get list of available ``mail.thread`` submodels.
|
||||
|
||||
:return [(model, name), ...]:
|
||||
Tuple list of available models that can receive messages.
|
||||
"""
|
||||
models = self.env["ir.model.fields"].search([
|
||||
("name", "=", "message_partner_ids"),
|
||||
]).mapped("model_id")
|
||||
# Exclude AbstractModel
|
||||
return [(m.model, m.name) for m in models
|
||||
if getattr(self.env[m.model], "_auto")]
|
||||
|
||||
# TODO New api on v10+
|
||||
# pylint: disable=old-api7-method-defined
|
||||
def onchange_server_type(self, cr, uid, ids, server_type=False, ssl=False,
|
||||
object_id=False):
|
||||
"""Remove :attr:`default_thread_id` if there is :attr:`object_id`."""
|
||||
result = super(FetchmailServer, self).onchange_server_type(
|
||||
cr, uid, ids, server_type, ssl, object_id,
|
||||
)
|
||||
if object_id:
|
||||
result["value"]["default_thread_id"] = False
|
||||
return result
|
||||
|
||||
@api.multi
|
||||
@api.onchange("default_thread_id")
|
||||
def _onchange_remove_object_id(self):
|
||||
"""Remove :attr:`object_id` if there is :attr:`default_thread_id`."""
|
||||
if self.default_thread_id:
|
||||
self.object_id = False
|
|
@ -0,0 +1,30 @@
|
|||
# -*- coding: utf-8 -*-
|
||||
# Copyright 2017 Jairo Llopis <jairo.llopis@tecnativa.com>
|
||||
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
|
||||
|
||||
from openerp import api, models
|
||||
|
||||
|
||||
class MailThread(models.AbstractModel):
|
||||
_inherit = "mail.thread"
|
||||
|
||||
@api.model
|
||||
def message_process(self, model, message, custom_values=None,
|
||||
save_original=False, strip_attachments=False,
|
||||
thread_id=None):
|
||||
server = self.env["fetchmail.server"].browse(
|
||||
self.env.context.get("fetchmail_server_id"))
|
||||
if server.default_thread_id and not (model or thread_id):
|
||||
model = server.default_thread_id._name
|
||||
thread_id = server.default_thread_id.id
|
||||
return super(
|
||||
MailThread,
|
||||
self.with_context(mail_create_nosubscribe=True)
|
||||
).message_process(
|
||||
model,
|
||||
message,
|
||||
custom_values,
|
||||
save_original,
|
||||
strip_attachments,
|
||||
thread_id,
|
||||
)
|
Binary file not shown.
After Width: | Height: | Size: 9.2 KiB |
|
@ -0,0 +1,5 @@
|
|||
# -*- coding: utf-8 -*-
|
||||
# Copyright 2017 Jairo Llopis <jairo.llopis@tecnativa.com>
|
||||
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
|
||||
|
||||
from . import test_fetchmail
|
|
@ -0,0 +1,61 @@
|
|||
# -*- coding: utf-8 -*-
|
||||
# Copyright 2017 Jairo Llopis <jairo.llopis@tecnativa.com>
|
||||
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
|
||||
|
||||
from openerp.addons.mail.tests.test_mail_gateway import MAIL_TEMPLATE
|
||||
from openerp.tests.common import SavepointCase
|
||||
from openerp.tools import mute_logger
|
||||
|
||||
|
||||
class FetchmailCase(SavepointCase):
|
||||
@classmethod
|
||||
def setUpClass(cls):
|
||||
super(FetchmailCase, cls).setUpClass()
|
||||
cls.server = cls.env.ref("fetchmail_thread_default.demo_server")
|
||||
cls.sink = cls.env.ref("fetchmail_thread_default.demo_sink")
|
||||
cls.MailThread = cls.env["mail.thread"]
|
||||
|
||||
def test_available_models(self):
|
||||
"""Non-``mail.thread`` models don't appear."""
|
||||
for record in self.server._get_thread_models():
|
||||
self.assertNotEqual(record[0], "mail.message")
|
||||
|
||||
def test_emptying_default_thread(self):
|
||||
"""Choosing an ``object_id`` empties ``default_thread_id``."""
|
||||
self.assertEqual(
|
||||
self.server.onchange_server_type(object_id=1)
|
||||
["value"]["default_thread_id"],
|
||||
False)
|
||||
|
||||
def test_emptying_object(self):
|
||||
"""Choosing a ``default_thread_id`` empties ``object_id``."""
|
||||
self.server.object_id = self.env["ir.model"].search([], limit=1)
|
||||
self.server._onchange_remove_object_id()
|
||||
self.assertFalse(self.server.object_id)
|
||||
|
||||
@mute_logger('openerp.addons.mail.models.mail_thread', 'openerp.models')
|
||||
def test_unbound_incoming_email(self):
|
||||
"""An unbound incoming email gets posted to the sink."""
|
||||
# Imitate what self.server.feth_mail() would do
|
||||
result = (
|
||||
self.MailThread.with_context(fetchmail_server_id=self.server.id)
|
||||
.message_process(
|
||||
self.server.object_id.model,
|
||||
MAIL_TEMPLATE.format(
|
||||
email_from="spambot@example.com",
|
||||
to="you@example.com",
|
||||
cc="nobody@example.com",
|
||||
subject="I'm a robot, hello",
|
||||
extra="",
|
||||
msg_id="<fitter.happier.more.productive@example.com>",
|
||||
),
|
||||
save_original=self.server.original,
|
||||
strip_attachments=not self.server.attach,
|
||||
)
|
||||
)
|
||||
self.assertEqual(self.server.default_thread_id, self.sink)
|
||||
self.assertEqual(result, self.sink.id)
|
||||
# Nobody subscribed
|
||||
self.assertFalse(self.sink.message_partner_ids)
|
||||
# Message entered channel
|
||||
self.assertEqual(self.sink.message_ids.subject, "I'm a robot, hello")
|
|
@ -0,0 +1,18 @@
|
|||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<!-- Copyright 2017 Jairo Llopis <jairo.llopis@tecnativa.com>
|
||||
License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). -->
|
||||
|
||||
<odoo>
|
||||
|
||||
<record id="view_email_server_form" model="ir.ui.view">
|
||||
<field name="name">Add default thread</field>
|
||||
<field name="model">fetchmail.server</field>
|
||||
<field name="inherit_id" ref="fetchmail.view_email_server_form"/>
|
||||
<field name="arch" type="xml">
|
||||
<xpath expr="//field[@name='object_id']" position="after">
|
||||
<field name="default_thread_id"/>
|
||||
</xpath>
|
||||
</field>
|
||||
</record>
|
||||
|
||||
</odoo>
|
Loading…
Reference in New Issue