Browse Source
Merge pull request #216 from Tecnativa/10.0-imp-mail_tracking_mailgun
Merge pull request #216 from Tecnativa/10.0-imp-mail_tracking_mailgun
[10.0][IMP] mail_tracking_mailgun: add partner mail checkspull/217/head
Pedro M. Baeza
7 years ago
committed by
GitHub
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
13 changed files with 548 additions and 18 deletions
-
26mail_tracking_mailgun/README.rst
-
1mail_tracking_mailgun/__init__.py
-
11mail_tracking_mailgun/__manifest__.py
-
133mail_tracking_mailgun/i18n/es.po
-
3mail_tracking_mailgun/models/__init__.py
-
2mail_tracking_mailgun/models/ir_mail_server.py
-
58mail_tracking_mailgun/models/mail_tracking_email.py
-
21mail_tracking_mailgun/models/mail_tracking_event.py
-
135mail_tracking_mailgun/models/res_partner.py
-
1mail_tracking_mailgun/tests/__init__.py
-
131mail_tracking_mailgun/tests/test_mailgun.py
-
16mail_tracking_mailgun/views/mail_tracking_email.xml
-
28mail_tracking_mailgun/views/res_partner.xml
@ -1,5 +1,4 @@ |
|||
# -*- coding: utf-8 -*- |
|||
# Copyright 2016 Antonio Espinosa <antonio.espinosa@tecnativa.com> |
|||
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). |
|||
|
|||
from . import models |
@ -1,6 +1,7 @@ |
|||
# -*- coding: utf-8 -*- |
|||
# Copyright 2016 Antonio Espinosa - <antonio.espinosa@tecnativa.com> |
|||
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). |
|||
|
|||
from . import ir_mail_server |
|||
from . import mail_tracking_email |
|||
from . import mail_tracking_event |
|||
from . import res_partner |
@ -0,0 +1,21 @@ |
|||
# -*- coding: utf-8 -*- |
|||
# Copyright 2017 Tecnativa - David Vidal |
|||
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl.html). |
|||
|
|||
from openerp import models, fields |
|||
|
|||
|
|||
class MailTrackingEvent(models.Model): |
|||
_inherit = "mail.tracking.event" |
|||
|
|||
mailgun_id = fields.Char( |
|||
string="Mailgun Event ID", |
|||
copy="False", |
|||
readonly=True, |
|||
) |
|||
|
|||
def _process_data(self, tracking_email, metadata, event_type, state): |
|||
res = super(MailTrackingEvent, self)._process_data( |
|||
tracking_email, metadata, event_type, state) |
|||
res.update({'mailgun_id': metadata.get('mailgun_id', False)}) |
|||
return res |
@ -0,0 +1,135 @@ |
|||
# -*- coding: utf-8 -*- |
|||
# Copyright 2016 Tecnativa - Antonio Espinosa |
|||
# Copyright 2016 Tecnativa - Carlos Dauden |
|||
# Copyright 2017 Tecnativa - Pedro M. Baeza |
|||
# Copyright 2017 Tecnativa - David Vidal |
|||
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl.html). |
|||
|
|||
import requests |
|||
import json |
|||
|
|||
from openerp import _, api, models |
|||
from openerp.exceptions import UserError |
|||
|
|||
|
|||
class ResPartner(models.Model): |
|||
_inherit = 'res.partner' |
|||
|
|||
@api.multi |
|||
def email_bounced_set(self, tracking_emails, reason, event=None): |
|||
res = super(ResPartner, self).email_bounced_set( |
|||
tracking_emails, reason, event=event) |
|||
self._email_bounced_set(reason, event) |
|||
return res |
|||
|
|||
@api.multi |
|||
def _email_bounced_set(self, reason, event): |
|||
for partner in self: |
|||
if not partner.email: |
|||
continue |
|||
body = _('Email has been bounced: %s\n' |
|||
'Reason: %s\n' |
|||
'Event: %s') % (partner.email, reason, |
|||
event['Message-Id'] or '') |
|||
partner.message_post(body=body) |
|||
|
|||
@api.multi |
|||
def check_email_validity(self): |
|||
""" |
|||
Checks mailbox validity with Mailgun's API |
|||
API documentation: |
|||
https://documentation.mailgun.com/en/latest/api-email-validation.html |
|||
""" |
|||
api_key, api_url, domain, validation_key = self.env[ |
|||
'mail.tracking.email']._mailgun_values() |
|||
if not validation_key: |
|||
raise UserError(_('You need to configure mailgun.validation_key' |
|||
' in order to be able to check mails validity')) |
|||
for partner in self: |
|||
res = requests.get( |
|||
"%s/address/validate" % api_url, |
|||
auth=("api", validation_key), params={ |
|||
"address": partner.email, |
|||
"mailbox_verification": True, |
|||
}) |
|||
if not res or res.status_code != 200: |
|||
raise UserError(_( |
|||
'Error %s trying to ' |
|||
'check mail' % res.status_code or 'of connection')) |
|||
content = json.loads(res.content, res.apparent_encoding) |
|||
if 'mailbox_verification' not in content: |
|||
raise UserError( |
|||
_("Mailgun Error. Mailbox verification value wasn't" |
|||
" returned")) |
|||
# Not a valid address: API sets 'is_valid' as False |
|||
# and 'mailbox_verification' as None |
|||
if not content['is_valid']: |
|||
partner.email_bounced = True |
|||
raise UserError( |
|||
_('%s is not a valid email address. Please check it ' |
|||
'in order to avoid sending issues') % (partner.email)) |
|||
# If the mailbox is not valid API returns 'mailbox_verification' |
|||
# as a string with value 'false' |
|||
if content['mailbox_verification'] == 'false': |
|||
partner.email_bounced = True |
|||
raise UserError( |
|||
_('%s failed the mailbox verification. Please check it ' |
|||
'in order to avoid sending issues') % (partner.email)) |
|||
# If Mailgun can't complete the validation request the API returns |
|||
# 'mailbox_verification' as a string set to 'unknown' |
|||
if content['mailbox_verification'] == 'unknown': |
|||
raise UserError( |
|||
_("%s couldn't be verified. Either the request couln't be " |
|||
"completed or the mailbox provider doesn't support " |
|||
"email verification") % (partner.email)) |
|||
|
|||
@api.multi |
|||
def check_email_bounced(self): |
|||
""" |
|||
Checks if the partner's email is in Mailgun's bounces list |
|||
API documentation: |
|||
https://documentation.mailgun.com/en/latest/api-suppressions.html |
|||
""" |
|||
api_key, api_url, domain, validation_key = self.env[ |
|||
'mail.tracking.email']._mailgun_values() |
|||
for partner in self: |
|||
res = requests.get( |
|||
'%s/%s/bounces/%s' % (api_url, domain, partner.email), |
|||
auth=("api", api_key)) |
|||
if res.status_code == 200 and not partner.email_bounced: |
|||
partner.email_bounced = True |
|||
elif res.status_code == 404 and partner.email_bounced: |
|||
partner.email_bounced = False |
|||
|
|||
@api.multi |
|||
def force_set_bounced(self): |
|||
""" |
|||
Forces partner's email into Mailgun's bounces list |
|||
API documentation: |
|||
https://documentation.mailgun.com/en/latest/api-suppressions.html |
|||
""" |
|||
api_key, api_url, domain, validation_key = self.env[ |
|||
'mail.tracking.email']._mailgun_values() |
|||
for partner in self: |
|||
res = requests.post( |
|||
'%s/%s/bounces' % (api_url, domain), |
|||
auth=("api", api_key), |
|||
data={'address': partner.email}) |
|||
partner.email_bounced = ( |
|||
res.status_code == 200 and not partner.email_bounced) |
|||
|
|||
@api.multi |
|||
def force_unset_bounced(self): |
|||
""" |
|||
Forces partner's email deletion from Mailgun's bounces list |
|||
API documentation: |
|||
https://documentation.mailgun.com/en/latest/api-suppressions.html |
|||
""" |
|||
api_key, api_url, domain, validation_key = self.env[ |
|||
'mail.tracking.email']._mailgun_values() |
|||
for partner in self: |
|||
res = requests.delete( |
|||
'%s/%s/bounces/%s' % (api_url, domain, partner.email), |
|||
auth=("api", api_key)) |
|||
if res.status_code in (200, 404) and partner.email_bounced: |
|||
partner.email_bounced = False |
@ -1,5 +1,4 @@ |
|||
# -*- coding: utf-8 -*- |
|||
# Copyright 2016 Antonio Espinosa - <antonio.espinosa@tecnativa.com> |
|||
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). |
|||
|
|||
from . import test_mailgun |
@ -0,0 +1,16 @@ |
|||
<?xml version="1.0" encoding="utf-8"?> |
|||
<odoo> |
|||
|
|||
<record id="mailgun_manual_check" model="ir.ui.view"> |
|||
<field name="name">Manual Mailgun check</field> |
|||
<field name="model">mail.tracking.email</field> |
|||
<field name="inherit_id" ref="mail_tracking.view_mail_tracking_email_form"/> |
|||
<field name="arch" type="xml"> |
|||
<field name="state" position="before"> |
|||
<button name="action_manual_check_mailgun" |
|||
type="object" string="Re-sync Mailgun"/> |
|||
</field> |
|||
</field> |
|||
</record> |
|||
|
|||
</odoo> |
@ -0,0 +1,28 @@ |
|||
<?xml version="1.0" encoding="utf-8"?> |
|||
<!-- Copyright 2016 Tecnativa - Pedro M. Baeza |
|||
Copyright 2016 Carlos Dauden - Tecnativa <carlos.dauden@tecnativa.com> |
|||
Copyright 2017 Tecnativa <vicent.cubellsn@tecnativa.com> |
|||
License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl.html). --> |
|||
<odoo> |
|||
|
|||
<record id="view_partner_form_mailgun" model="ir.ui.view"> |
|||
<field name="name">Partner Mailgun button</field> |
|||
<field name="model">res.partner</field> |
|||
<field name="inherit_id" ref="mail_tracking.view_partner_form"/> |
|||
<field name="arch" type="xml"> |
|||
<field name="email_bounced" position="after"> |
|||
<label for="check_email_bounced" string="Mailgun" |
|||
attrs="{'invisible': [('email', '=', False)]}"/> |
|||
<group col="3" name="mailgun_buttons" attrs="{'invisible': [('email', '=', False)]}"> |
|||
<button name="check_email_bounced" type="object" string="Check Mailgun" class="oe_link"/> |
|||
<button name="check_email_validity" type="object" string="Check email validity" class="oe_link"/> |
|||
<button name="force_set_bounced" type="object" string="Set Bounced" class="oe_link" |
|||
attrs="{'invisible': [('email_bounced', '=', True)]}"/> |
|||
<button name="force_unset_bounced" type="object" string="Unset Bounced" class="oe_link" |
|||
attrs="{'invisible': [('email_bounced', '=', False)]}"/> |
|||
</group> |
|||
</field> |
|||
</field> |
|||
</record> |
|||
|
|||
</odoo> |
Write
Preview
Loading…
Cancel
Save
Reference in new issue