Browse Source

[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/159/head
Jairo Llopis 7 years ago
parent
commit
7ce34d22b2
  1. 93
      fetchmail_thread_default/README.rst
  2. 5
      fetchmail_thread_default/__init__.py
  3. 23
      fetchmail_thread_default/__openerp__.py
  4. 24
      fetchmail_thread_default/demo/data.xml
  5. 0
      fetchmail_thread_default/i18n/.empty
  6. 6
      fetchmail_thread_default/models/__init__.py
  7. 49
      fetchmail_thread_default/models/fetchmail_server.py
  8. 30
      fetchmail_thread_default/models/mail_thread.py
  9. BIN
      fetchmail_thread_default/static/description/icon.png
  10. 5
      fetchmail_thread_default/tests/__init__.py
  11. 61
      fetchmail_thread_default/tests/test_fetchmail.py
  12. 18
      fetchmail_thread_default/views/fetchmail_server_view.xml

93
fetchmail_thread_default/README.rst

@ -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.

5
fetchmail_thread_default/__init__.py

@ -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

23
fetchmail_thread_default/__openerp__.py

@ -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",
],
}

24
fetchmail_thread_default/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
fetchmail_thread_default/i18n/.empty

6
fetchmail_thread_default/models/__init__.py

@ -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

49
fetchmail_thread_default/models/fetchmail_server.py

@ -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

30
fetchmail_thread_default/models/mail_thread.py

@ -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,
)

BIN
fetchmail_thread_default/static/description/icon.png

After

Width: 128  |  Height: 128  |  Size: 9.2 KiB

5
fetchmail_thread_default/tests/__init__.py

@ -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

61
fetchmail_thread_default/tests/test_fetchmail.py

@ -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")

18
fetchmail_thread_default/views/fetchmail_server_view.xml

@ -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…
Cancel
Save