[IMP] mail_tracking: Store To recipients and omit aliases
This commit is contained in:
parent
103192362e
commit
bedeb56759
@ -76,6 +76,9 @@ These are all available status icons:
|
|||||||
.. |noemail| image:: https://raw.githubusercontent.com/OCA/social/13.0/mail_tracking/static/src/img/no_email.png
|
.. |noemail| image:: https://raw.githubusercontent.com/OCA/social/13.0/mail_tracking/static/src/img/no_email.png
|
||||||
:width: 10px
|
:width: 10px
|
||||||
|
|
||||||
|
.. |anonuser| image:: https://raw.githubusercontent.com/OCA/social/13.0/mail_tracking/static/src/img/anon_user.png
|
||||||
|
:width: 10px
|
||||||
|
|
||||||
|unknown| **Unknown**: No email tracking info available. Maybe this notified partner has 'Receive Inbox Notifications by Email' == 'Never'
|
|unknown| **Unknown**: No email tracking info available. Maybe this notified partner has 'Receive Inbox Notifications by Email' == 'Never'
|
||||||
|
|
||||||
|waiting| **Waiting**: Waiting to be sent
|
|waiting| **Waiting**: Waiting to be sent
|
||||||
@ -92,6 +95,8 @@ These are all available status icons:
|
|||||||
|
|
||||||
|noemail| **No Email**: The partner doesn't have a defined email
|
|noemail| **No Email**: The partner doesn't have a defined email
|
||||||
|
|
||||||
|
|anonuser| **No Partner**: The recipient doesn't have a defined partner
|
||||||
|
|
||||||
|
|
||||||
If you want to see all tracking emails and events you can go to
|
If you want to see all tracking emails and events you can go to
|
||||||
|
|
||||||
|
@ -9,3 +9,4 @@ from . import mail_tracking_event
|
|||||||
from . import res_partner
|
from . import res_partner
|
||||||
from . import mail_thread
|
from . import mail_thread
|
||||||
from . import mail_resend_message
|
from . import mail_resend_message
|
||||||
|
from . import mail_alias
|
||||||
|
33
mail_tracking/models/mail_alias.py
Normal file
33
mail_tracking/models/mail_alias.py
Normal file
@ -0,0 +1,33 @@
|
|||||||
|
# Copyright 2020 Alexandre Díaz
|
||||||
|
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl.html).
|
||||||
|
|
||||||
|
from odoo import api, models, tools
|
||||||
|
|
||||||
|
|
||||||
|
class MailAlias(models.Model):
|
||||||
|
_inherit = "mail.alias"
|
||||||
|
|
||||||
|
@api.model
|
||||||
|
@tools.ormcache()
|
||||||
|
def get_aliases(self):
|
||||||
|
return {
|
||||||
|
x["display_name"]
|
||||||
|
for x in self.search_read([("alias_name", "!=", False)], ["display_name"])
|
||||||
|
}
|
||||||
|
|
||||||
|
@api.model_create_multi
|
||||||
|
def create(self, vals_list):
|
||||||
|
res = super().create(vals_list)
|
||||||
|
self.clear_caches()
|
||||||
|
return res
|
||||||
|
|
||||||
|
def write(self, vals):
|
||||||
|
res = super().write(vals)
|
||||||
|
if "alias_name" in vals:
|
||||||
|
self.clear_caches()
|
||||||
|
return res
|
||||||
|
|
||||||
|
def unlink(self):
|
||||||
|
res = super().unlink()
|
||||||
|
self.clear_caches()
|
||||||
|
return res
|
@ -2,6 +2,8 @@
|
|||||||
# Copyright 2019 Alexandre Díaz
|
# Copyright 2019 Alexandre Díaz
|
||||||
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl.html).
|
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl.html).
|
||||||
|
|
||||||
|
from email.utils import getaddresses
|
||||||
|
|
||||||
from odoo import _, api, fields, models
|
from odoo import _, api, fields, models
|
||||||
from odoo.tools import email_split
|
from odoo.tools import email_split
|
||||||
|
|
||||||
@ -13,6 +15,7 @@ class MailMessage(models.Model):
|
|||||||
email_cc = fields.Char(
|
email_cc = fields.Char(
|
||||||
"Cc", help="Additional recipients that receive a " '"Carbon Copy" of the e-mail'
|
"Cc", help="Additional recipients that receive a " '"Carbon Copy" of the e-mail'
|
||||||
)
|
)
|
||||||
|
email_to = fields.Char("To", help="Raw TO recipients")
|
||||||
mail_tracking_ids = fields.One2many(
|
mail_tracking_ids = fields.One2many(
|
||||||
comodel_name="mail.tracking.email",
|
comodel_name="mail.tracking.email",
|
||||||
inverse_name="mail_message_id",
|
inverse_name="mail_message_id",
|
||||||
@ -105,11 +108,16 @@ class MailMessage(models.Model):
|
|||||||
.sudo()
|
.sudo()
|
||||||
.search([("mail_message_id", "=", message.id)])
|
.search([("mail_message_id", "=", message.id)])
|
||||||
)
|
)
|
||||||
# Get Cc recipients
|
# String to List
|
||||||
email_cc_list = email_split(message.email_cc)
|
email_cc_list = self._drop_aliases(email_split(message.email_cc))
|
||||||
if any(email_cc_list):
|
email_to_list = self._drop_aliases(email_split(message.email_to))
|
||||||
partners |= partners.search([("email", "in", email_cc_list)])
|
# Search related partners recipients
|
||||||
|
partners |= partners.search(
|
||||||
|
[("email", "in", email_cc_list + email_to_list)]
|
||||||
|
)
|
||||||
|
# Operate over set's instead of lists
|
||||||
email_cc_list = set(email_cc_list)
|
email_cc_list = set(email_cc_list)
|
||||||
|
email_to_list = set(email_to_list) - email_cc_list
|
||||||
# Search all trackings for this message
|
# Search all trackings for this message
|
||||||
for tracking in trackings:
|
for tracking in trackings:
|
||||||
status = self._partner_tracking_status_get(tracking)
|
status = self._partner_tracking_status_get(tracking)
|
||||||
@ -127,45 +135,66 @@ class MailMessage(models.Model):
|
|||||||
}
|
}
|
||||||
)
|
)
|
||||||
if tracking.partner_id:
|
if tracking.partner_id:
|
||||||
|
# Discard mails with tracking
|
||||||
email_cc_list.discard(tracking.partner_id.email)
|
email_cc_list.discard(tracking.partner_id.email)
|
||||||
|
email_to_list.discard(tracking.partner_id.email)
|
||||||
partners_already |= tracking.partner_id
|
partners_already |= tracking.partner_id
|
||||||
# Search all recipients for this message
|
# Search all partner recipients for this message
|
||||||
if message.partner_ids:
|
if message.partner_ids:
|
||||||
partners |= message.partner_ids
|
partners |= message.partner_ids
|
||||||
if message.notified_partner_ids:
|
if message.notified_partner_ids:
|
||||||
partners |= message.notified_partner_ids
|
partners |= message.notified_partner_ids
|
||||||
# Remove recipients already included
|
# Discard partner recipients already included
|
||||||
partners -= partners_already
|
partners -= partners_already
|
||||||
tracking_unkown_values = {
|
# Default tracking values
|
||||||
|
tracking_unknown_values = {
|
||||||
"status": "unknown",
|
"status": "unknown",
|
||||||
"status_human": self._partner_tracking_status_human_get("unknown"),
|
"status_human": self._partner_tracking_status_human_get("unknown"),
|
||||||
"error_type": False,
|
"error_type": False,
|
||||||
"error_description": False,
|
"error_description": False,
|
||||||
"tracking_id": False,
|
"tracking_id": False,
|
||||||
}
|
}
|
||||||
|
# Process tracking status of partner recipients without tracking
|
||||||
for partner in partners:
|
for partner in partners:
|
||||||
|
# Discard 'To' with partner
|
||||||
|
if partner.email in email_to_list:
|
||||||
|
email_to_list.discard(partner.email)
|
||||||
# If there is partners not included, then status is 'unknown'
|
# If there is partners not included, then status is 'unknown'
|
||||||
# and perhaps a Cc recipient
|
# and perhaps a Cc recipient
|
||||||
isCc = False
|
isCc = False
|
||||||
if partner.email in email_cc_list:
|
if partner.email in email_cc_list:
|
||||||
email_cc_list.discard(partner.email)
|
email_cc_list.discard(partner.email)
|
||||||
isCc = True
|
isCc = True
|
||||||
tracking_unkown_values.update(
|
tracking_status = tracking_unknown_values.copy()
|
||||||
|
tracking_status.update(
|
||||||
{"recipient": partner.name, "partner_id": partner.id, "isCc": isCc}
|
{"recipient": partner.name, "partner_id": partner.id, "isCc": isCc}
|
||||||
)
|
)
|
||||||
partner_trackings.append(tracking_unkown_values.copy())
|
partner_trackings.append(tracking_status)
|
||||||
for email in email_cc_list:
|
# Process Cc/To recipients without partner
|
||||||
# If there is Cc without partner
|
for cc, lst in [(True, email_cc_list), (False, email_to_list)]:
|
||||||
tracking_unkown_values.update(
|
for email in lst:
|
||||||
{"recipient": email, "partner_id": False, "isCc": True}
|
tracking_status = tracking_unknown_values.copy()
|
||||||
|
tracking_status.update(
|
||||||
|
{"recipient": email, "partner_id": False, "isCc": cc}
|
||||||
)
|
)
|
||||||
partner_trackings.append(tracking_unkown_values.copy())
|
partner_trackings.append(tracking_status)
|
||||||
res[message.id] = {
|
res[message.id] = {
|
||||||
"partner_trackings": partner_trackings,
|
"partner_trackings": partner_trackings,
|
||||||
"is_failed_message": message.is_failed_message,
|
"is_failed_message": message.is_failed_message,
|
||||||
}
|
}
|
||||||
return res
|
return res
|
||||||
|
|
||||||
|
@api.model
|
||||||
|
def _drop_aliases(self, mail_list):
|
||||||
|
aliases = self.env["mail.alias"].get_aliases()
|
||||||
|
|
||||||
|
def _filter_alias(email):
|
||||||
|
email_wn = getaddresses([email])[0][1]
|
||||||
|
if email_wn not in aliases:
|
||||||
|
return email_wn
|
||||||
|
|
||||||
|
return list(filter(_filter_alias, mail_list))
|
||||||
|
|
||||||
@api.model
|
@api.model
|
||||||
def _message_read_dict_postprocess(self, messages, message_tree):
|
def _message_read_dict_postprocess(self, messages, message_tree):
|
||||||
"""Preare values to be used by the chatter widget"""
|
"""Preare values to be used by the chatter widget"""
|
||||||
|
@ -33,41 +33,48 @@ class MailThread(models.AbstractModel):
|
|||||||
def message_post(self, *args, **kwargs):
|
def message_post(self, *args, **kwargs):
|
||||||
"""Adds CC recipient to the message.
|
"""Adds CC recipient to the message.
|
||||||
|
|
||||||
Because Odoo implementation avoid store cc recipients we ensure that
|
Because Odoo implementation avoid store 'from, to, cc' recipients we
|
||||||
this information its written into the mail.message record.
|
ensure that this information its written into the mail.message record.
|
||||||
"""
|
"""
|
||||||
new_message = super().message_post(*args, **kwargs)
|
kwargs.update(
|
||||||
email_cc = kwargs.get("cc")
|
{"email_cc": kwargs.get("cc", False), "email_to": kwargs.get("to", False)}
|
||||||
if email_cc:
|
)
|
||||||
new_message.sudo().write({"email_cc": email_cc})
|
return super().message_post(*args, **kwargs)
|
||||||
return new_message
|
|
||||||
|
|
||||||
def _message_get_suggested_recipients(self):
|
def _message_get_suggested_recipients(self):
|
||||||
"""Adds email Cc recipients as suggested recipients.
|
"""Adds email 'extra' recipients as suggested recipients.
|
||||||
|
|
||||||
If the recipient has a res.partner, use it.
|
If the recipient has a res.partner, use it.
|
||||||
"""
|
"""
|
||||||
res = super()._message_get_suggested_recipients()
|
res = super()._message_get_suggested_recipients()
|
||||||
|
self._add_extra_recipients_suggestions(res, "email_cc", _("Cc"))
|
||||||
|
self._add_extra_recipients_suggestions(res, "email_to", _("Anon. To"))
|
||||||
|
return res
|
||||||
|
|
||||||
|
def _add_extra_recipients_suggestions(self, suggestions, field_mail, reason):
|
||||||
ResPartnerObj = self.env["res.partner"]
|
ResPartnerObj = self.env["res.partner"]
|
||||||
email_cc_formated_list = []
|
aliases = self.env["mail.alias"].get_aliases()
|
||||||
|
email_extra_formated_list = []
|
||||||
for record in self:
|
for record in self:
|
||||||
emails_cc = record.message_ids.mapped("email_cc")
|
emails_extra = record.message_ids.mapped(field_mail)
|
||||||
for email in emails_cc:
|
for email in emails_extra:
|
||||||
email_cc_formated_list.extend(email_split_and_format(email))
|
email_extra_formated_list.extend(email_split_and_format(email))
|
||||||
email_cc_formated_list = set(email_cc_formated_list)
|
email_extra_formated_list = set(email_extra_formated_list)
|
||||||
for cc in email_cc_formated_list:
|
email_extra_list = [x[1] for x in getaddresses(email_extra_formated_list)]
|
||||||
email_parts = getaddresses([cc])[0]
|
partners_info = self._message_partner_info_from_emails(email_extra_list)
|
||||||
partner_id = record._message_partner_info_from_emails([email_parts[1]])[
|
for pinfo in partners_info:
|
||||||
0
|
partner_id = pinfo["partner_id"]
|
||||||
].get("partner_id")
|
email = pinfo["full_name"]
|
||||||
if not partner_id:
|
if not partner_id:
|
||||||
record._message_add_suggested_recipient(res, email=cc, reason=_("Cc"))
|
if email not in aliases:
|
||||||
|
self._message_add_suggested_recipient(
|
||||||
|
suggestions, email=email, reason=reason
|
||||||
|
)
|
||||||
else:
|
else:
|
||||||
partner = ResPartnerObj.browse(partner_id)
|
partner = ResPartnerObj.browse(partner_id)
|
||||||
record._message_add_suggested_recipient(
|
self._message_add_suggested_recipient(
|
||||||
res, partner=partner, reason=_("Cc")
|
suggestions, partner=partner, reason=reason
|
||||||
)
|
)
|
||||||
return res
|
|
||||||
|
|
||||||
@api.model
|
@api.model
|
||||||
def _fields_view_get(
|
def _fields_view_get(
|
||||||
|
@ -28,6 +28,9 @@ These are all available status icons:
|
|||||||
.. |noemail| image:: ../static/src/img/no_email.png
|
.. |noemail| image:: ../static/src/img/no_email.png
|
||||||
:width: 10px
|
:width: 10px
|
||||||
|
|
||||||
|
.. |anonuser| image:: ../static/src/img/anon_user.png
|
||||||
|
:width: 10px
|
||||||
|
|
||||||
|unknown| **Unknown**: No email tracking info available. Maybe this notified partner has 'Receive Inbox Notifications by Email' == 'Never'
|
|unknown| **Unknown**: No email tracking info available. Maybe this notified partner has 'Receive Inbox Notifications by Email' == 'Never'
|
||||||
|
|
||||||
|waiting| **Waiting**: Waiting to be sent
|
|waiting| **Waiting**: Waiting to be sent
|
||||||
@ -44,6 +47,8 @@ These are all available status icons:
|
|||||||
|
|
||||||
|noemail| **No Email**: The partner doesn't have a defined email
|
|noemail| **No Email**: The partner doesn't have a defined email
|
||||||
|
|
||||||
|
|anonuser| **No Partner**: The recipient doesn't have a defined partner
|
||||||
|
|
||||||
|
|
||||||
If you want to see all tracking emails and events you can go to
|
If you want to see all tracking emails and events you can go to
|
||||||
|
|
||||||
|
@ -412,6 +412,7 @@ status icon will appear just right to name of notified partner.</p>
|
|||||||
<p><img alt="opened" src="https://raw.githubusercontent.com/OCA/social/13.0/mail_tracking/static/src/img/opened.png" style="width: 15px;" /> <strong>Opened</strong>: Opened by partner</p>
|
<p><img alt="opened" src="https://raw.githubusercontent.com/OCA/social/13.0/mail_tracking/static/src/img/opened.png" style="width: 15px;" /> <strong>Opened</strong>: Opened by partner</p>
|
||||||
<p><img alt="cc" src="https://raw.githubusercontent.com/OCA/social/13.0/mail_tracking/static/src/img/cc.png" style="width: 10px;" /> <strong>Cc</strong>: It’s a Carbon-Copy recipient. Can’t know the status so is ‘Unknown’</p>
|
<p><img alt="cc" src="https://raw.githubusercontent.com/OCA/social/13.0/mail_tracking/static/src/img/cc.png" style="width: 10px;" /> <strong>Cc</strong>: It’s a Carbon-Copy recipient. Can’t know the status so is ‘Unknown’</p>
|
||||||
<p><img alt="noemail" src="https://raw.githubusercontent.com/OCA/social/13.0/mail_tracking/static/src/img/no_email.png" style="width: 10px;" /> <strong>No Email</strong>: The partner doesn’t have a defined email</p>
|
<p><img alt="noemail" src="https://raw.githubusercontent.com/OCA/social/13.0/mail_tracking/static/src/img/no_email.png" style="width: 10px;" /> <strong>No Email</strong>: The partner doesn’t have a defined email</p>
|
||||||
|
<p><img alt="anonuser" src="https://raw.githubusercontent.com/OCA/social/13.0/mail_tracking/static/src/img/anon_user.png" style="width: 10px;" /> <strong>No Partner</strong>: The recipient doesn’t have a defined partner</p>
|
||||||
<p>If you want to see all tracking emails and events you can go to</p>
|
<p>If you want to see all tracking emails and events you can go to</p>
|
||||||
<ul class="simple">
|
<ul class="simple">
|
||||||
<li>Settings > Technical > Email > Tracking emails</li>
|
<li>Settings > Technical > Email > Tracking emails</li>
|
||||||
|
BIN
mail_tracking/static/src/img/anon_user.png
Normal file
BIN
mail_tracking/static/src/img/anon_user.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 483 B |
@ -10,6 +10,11 @@
|
|||||||
<i class="fa fa-cc"></i>
|
<i class="fa fa-cc"></i>
|
||||||
</span>
|
</span>
|
||||||
</t>
|
</t>
|
||||||
|
<t t-elif="!tracking['isCc'] && !tracking['partner_id']">
|
||||||
|
<span class="mail_anon_recipient">
|
||||||
|
<i class="fa fa-low-vision"></i>
|
||||||
|
</span>
|
||||||
|
</t>
|
||||||
<t t-elif="tracking['status'] === 'unknown'">
|
<t t-elif="tracking['status'] === 'unknown'">
|
||||||
<span class="mail_tracking_unknown">
|
<span class="mail_tracking_unknown">
|
||||||
<i class="fa fa-ban"></i>
|
<i class="fa fa-ban"></i>
|
||||||
|
@ -9,7 +9,7 @@ import psycopg2
|
|||||||
import psycopg2.errorcodes
|
import psycopg2.errorcodes
|
||||||
from lxml import etree
|
from lxml import etree
|
||||||
|
|
||||||
from odoo import _, http
|
from odoo import http
|
||||||
from odoo.tests.common import TransactionCase
|
from odoo.tests.common import TransactionCase
|
||||||
from odoo.tools import mute_logger
|
from odoo.tools import mute_logger
|
||||||
|
|
||||||
@ -179,8 +179,9 @@ class TestMailTracking(TransactionCase):
|
|||||||
"login": "sender-test",
|
"login": "sender-test",
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
|
# pylint: disable=C8107
|
||||||
message = self.recipient.with_user(sender_user).message_post(
|
message = self.recipient.with_user(sender_user).message_post(
|
||||||
body=_("<p>This is a test message</p>"),
|
body="<p>This is a test message</p>",
|
||||||
cc="unnamed@test.com, sender@example.com",
|
cc="unnamed@test.com, sender@example.com",
|
||||||
)
|
)
|
||||||
# suggested recipients
|
# suggested recipients
|
||||||
|
Loading…
Reference in New Issue
Block a user