[FIX] mass_mailing_list_dynamic, mass_mailing_partner: allow bouncing synced contacts

The previous implementation in `mass_mailing_partner` overwrote `create()` and `write()` in a way that always updated all fields.

However, `mass_mailing_list_dynamic` added a constraint on 4 fields, called `_check_no_manual_edits_on_fully_synced_lists()`.

The combination of these 2 things made that constraint to be checked *always*, regardless on which fields were being updated.

Thus, when sending a mass mailing based on a fully synced list, and processing bounces, we would get errors always. Even when the `message_bounce` field shouldn't be constrained.

@moduon MT-8513
pull/1529/head
Jairo Llopis 2024-12-26 10:12:31 +00:00
parent ddf6d2ef9e
commit 8ac9269860
No known key found for this signature in database
GPG Key ID: B24A1D10508180D8
10 changed files with 86 additions and 42 deletions

View File

@ -126,6 +126,10 @@ Contributors
* Nguyễn Minh Chiến <chien@trobz.com>
* `Moduon <https://www.moduon.team>`_:
* Jairo Llopis
Other credits
~~~~~~~~~~~~~

View File

@ -17,3 +17,7 @@
* `Trobz <https://trobz.com>`_:
* Nguyễn Minh Chiến <chien@trobz.com>
* `Moduon <https://www.moduon.team>`_:
* Jairo Llopis

View File

@ -1,4 +1,3 @@
<?xml version="1.0" encoding="utf-8"?>
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml" xml:lang="en" lang="en">
<head>
@ -9,10 +8,11 @@
/*
:Author: David Goodger (goodger@python.org)
:Id: $Id: html4css1.css 8954 2022-01-20 10:10:25Z milde $
:Id: $Id: html4css1.css 9511 2024-01-13 09:50:07Z milde $
:Copyright: This stylesheet has been placed in the public domain.
Default cascading style sheet for the HTML output of Docutils.
Despite the name, some widely supported CSS2 features are used.
See https://docutils.sourceforge.io/docs/howto/html-stylesheets.html for how to
customize this style sheet.
@ -275,7 +275,7 @@ pre.literal-block, pre.doctest-block, pre.math, pre.code {
margin-left: 2em ;
margin-right: 2em }
pre.code .ln { color: grey; } /* line numbers */
pre.code .ln { color: gray; } /* line numbers */
pre.code, code { background-color: #eeeeee }
pre.code .comment, code .comment { color: #5C6576 }
pre.code .keyword, code .keyword { color: #3B0D06; font-weight: bold }
@ -301,7 +301,7 @@ span.option {
span.pre {
white-space: pre }
span.problematic {
span.problematic, pre.problematic {
color: red }
span.section-subtitle {
@ -479,6 +479,13 @@ If you spotted it first, help us to smash it by providing a detailed and welcome
</ul>
</blockquote>
</li>
<li><p class="first"><a class="reference external" href="https://www.moduon.team">Moduon</a>:</p>
<blockquote>
<ul class="simple">
<li>Jairo Llopis</li>
</ul>
</blockquote>
</li>
</ul>
</div>
<div class="section" id="other-credits">
@ -488,7 +495,9 @@ If you spotted it first, help us to smash it by providing a detailed and welcome
<div class="section" id="maintainers">
<h2><a class="toc-backref" href="#toc-entry-8">Maintainers</a></h2>
<p>This module is maintained by the OCA.</p>
<a class="reference external image-reference" href="https://odoo-community.org"><img alt="Odoo Community Association" src="https://odoo-community.org/logo.png" /></a>
<a class="reference external image-reference" href="https://odoo-community.org">
<img alt="Odoo Community Association" src="https://odoo-community.org/logo.png" />
</a>
<p>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.</p>

View File

@ -14,15 +14,16 @@ class DynamicListCase(common.TransactionCase):
def setUpClass(cls):
super().setUpClass()
cls.tag = cls.env["res.partner.category"].create({"name": "testing tag"})
cls.partners = cls.env["res.partner"]
for number in range(5):
cls.partners |= cls.partners.create(
cls.partners = cls.env["res.partner"].create(
[
{
"name": "partner %d" % number,
"category_id": [(4, cls.tag.id, False)],
"email": "%d@example.com" % number,
}
)
for number in range(5)
]
)
cls.list = cls.env["mailing.list"].create(
{
"name": "test list",
@ -180,3 +181,17 @@ class DynamicListCase(common.TransactionCase):
wizard.action_merge()
self.assertTrue(partner_1.id in self.list.contact_ids.mapped("partner_id").ids)
self.assertTrue(partner_1.id in list2.contact_ids.mapped("partner_id").ids)
def test_synced_contacts_can_be_bounced(self):
self.list.sync_method = "full"
self.list.action_sync()
contact = self.env["mailing.contact"].search(
[
("list_ids", "in", self.list.ids),
("partner_id", "=", self.partners[0].id),
]
)
# A bounce arrives through fetchmail
contact._message_receive_bounce(contact.email, contact.partner_id)
# The contact is marked as bounced
self.assertEqual(contact.message_bounce, 1)

View File

@ -98,6 +98,10 @@ Contributors
* Nguyễn Minh Chiến <chien@trobz.com>
* `Moduon <https://www.moduon.team>`_:
* Jairo Llopis
Other credits
~~~~~~~~~~~~~

View File

@ -53,37 +53,30 @@ class MailingContact(models.Model):
)
self.country_id = self.partner_id.country_id
@api.model
def _get_contact_vals(self, origin_vals):
record = self.new(origin_vals)
if not record.partner_id:
record._set_partner()
record._onchange_partner_mass_mailing_partner()
new_vals = record._convert_to_write(record._cache)
new_vals.update(
subscription_list_ids=origin_vals.get("subscription_list_ids", []),
list_ids=origin_vals.get("list_ids", []),
)
if new_vals.get("partner_id") and "tag_ids" in new_vals:
# When there is a partner, tag_ids must get value from the compute function
# otherwise, its values will be different from partner
del new_vals["tag_ids"]
return new_vals
def _overwrite_partner(self, vals, creating=False):
"""Overwrite partner and update contact data if needed."""
self.ensure_one()
if self.env.context.get("mass_mailing_partner_writing"):
return
_self = self.with_context(mass_mailing_partner_writing=True)
prev_partner = _self.partner_id
if "partner_id" not in vals:
_self._set_partner()
if creating or prev_partner != _self.partner_id:
_self._onchange_partner_mass_mailing_partner()
@api.model_create_multi
def create(self, vals_list):
new_vals_list = []
for vals in vals_list:
new_vals = self._get_contact_vals(vals)
new_vals_list.append(new_vals)
return super().create(new_vals_list)
result = super().create(vals_list)
for contact, vals in zip(result, vals_list):
contact._overwrite_partner(vals, True)
return result
def write(self, vals):
result = super().write(vals)
for contact in self:
origin_vals = contact.copy_data(vals)[0]
new_vals = self._get_contact_vals(origin_vals)
super(MailingContact, contact).write(new_vals)
return True
contact._overwrite_partner(vals)
return result
def _get_categories(self):
ca_ids = (
@ -113,6 +106,8 @@ class MailingContact(models.Model):
email = self.email.strip()
partner = m_partner.search([("email", "=ilike", email)], limit=1)
if partner:
if partner == self.partner_id:
return
# Partner found
self.partner_id = partner
else:

View File

@ -16,3 +16,7 @@
* `Trobz <https://trobz.com>`_:
* Nguyễn Minh Chiến <chien@trobz.com>
* `Moduon <https://www.moduon.team>`_:
* Jairo Llopis

View File

@ -1,4 +1,3 @@
<?xml version="1.0" encoding="utf-8"?>
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml" xml:lang="en" lang="en">
<head>
@ -9,10 +8,11 @@
/*
:Author: David Goodger (goodger@python.org)
:Id: $Id: html4css1.css 8954 2022-01-20 10:10:25Z milde $
:Id: $Id: html4css1.css 9511 2024-01-13 09:50:07Z milde $
:Copyright: This stylesheet has been placed in the public domain.
Default cascading style sheet for the HTML output of Docutils.
Despite the name, some widely supported CSS2 features are used.
See https://docutils.sourceforge.io/docs/howto/html-stylesheets.html for how to
customize this style sheet.
@ -275,7 +275,7 @@ pre.literal-block, pre.doctest-block, pre.math, pre.code {
margin-left: 2em ;
margin-right: 2em }
pre.code .ln { color: grey; } /* line numbers */
pre.code .ln { color: gray; } /* line numbers */
pre.code, code { background-color: #eeeeee }
pre.code .comment, code .comment { color: #5C6576 }
pre.code .keyword, code .keyword { color: #3B0D06; font-weight: bold }
@ -301,7 +301,7 @@ span.option {
span.pre {
white-space: pre }
span.problematic {
span.problematic, pre.problematic {
color: red }
span.section-subtitle {
@ -447,6 +447,13 @@ If you spotted it first, help us to smash it by providing a detailed and welcome
</ul>
</blockquote>
</li>
<li><p class="first"><a class="reference external" href="https://www.moduon.team">Moduon</a>:</p>
<blockquote>
<ul class="simple">
<li>Jairo Llopis</li>
</ul>
</blockquote>
</li>
</ul>
</div>
<div class="section" id="other-credits">
@ -456,7 +463,9 @@ If you spotted it first, help us to smash it by providing a detailed and welcome
<div class="section" id="maintainers">
<h1>Maintainers</h1>
<p>This module is maintained by the OCA.</p>
<a class="reference external image-reference" href="https://odoo-community.org"><img alt="Odoo Community Association" src="https://odoo-community.org/logo.png" /></a>
<a class="reference external image-reference" href="https://odoo-community.org">
<img alt="Odoo Community Association" src="https://odoo-community.org/logo.png" />
</a>
<p>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.</p>

View File

@ -14,7 +14,7 @@ class PartnerMailListWizardCase(base.BaseCase):
wizard = self.env["partner.mail.list.wizard"].create(
{"mail_list_id": self.mailing_list.id}
)
wizard.partner_ids = [self.partner.id]
wizard.partner_ids = self.partner
wizard.add_to_mail_list()
contacts = self.env["mailing.contact"].search(
[("partner_id", "=", self.partner.id)]

View File

@ -36,8 +36,8 @@ class PartnerMailListWizard(models.TransientModel):
contact_vals = {
"partner_id": partner.id,
"list_ids": [(4, self.mail_list_id.id)],
"title_id": partner.title or False,
"title_id": partner.title.id or False,
"company_name": partner.company_id.name or False,
"country_id": partner.country_id or False,
"country_id": partner.country_id.id or False,
}
contact_obj.create(contact_vals)