odoo18/addons/test_mail/tests/test_mail_scheduled_message.py

233 lines
11 KiB
Python

# Part of Odoo. See LICENSE file for full copyright and licensing details.
from odoo.addons.base.tests.test_ir_cron import CronMixinCase
from odoo.addons.mail.tests.common import MailCommon
from odoo.addons.test_mail.models.mail_test_lead import MailTestTLead
from odoo.addons.test_mail.tests.common import TestRecipients
from odoo.exceptions import AccessError, UserError, ValidationError
from odoo.fields import Datetime as FieldDatetime
from odoo.tests import tagged, users
from odoo.tools import mute_logger
from unittest.mock import patch
@tagged('mail_scheduled_message')
class TestScheduledMessage(MailCommon, TestRecipients):
""" Test Scheduled Message internals """
@classmethod
def setUpClass(cls):
super().setUpClass()
# force 'now' to ease test about schedulers
cls.reference_now = FieldDatetime.to_datetime('2022-12-24 12:00:00')
with cls.mock_datetime_and_now(cls, cls.reference_now):
cls.test_record = cls.env['mail.test.ticket'].with_context(cls._test_context).create([{
'name': 'Test Record',
'customer_id': cls.partner_1.id,
'user_id': cls.user_employee.id,
}])
cls.hidden_scheduled_message, cls.visible_scheduled_message = cls.env['mail.scheduled.message'].create([
{
'author_id': cls.partner_admin.id,
'model': cls.partner_employee._name,
'res_id': cls.partner_employee.id,
'body': 'Hidden Scheduled Message',
'scheduled_date': '2022-12-24 15:00:00',
},
{
'author_id': cls.partner_admin.id,
'model': cls.test_record._name,
'res_id': cls.test_record.id,
'body': 'Visible Scheduled Message',
'scheduled_date': '2022-12-24 15:00:00',
},
]).with_user(cls.user_employee)
def schedule_message(self, target_record=None, author_id=None, **kwargs):
with self.mock_datetime_and_now(self.reference_now):
return self.env['mail.scheduled.message'].create({
'author_id': author_id or self.env.user.partner_id.id,
'model': target_record._name if target_record else kwargs.pop('model'),
'res_id': target_record.id if target_record else kwargs.pop('res_id'),
'body': kwargs.pop('body', 'Test Body'),
'scheduled_date': kwargs.pop('scheduled_date', '2022-12-24 15:00:00'),
**kwargs,
})
class TestScheduledMessageAccess(TestScheduledMessage):
@users('employee')
def test_scheduled_message_model_without_post_right(self):
# creation on a record that the user cannot post to
with self.assertRaises(AccessError):
self.schedule_message(self.partner_employee)
# read a message scheduled on a record the user can't post to
with self.assertRaises(AccessError):
self.hidden_scheduled_message.read()
# search a message scheduled on a record the user can't post to
self.assertFalse(self.env['mail.scheduled.message'].search([['id', '=', self.hidden_scheduled_message.id]]))
# write on a message scheduled on a record the user can't post to
with self.assertRaises(AccessError):
self.hidden_scheduled_message.write({'body': 'boum'})
# post a message scheduled on a record the user can't post to
with self.assertRaises(AccessError):
self.hidden_scheduled_message.post_message()
# unlink a message scheduled on a record the user can't post to
with self.assertRaises(AccessError):
self.hidden_scheduled_message.unlink()
@users('employee')
def test_scheduled_message_model_with_post_right(self):
# read a message scheduled by another user on a record the user can post to
self.visible_scheduled_message.read()
# search a message scheduled by another user on a record the user can post to
self.assertEqual(self.env['mail.scheduled.message'].search([['id', '=', self.visible_scheduled_message.id]]), self.visible_scheduled_message)
# write on a message scheduled by another user on a record the user can post to
with self.assertRaises(AccessError):
self.visible_scheduled_message.write({'body': 'boum'})
# post a message scheduled on a record the user can post to
with self.assertRaises(UserError):
self.visible_scheduled_message.post_message()
# unlink a message scheduled on a record the user can post to
self.visible_scheduled_message.unlink()
@users('employee')
def test_own_scheduled_message(self):
# create a scheduled message on a record the user can post to
scheduled_message = self.schedule_message(self.test_record)
# read own scheduled message
scheduled_message.read()
# search own scheduled message
self.assertEqual(self.env['mail.scheduled.message'].search([['id', '=', scheduled_message.id]]), scheduled_message)
# write on own scheduled message
scheduled_message.write({'body': 'Hello!'})
# unlink own scheduled message
scheduled_message.unlink()
class TestScheduledMessageBusiness(TestScheduledMessage, CronMixinCase):
@users('employee')
def test_scheduled_message_restrictions(self):
# cannot schedule a message in the past
with self.assertRaises(ValidationError):
self.schedule_message(self.test_record, scheduled_date='2022-12-24 10:00:00')
# cannot schedule a message on a model without thread
# with admin as employee does not have write access on res.users)
with self.with_user("admin"), self.assertRaises(ValidationError):
self.schedule_message(self.user_employee)
scheduled_message = self.schedule_message(self.test_record)
# cannot reschedule a message in the past
with self.assertRaises(ValidationError):
scheduled_message.write({'scheduled_date': '2022-12-24 14:00:00'})
# cannot change target record of scheduled message
with self.assertRaises(UserError):
scheduled_message.write({'res_id': 2})
with self.assertRaises(UserError):
scheduled_message.write({'model': 'mail.test.track'})
# unlink the test record should also unlink the test message
self.test_record.sudo().unlink()
self.assertFalse(scheduled_message.exists())
@users('employee')
def test_scheduled_message_posting(self):
schedule_cron_id = self.env.ref('mail.ir_cron_post_scheduled_message').id
test_lead = self.env["mail.test.lead"].create({})
with self.mock_mail_gateway(), \
self.mock_mail_app(), \
self.capture_triggers(schedule_cron_id) as capt:
scheduled_message_id = self.schedule_message(
self.test_record,
scheduled_date='2022-12-24 14:00:00',
partner_ids=self.test_record.customer_id,
body="success",
subject="Test subject",
).id
# cron should be triggered at scheduled date
self.assertEqual(capt.records['call_at'], FieldDatetime.to_datetime('2022-12-24 14:00:00'))
# no message created or mail sent
self.assertFalse(self.test_record.message_ids)
self.assertFalse(self._new_mails)
# add a scheduled message that will fail to check that it won't block the cron
failing_schedueld_message_id = self.schedule_message(
test_lead,
scheduled_date='2022-12-24 14:00:00',
partner_ids=self.test_record.customer_id,
body="fail",
).id
def _message_post_after_hook(self, message, values):
raise Exception("Boum!")
with self.mock_datetime_and_now('2022-12-24 14:00:00'),\
patch.object(MailTestTLead, '_message_post_after_hook', _message_post_after_hook),\
mute_logger('odoo.addons.mail.models.mail_scheduled_message'):
self.env['mail.scheduled.message'].with_user(self.user_root)._post_messages_cron()
# one scheduled message failed, only one mail should be sent
self.assertEqual(len(self._new_mails), 1)
# user should be notified about the failed posting
self.assertMailNotifications(
self._new_msgs.filtered(lambda m: not m.model),
[{
'content': f"<p>The message scheduled on {test_lead._name}({test_lead.id}) with"
" the following content could not be sent:<br>-----<br></p><p>fail</p><br>-----<br>",
'message_type': 'user_notification',
'subtype': 'mail.mt_note',
'message_values': {
'author_id': self.partner_root,
'model': False,
'res_id': False,
'subject': "A scheduled message could not be sent",
},
'notif': [
{'partner': self.partner_employee, 'type': 'inbox'}
]
}])
# other message should be posted and mail should be sent
self.assertMailNotifications(
self._new_msgs.filtered(lambda m: m.model == self.test_record._name),
[{
'content': "<p>success</p>",
'message_type': 'notification',
'message_values': {
'author_id': self.partner_employee,
'model': self.test_record._name,
'res_id': self.test_record.id,
'subject': "Test subject",
},
'notif': [
{'partner': self.test_record.customer_id, 'type': 'email'}
]
}]
)
self.assertEqual(self._new_mails[0].state, 'sent')
# scheduled messages shouldn't exist anymore
self.assertFalse(self.env['mail.scheduled.message'].search([['id', 'in', [scheduled_message_id, failing_schedueld_message_id]]]))
@users('employee')
def test_scheduled_message_posting_on_scheduled_time(self):
""" Ensure scheduled message is posted and sent at the scheduled time. """
self.test_record.message_subscribe(partner_ids=[self.partner_1.id])
self.schedule_message(
self.test_record,
scheduled_date=FieldDatetime.to_string(self.reference_now),
)
with self.mock_mail_gateway(), self.mock_datetime_and_now(self.reference_now):
# Needed to get force_send disabled due to mail_notify_force_send in the context
self.env.ref('mail.ir_cron_post_scheduled_message').with_user(self.user_admin).method_direct_trigger()
# Message is posted and mail is sent on time
self.assertEqual(len(self._new_mails), 1)
self.assertMailMailWRecord(
self.test_record,
[self.partner_1],
'sent',
author=self.env.user.partner_id,
)