|
@ -4,55 +4,51 @@ |
|
|
from contextlib import contextmanager |
|
|
from contextlib import contextmanager |
|
|
|
|
|
|
|
|
from odoo.exceptions import ValidationError |
|
|
from odoo.exceptions import ValidationError |
|
|
from odoo.tests.common import HttpCase, Form |
|
|
|
|
|
|
|
|
from odoo.tests.common import Form, HttpCase |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class ActivityCase(HttpCase): |
|
|
class ActivityCase(HttpCase): |
|
|
def setUp(self): |
|
|
def setUp(self): |
|
|
super(ActivityCase, self).setUp() |
|
|
super(ActivityCase, self).setUp() |
|
|
self.cron = self.env.ref("privacy_consent.cron_auto_consent") |
|
|
self.cron = self.env.ref("privacy_consent.cron_auto_consent") |
|
|
self.cron_mail_queue = self.env.ref( |
|
|
|
|
|
"mail.ir_cron_mail_scheduler_action") |
|
|
|
|
|
|
|
|
self.cron_mail_queue = self.env.ref("mail.ir_cron_mail_scheduler_action") |
|
|
self.sync_blacklist = self.env.ref("privacy_consent.sync_blacklist") |
|
|
self.sync_blacklist = self.env.ref("privacy_consent.sync_blacklist") |
|
|
self.mt_consent_consent_new = self.env.ref( |
|
|
self.mt_consent_consent_new = self.env.ref( |
|
|
"privacy_consent.mt_consent_consent_new") |
|
|
|
|
|
|
|
|
"privacy_consent.mt_consent_consent_new" |
|
|
|
|
|
) |
|
|
self.mt_consent_acceptance_changed = self.env.ref( |
|
|
self.mt_consent_acceptance_changed = self.env.ref( |
|
|
"privacy_consent.mt_consent_acceptance_changed") |
|
|
|
|
|
|
|
|
"privacy_consent.mt_consent_acceptance_changed" |
|
|
|
|
|
) |
|
|
self.mt_consent_state_changed = self.env.ref( |
|
|
self.mt_consent_state_changed = self.env.ref( |
|
|
"privacy_consent.mt_consent_state_changed") |
|
|
|
|
|
|
|
|
"privacy_consent.mt_consent_state_changed" |
|
|
|
|
|
) |
|
|
# Some partners to ask for consent |
|
|
# Some partners to ask for consent |
|
|
self.partners = self.env["res.partner"] |
|
|
self.partners = self.env["res.partner"] |
|
|
self.partners += self.partners.create({ |
|
|
|
|
|
"name": "consent-partner-0", |
|
|
|
|
|
"email": "partner0@example.com", |
|
|
|
|
|
}) |
|
|
|
|
|
self.partners += self.partners.create({ |
|
|
|
|
|
"name": "consent-partner-1", |
|
|
|
|
|
"email": "partner1@example.com", |
|
|
|
|
|
}) |
|
|
|
|
|
self.partners += self.partners.create({ |
|
|
|
|
|
"name": "consent-partner-2", |
|
|
|
|
|
"email": "partner2@example.com", |
|
|
|
|
|
}) |
|
|
|
|
|
|
|
|
self.partners += self.partners.create( |
|
|
|
|
|
{"name": "consent-partner-0", "email": "partner0@example.com"} |
|
|
|
|
|
) |
|
|
|
|
|
self.partners += self.partners.create( |
|
|
|
|
|
{"name": "consent-partner-1", "email": "partner1@example.com"} |
|
|
|
|
|
) |
|
|
|
|
|
self.partners += self.partners.create( |
|
|
|
|
|
{"name": "consent-partner-2", "email": "partner2@example.com"} |
|
|
|
|
|
) |
|
|
# Partner without email, on purpose |
|
|
# Partner without email, on purpose |
|
|
self.partners += self.partners.create({ |
|
|
|
|
|
"name": "consent-partner-3", |
|
|
|
|
|
}) |
|
|
|
|
|
|
|
|
self.partners += self.partners.create({"name": "consent-partner-3"}) |
|
|
# Partner with wrong email, on purpose |
|
|
# Partner with wrong email, on purpose |
|
|
self.partners += self.partners.create({ |
|
|
|
|
|
"name": "consent-partner-4", |
|
|
|
|
|
"email": "wrong-mail", |
|
|
|
|
|
}) |
|
|
|
|
|
|
|
|
self.partners += self.partners.create( |
|
|
|
|
|
{"name": "consent-partner-4", "email": "wrong-mail"} |
|
|
|
|
|
) |
|
|
# Blacklist some partners |
|
|
# Blacklist some partners |
|
|
self.blacklists = self.env["mail.blacklist"] |
|
|
self.blacklists = self.env["mail.blacklist"] |
|
|
self.blacklists += self.blacklists._add("partner1@example.com") |
|
|
self.blacklists += self.blacklists._add("partner1@example.com") |
|
|
# Activity without consent |
|
|
# Activity without consent |
|
|
self.activity_noconsent = self.env["privacy.activity"].create({ |
|
|
|
|
|
"name": "activity_noconsent", |
|
|
|
|
|
"description": "I'm activity 1", |
|
|
|
|
|
}) |
|
|
|
|
|
|
|
|
self.activity_noconsent = self.env["privacy.activity"].create( |
|
|
|
|
|
{"name": "activity_noconsent", "description": "I'm activity 1"} |
|
|
|
|
|
) |
|
|
# Activity with auto consent, for all partners |
|
|
# Activity with auto consent, for all partners |
|
|
self.activity_auto = self.env["privacy.activity"].create({ |
|
|
|
|
|
|
|
|
self.activity_auto = self.env["privacy.activity"].create( |
|
|
|
|
|
{ |
|
|
"name": "activity_auto", |
|
|
"name": "activity_auto", |
|
|
"description": "I'm activity auto", |
|
|
"description": "I'm activity auto", |
|
|
"subject_find": True, |
|
|
"subject_find": True, |
|
@ -60,9 +56,11 @@ class ActivityCase(HttpCase): |
|
|
"consent_required": "auto", |
|
|
"consent_required": "auto", |
|
|
"default_consent": True, |
|
|
"default_consent": True, |
|
|
"server_action_id": self.sync_blacklist.id, |
|
|
"server_action_id": self.sync_blacklist.id, |
|
|
}) |
|
|
|
|
|
|
|
|
} |
|
|
|
|
|
) |
|
|
# Activity with manual consent, skipping partner 0 |
|
|
# Activity with manual consent, skipping partner 0 |
|
|
self.activity_manual = self.env["privacy.activity"].create({ |
|
|
|
|
|
|
|
|
self.activity_manual = self.env["privacy.activity"].create( |
|
|
|
|
|
{ |
|
|
"name": "activity_manual", |
|
|
"name": "activity_manual", |
|
|
"description": "I'm activity 3", |
|
|
"description": "I'm activity 3", |
|
|
"subject_find": True, |
|
|
"subject_find": True, |
|
@ -70,44 +68,31 @@ class ActivityCase(HttpCase): |
|
|
"consent_required": "manual", |
|
|
"consent_required": "manual", |
|
|
"default_consent": False, |
|
|
"default_consent": False, |
|
|
"server_action_id": self.sync_blacklist.id, |
|
|
"server_action_id": self.sync_blacklist.id, |
|
|
}) |
|
|
|
|
|
|
|
|
} |
|
|
|
|
|
) |
|
|
|
|
|
|
|
|
@contextmanager |
|
|
@contextmanager |
|
|
def _patch_build(self): |
|
|
def _patch_build(self): |
|
|
self._built_messages = [] |
|
|
self._built_messages = [] |
|
|
IMS = self.env['ir.mail_server'] |
|
|
|
|
|
|
|
|
IMS = self.env["ir.mail_server"] |
|
|
|
|
|
|
|
|
def _build_email( |
|
|
|
|
|
_self, |
|
|
|
|
|
email_from, |
|
|
|
|
|
email_to, |
|
|
|
|
|
subject, |
|
|
|
|
|
body, |
|
|
|
|
|
*args, |
|
|
|
|
|
**kwargs |
|
|
|
|
|
): |
|
|
|
|
|
|
|
|
def _build_email(_self, email_from, email_to, subject, body, *args, **kwargs): |
|
|
self._built_messages.append(body) |
|
|
self._built_messages.append(body) |
|
|
return _build_email.origin( |
|
|
return _build_email.origin( |
|
|
_self, |
|
|
|
|
|
email_from, |
|
|
|
|
|
email_to, |
|
|
|
|
|
subject, |
|
|
|
|
|
body, |
|
|
|
|
|
*args, |
|
|
|
|
|
**kwargs, |
|
|
|
|
|
|
|
|
_self, email_from, email_to, subject, body, *args, **kwargs, |
|
|
) |
|
|
) |
|
|
|
|
|
|
|
|
try: |
|
|
try: |
|
|
IMS._patch_method('build_email', _build_email) |
|
|
|
|
|
|
|
|
IMS._patch_method("build_email", _build_email) |
|
|
yield |
|
|
yield |
|
|
finally: |
|
|
finally: |
|
|
IMS._revert_method('build_email') |
|
|
|
|
|
|
|
|
IMS._revert_method("build_email") |
|
|
|
|
|
|
|
|
def check_activity_auto_properly_sent(self): |
|
|
def check_activity_auto_properly_sent(self): |
|
|
"""Check emails sent by ``self.activity_auto``.""" |
|
|
"""Check emails sent by ``self.activity_auto``.""" |
|
|
consents = self.env["privacy.consent"].search([ |
|
|
|
|
|
("activity_id", "=", self.activity_auto.id), |
|
|
|
|
|
]) |
|
|
|
|
|
|
|
|
consents = self.env["privacy.consent"].search( |
|
|
|
|
|
[("activity_id", "=", self.activity_auto.id)] |
|
|
|
|
|
) |
|
|
# Check pending mails |
|
|
# Check pending mails |
|
|
for consent in consents: |
|
|
for consent in consents: |
|
|
self.assertEqual(consent.state, "draft") |
|
|
self.assertEqual(consent.state, "draft") |
|
@ -116,40 +101,33 @@ class ActivityCase(HttpCase): |
|
|
# Check sent mails |
|
|
# Check sent mails |
|
|
with self._patch_build(): |
|
|
with self._patch_build(): |
|
|
self.cron_mail_queue.method_direct_trigger() |
|
|
self.cron_mail_queue.method_direct_trigger() |
|
|
for index, consent in enumerate(consents): |
|
|
|
|
|
|
|
|
for consent in consents: |
|
|
good_email = "@" in (consent.partner_id.email or "") |
|
|
good_email = "@" in (consent.partner_id.email or "") |
|
|
expected_messages = 3 if good_email else 2 |
|
|
expected_messages = 3 if good_email else 2 |
|
|
self.assertEqual( |
|
|
self.assertEqual( |
|
|
consent.state, |
|
|
|
|
|
"sent" if good_email else "draft", |
|
|
|
|
|
|
|
|
consent.state, "sent" if good_email else "draft", |
|
|
) |
|
|
) |
|
|
messages = consent.message_ids |
|
|
messages = consent.message_ids |
|
|
self.assertEqual(len(messages), expected_messages) |
|
|
self.assertEqual(len(messages), expected_messages) |
|
|
# 2nd message notifies creation |
|
|
# 2nd message notifies creation |
|
|
self.assertEqual( |
|
|
self.assertEqual( |
|
|
messages[expected_messages - 1].subtype_id, |
|
|
|
|
|
self.mt_consent_consent_new, |
|
|
|
|
|
|
|
|
messages[expected_messages - 1].subtype_id, self.mt_consent_consent_new, |
|
|
) |
|
|
) |
|
|
# 3rd message notifies subject |
|
|
# 3rd message notifies subject |
|
|
# Placeholder links should be logged |
|
|
# Placeholder links should be logged |
|
|
self.assertIn( |
|
|
self.assertIn( |
|
|
"/privacy/consent/accept/", |
|
|
|
|
|
messages[expected_messages - 2].body) |
|
|
|
|
|
|
|
|
"/privacy/consent/accept/", messages[expected_messages - 2].body |
|
|
|
|
|
) |
|
|
self.assertIn( |
|
|
self.assertIn( |
|
|
"/privacy/consent/reject/", |
|
|
|
|
|
messages[expected_messages - 2].body) |
|
|
|
|
|
|
|
|
"/privacy/consent/reject/", messages[expected_messages - 2].body |
|
|
|
|
|
) |
|
|
# Tokenized links shouldn't be logged |
|
|
# Tokenized links shouldn't be logged |
|
|
self.assertNotIn( |
|
|
|
|
|
consent._url(True), |
|
|
|
|
|
messages[expected_messages - 2].body) |
|
|
|
|
|
self.assertNotIn( |
|
|
|
|
|
consent._url(False), |
|
|
|
|
|
messages[expected_messages - 2].body) |
|
|
|
|
|
|
|
|
self.assertNotIn(consent._url(True), messages[expected_messages - 2].body) |
|
|
|
|
|
self.assertNotIn(consent._url(False), messages[expected_messages - 2].body) |
|
|
# 4th message contains the state change |
|
|
# 4th message contains the state change |
|
|
if good_email: |
|
|
if good_email: |
|
|
self.assertEqual( |
|
|
self.assertEqual( |
|
|
messages[0].subtype_id, |
|
|
|
|
|
self.mt_consent_state_changed, |
|
|
|
|
|
|
|
|
messages[0].subtype_id, self.mt_consent_state_changed, |
|
|
) |
|
|
) |
|
|
# Partner's is_blacklisted should be synced with default consent |
|
|
# Partner's is_blacklisted should be synced with default consent |
|
|
self.assertFalse(consent.partner_id.is_blacklisted) |
|
|
self.assertFalse(consent.partner_id.is_blacklisted) |
|
@ -166,23 +144,21 @@ class ActivityCase(HttpCase): |
|
|
"""We have a good mail template by default.""" |
|
|
"""We have a good mail template by default.""" |
|
|
good = self.env.ref("privacy_consent.template_consent") |
|
|
good = self.env.ref("privacy_consent.template_consent") |
|
|
self.assertEqual( |
|
|
self.assertEqual( |
|
|
self.activity_noconsent.consent_template_id, |
|
|
|
|
|
good, |
|
|
|
|
|
|
|
|
self.activity_noconsent.consent_template_id, good, |
|
|
) |
|
|
) |
|
|
self.assertEqual( |
|
|
self.assertEqual( |
|
|
self.activity_noconsent.consent_template_default_body_html, |
|
|
|
|
|
good.body_html, |
|
|
|
|
|
|
|
|
self.activity_noconsent.consent_template_default_body_html, good.body_html, |
|
|
) |
|
|
) |
|
|
self.assertEqual( |
|
|
self.assertEqual( |
|
|
self.activity_noconsent.consent_template_default_subject, |
|
|
|
|
|
good.subject, |
|
|
|
|
|
|
|
|
self.activity_noconsent.consent_template_default_subject, good.subject, |
|
|
) |
|
|
) |
|
|
|
|
|
|
|
|
def test_find_subject_if_consent_required(self): |
|
|
def test_find_subject_if_consent_required(self): |
|
|
"""If user wants to require consent, it needs subjects.""" |
|
|
"""If user wants to require consent, it needs subjects.""" |
|
|
# Test the onchange helper |
|
|
# Test the onchange helper |
|
|
onchange_activity1 = self.env["privacy.activity"].new( |
|
|
onchange_activity1 = self.env["privacy.activity"].new( |
|
|
self.activity_noconsent.copy_data()[0]) |
|
|
|
|
|
|
|
|
self.activity_noconsent.copy_data()[0] |
|
|
|
|
|
) |
|
|
self.assertFalse(onchange_activity1.subject_find) |
|
|
self.assertFalse(onchange_activity1.subject_find) |
|
|
onchange_activity1.consent_required = "auto" |
|
|
onchange_activity1.consent_required = "auto" |
|
|
onchange_activity1._onchange_consent_required_subject_find() |
|
|
onchange_activity1._onchange_consent_required_subject_find() |
|
@ -209,8 +185,7 @@ class ActivityCase(HttpCase): |
|
|
consents = self.env[result["res_model"]].search(result["domain"]) |
|
|
consents = self.env[result["res_model"]].search(result["domain"]) |
|
|
self.assertEqual(consents.mapped("state"), ["draft"] * 3) |
|
|
self.assertEqual(consents.mapped("state"), ["draft"] * 3) |
|
|
self.assertEqual( |
|
|
self.assertEqual( |
|
|
consents.mapped("partner_id.is_blacklisted"), |
|
|
|
|
|
[False] * 3, |
|
|
|
|
|
|
|
|
consents.mapped("partner_id.is_blacklisted"), [False] * 3, |
|
|
) |
|
|
) |
|
|
self.assertEqual(consents.mapped("accepted"), [False] * 3) |
|
|
self.assertEqual(consents.mapped("accepted"), [False] * 3) |
|
|
self.assertEqual(consents.mapped("last_metadata"), [False] * 3) |
|
|
self.assertEqual(consents.mapped("last_metadata"), [False] * 3) |
|
@ -243,8 +218,7 @@ class ActivityCase(HttpCase): |
|
|
self.assertEqual(messages[0].subtype_id, self.mt_consent_state_changed) |
|
|
self.assertEqual(messages[0].subtype_id, self.mt_consent_state_changed) |
|
|
self.assertEqual(consents.mapped("state"), ["sent", "draft", "draft"]) |
|
|
self.assertEqual(consents.mapped("state"), ["sent", "draft", "draft"]) |
|
|
self.assertEqual( |
|
|
self.assertEqual( |
|
|
consents.mapped("partner_id.is_blacklisted"), |
|
|
|
|
|
[True, False, False], |
|
|
|
|
|
|
|
|
consents.mapped("partner_id.is_blacklisted"), [True, False, False], |
|
|
) |
|
|
) |
|
|
# Placeholder links should be logged |
|
|
# Placeholder links should be logged |
|
|
self.assertTrue("/privacy/consent/accept/" in messages[1].body) |
|
|
self.assertTrue("/privacy/consent/accept/" in messages[1].body) |
|
@ -264,11 +238,9 @@ class ActivityCase(HttpCase): |
|
|
self.assertEqual(consents.mapped("accepted"), [True, False, False]) |
|
|
self.assertEqual(consents.mapped("accepted"), [True, False, False]) |
|
|
self.assertTrue(consents[0].last_metadata) |
|
|
self.assertTrue(consents[0].last_metadata) |
|
|
self.assertFalse(consents[0].partner_id.is_blacklisted) |
|
|
self.assertFalse(consents[0].partner_id.is_blacklisted) |
|
|
|
|
|
self.assertEqual(consents.mapped("state"), ["answered", "draft", "draft"]) |
|
|
self.assertEqual( |
|
|
self.assertEqual( |
|
|
consents.mapped("state"), ["answered", "draft", "draft"]) |
|
|
|
|
|
self.assertEqual( |
|
|
|
|
|
consents[0].message_ids[0].subtype_id, |
|
|
|
|
|
self.mt_consent_acceptance_changed, |
|
|
|
|
|
|
|
|
consents[0].message_ids[0].subtype_id, self.mt_consent_acceptance_changed, |
|
|
) |
|
|
) |
|
|
# Visit tokenized reject URL |
|
|
# Visit tokenized reject URL |
|
|
result = self.url_open(reject_url).text |
|
|
result = self.url_open(reject_url).text |
|
@ -280,11 +252,9 @@ class ActivityCase(HttpCase): |
|
|
self.assertEqual(consents.mapped("accepted"), [False, False, False]) |
|
|
self.assertEqual(consents.mapped("accepted"), [False, False, False]) |
|
|
self.assertTrue(consents[0].last_metadata) |
|
|
self.assertTrue(consents[0].last_metadata) |
|
|
self.assertTrue(consents[0].partner_id.is_blacklisted) |
|
|
self.assertTrue(consents[0].partner_id.is_blacklisted) |
|
|
|
|
|
self.assertEqual(consents.mapped("state"), ["answered", "draft", "draft"]) |
|
|
self.assertEqual( |
|
|
self.assertEqual( |
|
|
consents.mapped("state"), ["answered", "draft", "draft"]) |
|
|
|
|
|
self.assertEqual( |
|
|
|
|
|
consents[0].message_ids[0].subtype_id, |
|
|
|
|
|
self.mt_consent_acceptance_changed, |
|
|
|
|
|
|
|
|
consents[0].message_ids[0].subtype_id, self.mt_consent_acceptance_changed, |
|
|
) |
|
|
) |
|
|
self.assertFalse(consents[1].last_metadata) |
|
|
self.assertFalse(consents[1].last_metadata) |
|
|
|
|
|
|
|
|