2017-10-17 20:38:11 +02:00
|
|
|
# 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
|
|
|
|
|
2017-12-22 19:23:40 +01:00
|
|
|
from odoo import _, api, models
|
|
|
|
from odoo.exceptions import UserError
|
2017-10-17 20:38:11 +02:00
|
|
|
|
|
|
|
|
|
|
|
class ResPartner(models.Model):
|
2020-04-02 11:15:00 +02:00
|
|
|
_inherit = "res.partner"
|
2017-10-17 20:38:11 +02:00
|
|
|
|
|
|
|
@api.multi
|
|
|
|
def email_bounced_set(self, tracking_emails, reason, event=None):
|
|
|
|
res = super(ResPartner, self).email_bounced_set(
|
2020-04-02 11:15:00 +02:00
|
|
|
tracking_emails, reason, event=event
|
|
|
|
)
|
2017-10-17 20:38:11 +02:00
|
|
|
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
|
2020-04-02 11:15:00 +02:00
|
|
|
event = event or self.env["mail.tracking.event"]
|
2019-09-05 16:46:22 +02:00
|
|
|
event_str = """
|
|
|
|
<a href="#"
|
|
|
|
data-oe-model="mail.tracking.event" data-oe-id="%d">%s</a>
|
2020-04-02 11:15:00 +02:00
|
|
|
""" % (
|
|
|
|
event.id or 0,
|
|
|
|
event.id or _("unknown"),
|
|
|
|
)
|
|
|
|
body = _("Email has been bounced: %s\nReason: %s\nEvent: %s") % (
|
|
|
|
partner.email,
|
|
|
|
reason,
|
|
|
|
event_str,
|
|
|
|
)
|
2017-10-17 20:38:11 +02:00
|
|
|
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[
|
2020-04-02 11:15:00 +02:00
|
|
|
"mail.tracking.email"
|
|
|
|
]._mailgun_values()
|
2017-10-17 20:38:11 +02:00
|
|
|
if not validation_key:
|
2020-04-02 11:15:00 +02:00
|
|
|
raise UserError(
|
|
|
|
_(
|
|
|
|
"You need to configure mailgun.validation_key"
|
|
|
|
" in order to be able to check mails validity"
|
|
|
|
)
|
|
|
|
)
|
|
|
|
for partner in self.filtered("email"):
|
2017-10-17 20:38:11 +02:00
|
|
|
res = requests.get(
|
2018-08-02 10:53:59 +02:00
|
|
|
# Validation API url is always the same
|
2020-04-02 11:15:00 +02:00
|
|
|
"https://api.mailgun.net/v3/address/validate",
|
|
|
|
auth=("api", validation_key),
|
|
|
|
params={"address": partner.email, "mailbox_verification": True},
|
|
|
|
)
|
|
|
|
if (
|
|
|
|
not res
|
|
|
|
or res.status_code != 200
|
|
|
|
and not self.env.context.get("mailgun_auto_check")
|
|
|
|
):
|
|
|
|
raise UserError(
|
|
|
|
_(
|
|
|
|
"Error %s trying to check mail" % res.status_code
|
|
|
|
or "of connection"
|
|
|
|
)
|
|
|
|
)
|
2018-05-08 18:24:36 +02:00
|
|
|
content = res.json()
|
2020-04-02 11:15:00 +02:00
|
|
|
if "mailbox_verification" not in content:
|
|
|
|
if not self.env.context.get("mailgun_auto_check"):
|
2018-02-16 13:15:32 +01:00
|
|
|
raise UserError(
|
2020-04-02 11:15:00 +02:00
|
|
|
_(
|
|
|
|
"Mailgun Error. Mailbox verification value wasn't"
|
|
|
|
" returned"
|
|
|
|
)
|
|
|
|
)
|
2017-10-17 20:38:11 +02:00
|
|
|
# Not a valid address: API sets 'is_valid' as False
|
|
|
|
# and 'mailbox_verification' as None
|
2020-04-02 11:15:00 +02:00
|
|
|
if not content["is_valid"]:
|
2017-10-17 20:38:11 +02:00
|
|
|
partner.email_bounced = True
|
2020-04-02 11:15:00 +02:00
|
|
|
body = (
|
|
|
|
_(
|
|
|
|
"%s is not a valid email address. Please check it"
|
|
|
|
" in order to avoid sending issues"
|
|
|
|
)
|
|
|
|
% partner.email
|
|
|
|
)
|
|
|
|
if not self.env.context.get("mailgun_auto_check"):
|
2018-02-16 13:15:32 +01:00
|
|
|
raise UserError(body)
|
|
|
|
partner.message_post(body=body)
|
2017-10-17 20:38:11 +02:00
|
|
|
# If the mailbox is not valid API returns 'mailbox_verification'
|
|
|
|
# as a string with value 'false'
|
2020-04-02 11:15:00 +02:00
|
|
|
if content["mailbox_verification"] == "false":
|
2017-10-17 20:38:11 +02:00
|
|
|
partner.email_bounced = True
|
2020-04-02 11:15:00 +02:00
|
|
|
body = (
|
|
|
|
_(
|
|
|
|
"%s failed the mailbox verification. Please check it"
|
|
|
|
" in order to avoid sending issues"
|
|
|
|
)
|
|
|
|
% partner.email
|
|
|
|
)
|
|
|
|
if not self.env.context.get("mailgun_auto_check"):
|
2018-02-16 13:15:32 +01:00
|
|
|
raise UserError(body)
|
|
|
|
partner.message_post(body=body)
|
2017-10-17 20:38:11 +02:00
|
|
|
# If Mailgun can't complete the validation request the API returns
|
|
|
|
# 'mailbox_verification' as a string set to 'unknown'
|
2020-04-02 11:15:00 +02:00
|
|
|
if content["mailbox_verification"] == "unknown":
|
|
|
|
if not self.env.context.get("mailgun_auto_check"):
|
2018-02-16 13:15:32 +01:00
|
|
|
raise UserError(
|
2020-04-02 11:15:00 +02:00
|
|
|
_(
|
|
|
|
"%s couldn't be verified. Either the request couln't"
|
|
|
|
" be completed or the mailbox provider doesn't "
|
|
|
|
"support email verification"
|
|
|
|
)
|
|
|
|
% (partner.email)
|
|
|
|
)
|
2017-10-17 20:38:11 +02:00
|
|
|
|
|
|
|
@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[
|
2020-04-02 11:15:00 +02:00
|
|
|
"mail.tracking.email"
|
|
|
|
]._mailgun_values()
|
2017-10-17 20:38:11 +02:00
|
|
|
for partner in self:
|
|
|
|
res = requests.get(
|
2020-04-02 11:15:00 +02:00
|
|
|
"{}/{}/bounces/{}".format(api_url, domain, partner.email),
|
|
|
|
auth=("api", api_key),
|
|
|
|
)
|
2017-10-17 20:38:11 +02:00
|
|
|
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[
|
2020-04-02 11:15:00 +02:00
|
|
|
"mail.tracking.email"
|
|
|
|
]._mailgun_values()
|
2017-10-17 20:38:11 +02:00
|
|
|
for partner in self:
|
|
|
|
res = requests.post(
|
2020-04-02 11:15:00 +02:00
|
|
|
"{}/{}/bounces".format(api_url, domain),
|
2017-10-17 20:38:11 +02:00
|
|
|
auth=("api", api_key),
|
2020-04-02 11:15:00 +02:00
|
|
|
data={"address": partner.email},
|
|
|
|
)
|
|
|
|
partner.email_bounced = res.status_code == 200 and not partner.email_bounced
|
2017-10-17 20:38:11 +02:00
|
|
|
|
|
|
|
@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[
|
2020-04-02 11:15:00 +02:00
|
|
|
"mail.tracking.email"
|
|
|
|
]._mailgun_values()
|
2017-10-17 20:38:11 +02:00
|
|
|
for partner in self:
|
|
|
|
res = requests.delete(
|
2020-04-02 11:15:00 +02:00
|
|
|
"{}/{}/bounces/{}".format(api_url, domain, partner.email),
|
|
|
|
auth=("api", api_key),
|
|
|
|
)
|
2017-10-17 20:38:11 +02:00
|
|
|
if res.status_code in (200, 404) and partner.email_bounced:
|
|
|
|
partner.email_bounced = False
|
2018-02-16 13:15:32 +01:00
|
|
|
|
|
|
|
def _autocheck_partner_email(self):
|
|
|
|
for partner in self:
|
2020-04-02 11:15:00 +02:00
|
|
|
partner.with_context(mailgun_auto_check=True).check_email_validity()
|
2018-02-16 13:15:32 +01:00
|
|
|
|
|
|
|
@api.model
|
|
|
|
def create(self, vals):
|
2020-04-02 11:15:00 +02:00
|
|
|
if "email" in vals and self.env["ir.config_parameter"].sudo().get_param(
|
|
|
|
"mailgun.auto_check_partner_email"
|
|
|
|
):
|
2018-02-16 13:15:32 +01:00
|
|
|
self._autocheck_partner_email()
|
|
|
|
return super(ResPartner, self).create(vals)
|
|
|
|
|
|
|
|
def write(self, vals):
|
2020-04-02 11:15:00 +02:00
|
|
|
if "email" in vals and self.env["ir.config_parameter"].sudo().get_param(
|
|
|
|
"mailgun.auto_check_partner_email"
|
|
|
|
):
|
2018-02-16 13:15:32 +01:00
|
|
|
self._autocheck_partner_email()
|
|
|
|
return super(ResPartner, self).write(vals)
|