Browse Source
ADD mail_outbound_static: Allow configuration of from email header
ADD mail_outbound_static: Allow configuration of from email header
* Allow for the configuration of the FROM header for outbound emails, and move the existing from header to Sender as per RFC-2822pull/388/head
Dave Lasley
6 years ago
committed by
eLBati
10 changed files with 309 additions and 0 deletions
-
62mail_outbound_static/README.rst
-
5mail_outbound_static/__init__.py
-
21mail_outbound_static/__manifest__.py
-
5mail_outbound_static/models/__init__.py
-
50mail_outbound_static/models/ir_mail_server.py
-
BINmail_outbound_static/static/description/icon.png
-
5mail_outbound_static/tests/__init__.py
-
70mail_outbound_static/tests/test.msg
-
70mail_outbound_static/tests/test_ir_mail_server.py
-
21mail_outbound_static/views/ir_mail_server_view.xml
@ -0,0 +1,62 @@ |
|||||
|
.. image:: https://img.shields.io/badge/license-LGPL--3-blue.svg |
||||
|
:target: http://www.gnu.org/licenses/lgpl.html |
||||
|
:alt: License: LGPL-3 |
||||
|
|
||||
|
==================== |
||||
|
Mail Outbound Static |
||||
|
==================== |
||||
|
|
||||
|
This module brings Odoo outbound emails in to strict compliance with RFC-2822 |
||||
|
by allowing for a statically configured From header, with the sender's e-mail |
||||
|
being appended into the proper Sender header instead. |
||||
|
|
||||
|
Usage |
||||
|
===== |
||||
|
|
||||
|
* Navigate to an Outbound Email Server |
||||
|
* Set the `Email From` option to an email address |
||||
|
|
||||
|
.. 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/10.0 |
||||
|
|
||||
|
Road Map |
||||
|
======== |
||||
|
|
||||
|
* Allow for domain-based whitelist that will not be manipulated |
||||
|
|
||||
|
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 |
||||
|
------------ |
||||
|
|
||||
|
* Dave Lasley <dave@laslabs.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 LasLabs Inc. |
||||
|
# License LGPL-3.0 or later (http://www.gnu.org/licenses/lgpl.html). |
||||
|
|
||||
|
from . import models |
@ -0,0 +1,21 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
# Copyright 2016-2017 LasLabs Inc. |
||||
|
# License LGPL-3.0 or later (http://www.gnu.org/licenses/lgpl.html). |
||||
|
|
||||
|
{ |
||||
|
'name': 'Mail Outbound Static', |
||||
|
'summary': 'Allows you to configure the from header for a mail server.', |
||||
|
'version': '10.0.1.0.1', |
||||
|
'category': 'Discuss', |
||||
|
'website': 'https://laslabs.com/', |
||||
|
'author': 'LasLabs, Odoo Community Association (OCA)', |
||||
|
'license': 'LGPL-3', |
||||
|
'application': False, |
||||
|
'installable': True, |
||||
|
'depends': [ |
||||
|
'base', |
||||
|
], |
||||
|
'data': [ |
||||
|
'views/ir_mail_server_view.xml', |
||||
|
], |
||||
|
} |
@ -0,0 +1,5 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
# Copyright 2017 LasLabs Inc. |
||||
|
# License LGPL-3.0 or later (http://www.gnu.org/licenses/lgpl.html). |
||||
|
|
||||
|
from . import ir_mail_server |
@ -0,0 +1,50 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
# Copyright 2017 LasLabs Inc. |
||||
|
# License LGPL-3.0 or later (http://www.gnu.org/licenses/lgpl.html). |
||||
|
|
||||
|
from odoo import api, fields, models |
||||
|
|
||||
|
|
||||
|
class IrMailServer(models.Model): |
||||
|
|
||||
|
_inherit = 'ir.mail_server' |
||||
|
|
||||
|
smtp_from = fields.Char( |
||||
|
string='Email From', |
||||
|
help='Set this in order to email from a specific address.' |
||||
|
) |
||||
|
|
||||
|
@api.model |
||||
|
def send_email(self, message, mail_server_id=None, smtp_server=None, |
||||
|
*args, **kwargs): |
||||
|
|
||||
|
# Replicate logic from core to get mail server |
||||
|
mail_server = None |
||||
|
if mail_server_id: |
||||
|
mail_server = self.sudo().browse(mail_server_id) |
||||
|
elif not smtp_server: |
||||
|
mail_server = self.sudo().search([], order='sequence', limit=1) |
||||
|
|
||||
|
if mail_server and mail_server.smtp_from: |
||||
|
split_from = message['From'].rsplit(' <', 1) |
||||
|
if len(split_from) > 1: |
||||
|
email_from = '%s <%s>' % ( |
||||
|
split_from[0], mail_server.smtp_from, |
||||
|
) |
||||
|
else: |
||||
|
email_from = mail_server.smtp_from |
||||
|
|
||||
|
message.replace_header('From', email_from) |
||||
|
bounce_alias = self.env['ir.config_parameter'].get_param( |
||||
|
"mail.bounce.alias") |
||||
|
if not bounce_alias: |
||||
|
# then, bounce handling is disabled and we want |
||||
|
# Return-Path = From |
||||
|
if message.has_key('Return-Path'): |
||||
|
message.replace_header('Return-Path', email_from) |
||||
|
else: |
||||
|
message.add_header('Return-Path', email_from) |
||||
|
|
||||
|
return super(IrMailServer, self).send_email( |
||||
|
message, mail_server_id, smtp_server, *args, **kwargs |
||||
|
) |
After Width: 128 | Height: 128 | Size: 9.2 KiB |
@ -0,0 +1,5 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
# Copyright 2017 LasLabs Inc. |
||||
|
# License LGPL-3.0 or later (http://www.gnu.org/licenses/lgpl.html). |
||||
|
|
||||
|
from . import test_ir_mail_server |
@ -0,0 +1,70 @@ |
|||||
|
Delivered-To: test@gmail.com |
||||
|
Received: by 10.74.138.167 with SMTP id m36csp7226976ooj; |
||||
|
Tue, 12 Sep 2017 10:37:56 -0700 (PDT) |
||||
|
X-Google-Smtp-Source: AOwi7QDKSb3BE6lIhVXub9wcPA/HxFKKpnNPconNr9f1L35SVw+EIm8itVQkbOdAW6TohImypmrF |
||||
|
X-Received: by 10.28.158.208 with SMTP id h199mr250060wme.47.1505237876258; |
||||
|
Tue, 12 Sep 2017 10:37:56 -0700 (PDT) |
||||
|
ARC-Seal: i=1; a=rsa-sha256; t=1505237876; cv=none; |
||||
|
d=google.com; s=arc-20160816; |
||||
|
b=E2B6KUxHOJQk1YrT12BpitEMCgkxyqEXcFlwPWKjA/i/Xyvlh+09spNOF4VPmD/ZJm |
||||
|
5lkY6hYyxvIH2RpRPeZVPkRIYhaEASkMIygdJu9Gd4weBdO2rd8iP/zSGHYyAmO/hLN2 |
||||
|
64hXtKexrWnO/YNWlpfhAo1kiwgSRVnZx55EopbWP49cy7BzKfwr1kHN0T9A5Lw1w+BW |
||||
|
ZrXdCX6LRxHS2USKHb76PAVt0bhwsM/ZznBauR2zNKYcPxAWzdpN/vK3BDmWUdqbbSaB |
||||
|
BOKINjuI9EmWynogDZE7Riu+sbc5QafE3owla1/2d0Bogp9FLtJe0YyQeW2qLvZKcmlI |
||||
|
ftSQ== |
||||
|
ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; |
||||
|
h=to:list-archive:list-unsubscribe:list-subscribe:precedence |
||||
|
:list-post:list-id:date:reply-to:from:subject:references:message-id |
||||
|
:mime-version:arc-authentication-results; |
||||
|
bh=DwvSiw5K7ryb4S8O/8HcIaGhJqbOxcXKsnPAr63iQZ4=; |
||||
|
b=U0Ac9Rqvv+tfqO9fCx+F79oZknn3rOv9N9ekViEuL5DtjpJxKDDkO1xw//sV3eRILT |
||||
|
nqGuxd2yQXwC4U+WAwraBwoLC3ScHb/9gWtzlrLCgv6WbNE7HZi5g6L8c0LWRN24cIe9 |
||||
|
AOdc/8fOdGoaL8yajrGEHgMz9B2KMltA9tZyxFOeKsyODxJ6iWjXcG1BSQTxERwosV3h |
||||
|
ch8AznQr7xLLvc/u9VTEqC5ome3RqsxKRxOGenEqIbCOr11sxwpZQdQcNR6faNRom3+2 |
||||
|
6gz++4tVIV9cqYX1j9eEU/ufoUzBJ6Uzm0jMGZZQOHAF+YX3tZUEsPmc75PsvRCAIWby |
||||
|
urMg== |
||||
|
ARC-Authentication-Results: i=1; mx.google.com; |
||||
|
spf=pass (google.com: domain of postmaster-odoo@odoo-community.org designates 2a01:4f8:a0:430d::2 as permitted sender) smtp.mailfrom=postmaster-odoo@odoo-community.org; |
||||
|
dmarc=fail (p=QUARANTINE sp=QUARANTINE dis=QUARANTINE) header.from=laslabs.com |
||||
|
Return-Path: <postmaster-odoo@odoo-community.org> |
||||
|
Received: from odoo-community.org (odoo-community.org. [2a01:4f8:a0:430d::2]) |
||||
|
by mx.google.com with ESMTP id j72si1795626wmg.60.2017.09.12.10.37.55 |
||||
|
for <test@gmail.com>; |
||||
|
Tue, 12 Sep 2017 10:37:56 -0700 (PDT) |
||||
|
Received-SPF: pass (google.com: domain of postmaster-odoo@odoo-community.org designates 2a01:4f8:a0:430d::2 as permitted sender) client-ip=2a01:4f8:a0:430d::2; |
||||
|
Authentication-Results: mx.google.com; |
||||
|
spf=pass (google.com: domain of postmaster-odoo@odoo-community.org designates 2a01:4f8:a0:430d::2 as permitted sender) smtp.mailfrom=postmaster-odoo@odoo-community.org; |
||||
|
dmarc=fail (p=QUARANTINE sp=QUARANTINE dis=QUARANTINE) header.from=laslabs.com |
||||
|
Received: from odoo.odoo-community.org (localhost.localdomain [127.0.0.1]) |
||||
|
by odoo-community.org (Postfix) with ESMTP id DB5DC2EC2277; |
||||
|
Tue, 12 Sep 2017 19:37:53 +0200 (CEST) |
||||
|
Content-Type: multipart/mixed; boundary="===============7439524030966430607==" |
||||
|
MIME-Version: 1.0 |
||||
|
Message-Id: <A1AAECD6-2A36-45E1-A0F8-4266F178D52C@laslabs.com> |
||||
|
references: <0db43737-b846-4890-6801-44ff9617e3b3@camptocamp.com> |
||||
|
Subject: Re: OCA Code sprint: sprint topics |
||||
|
From: Dave Lasley <test@laslabs.com> |
||||
|
Reply-To: "Odoo Community Association \(OCA\) Contributors" |
||||
|
<contributors@odoo-community.org> |
||||
|
Date: Tue, 12 Sep 2017 17:37:53 -0000 |
||||
|
List-Id: contributors.odoo-community.org |
||||
|
List-Post: <mailto:contributors@odoo-community.org> |
||||
|
Precedence: list |
||||
|
X-Auto-Response-Suppress: OOF |
||||
|
List-Subscribe: <https://odoo-community.org/groups> |
||||
|
List-Unsubscribe: <https://odoo-community.org/groups?unsubscribe> |
||||
|
List-Archive: <https://odoo-community.org/groups/contributors-15> |
||||
|
To: "Contributors" <contributors@odoo-community.org> |
||||
|
|
||||
|
--===============7439524030966430607== |
||||
|
Content-Type: multipart/alternative; |
||||
|
boundary="===============8317593469411551167==" |
||||
|
MIME-Version: 1.0 |
||||
|
|
||||
|
--===============8317593469411551167== |
||||
|
Content-Type: text/plain; charset="utf-8" |
||||
|
MIME-Version: 1.0 |
||||
|
Content-Transfer-Encoding: base64 |
||||
|
|
||||
|
VGhpcyBpcyBhIGZha2UsIHRlc3QgbWVzc2FnZQ== |
||||
|
--===============7439524030966430607==-- |
@ -0,0 +1,70 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
# Copyright 2017 LasLabs Inc. |
||||
|
# License LGPL-3.0 or later (http://www.gnu.org/licenses/lgpl.html). |
||||
|
|
||||
|
import os |
||||
|
import threading |
||||
|
|
||||
|
from mock import MagicMock |
||||
|
from email import message_from_string |
||||
|
|
||||
|
from odoo.tests.common import TransactionCase |
||||
|
|
||||
|
|
||||
|
class TestIrMailServer(TransactionCase): |
||||
|
|
||||
|
def setUp(self): |
||||
|
super(TestIrMailServer, self).setUp() |
||||
|
self.email_from = 'derp@example.com' |
||||
|
self.email_from_another = 'another@example.com' |
||||
|
self.Model = self.env['ir.mail_server'] |
||||
|
self.Model.search([]).write({'smtp_from': self.email_from}) |
||||
|
message_file = os.path.join( |
||||
|
os.path.dirname(os.path.realpath(__file__)), 'test.msg', |
||||
|
) |
||||
|
with open(message_file, 'r') as fh: |
||||
|
self.message = message_from_string(fh.read()) |
||||
|
|
||||
|
def _send_mail(self, message=None, mail_server_id=None, smtp_server=None): |
||||
|
if message is None: |
||||
|
message = self.message |
||||
|
connect = MagicMock() |
||||
|
thread = threading.currentThread() |
||||
|
setattr(thread, 'testing', False) |
||||
|
try: |
||||
|
self.Model._patch_method('connect', connect) |
||||
|
try: |
||||
|
self.Model.send_email(message, mail_server_id, smtp_server) |
||||
|
finally: |
||||
|
self.Model._revert_method('connect') |
||||
|
finally: |
||||
|
setattr(thread, 'testing', True) |
||||
|
send_from, send_to, message_string = connect().sendmail.call_args[0] |
||||
|
return message_from_string(message_string) |
||||
|
|
||||
|
def test_send_email_injects_from_no_canonical(self): |
||||
|
"""It should inject the FROM header correctly when no canonical name. |
||||
|
""" |
||||
|
self.message.replace_header('From', 'test@example.com') |
||||
|
message = self._send_mail() |
||||
|
self.assertEqual(message['From'], self.email_from) |
||||
|
|
||||
|
def test_send_email_injects_from_with_canonical(self): |
||||
|
"""It should inject the FROM header correctly with a canonical name. |
||||
|
|
||||
|
Note that there is an extra `<` in the canonical name to test for |
||||
|
proper handling in the split. |
||||
|
""" |
||||
|
user = 'Test < User' |
||||
|
self.message.replace_header('From', '%s <test@example.com>' % user) |
||||
|
message = self._send_mail() |
||||
|
self.assertEqual( |
||||
|
message['From'], |
||||
|
'%s <%s>' % (user, self.email_from), |
||||
|
) |
||||
|
|
||||
|
def test_send_email_injects_sender(self): |
||||
|
"""It should inject the Sender header into the email.""" |
||||
|
original_from = self.message['From'] |
||||
|
message = self._send_mail() |
||||
|
self.assertEqual(message['Sender'], original_from) |
@ -0,0 +1,21 @@ |
|||||
|
<?xml version="1.0" encoding="utf-8"?> |
||||
|
|
||||
|
<!-- |
||||
|
Copyright 2017 LasLabs Inc. |
||||
|
License LGPL-3 or later (http://www.gnu.org/licenses/lgpl.html). |
||||
|
--> |
||||
|
|
||||
|
<odoo> |
||||
|
|
||||
|
<record id="ir_mail_server_form" model="ir.ui.view"> |
||||
|
<field name="name">IR Mail Server - From Address</field> |
||||
|
<field name="model">ir.mail_server</field> |
||||
|
<field name="inherit_id" ref="base.ir_mail_server_form" /> |
||||
|
<field name="arch" type="xml"> |
||||
|
<xpath expr="//field[@name='smtp_pass']" position="after"> |
||||
|
<field name="smtp_from" /> |
||||
|
</xpath> |
||||
|
</field> |
||||
|
</record> |
||||
|
|
||||
|
</odoo> |
Write
Preview
Loading…
Cancel
Save
Reference in new issue