Browse Source
[ADD][fetchmail_thread_default] Default thread for incoming mails
[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/339/head
Jairo Llopis
8 years ago
committed by
David
12 changed files with 314 additions and 0 deletions
-
93fetchmail_thread_default/README.rst
-
5fetchmail_thread_default/__init__.py
-
23fetchmail_thread_default/__openerp__.py
-
24fetchmail_thread_default/demo/data.xml
-
0fetchmail_thread_default/i18n/.empty
-
6fetchmail_thread_default/models/__init__.py
-
49fetchmail_thread_default/models/fetchmail_server.py
-
30fetchmail_thread_default/models/mail_thread.py
-
BINfetchmail_thread_default/static/description/icon.png
-
5fetchmail_thread_default/tests/__init__.py
-
61fetchmail_thread_default/tests/test_fetchmail.py
-
18fetchmail_thread_default/views/fetchmail_server_view.xml
@ -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, |
||||
|
) |
After Width: 128 | Height: 128 | 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> |
Write
Preview
Loading…
Cancel
Save
Reference in new issue