# -*- coding: utf-8 -*- # Part of Odoo. See LICENSE file for full copyright and licensing details. from datetime import datetime, timedelta from freezegun import freeze_time from odoo import Command from odoo.addons.event.tests.common import EventCase from odoo import exceptions from odoo.fields import Datetime as FieldsDatetime from odoo.tests.common import users, Form, tagged from odoo.tools import mute_logger class TestEventInternalsCommon(EventCase): @classmethod def setUpClass(cls): super(TestEventInternalsCommon, cls).setUpClass() cls.event_type_complex = cls.env['event.type'].create({ 'name': 'Update Type', 'has_seats_limitation': True, 'seats_max': 30, 'default_timezone': 'Europe/Paris', 'event_type_ticket_ids': [ (0, 0, {'name': 'First Ticket',}), (0, 0, {'name': 'Second Ticket',}), ], 'event_type_mail_ids': [ (0, 0, { # right at subscription 'interval_unit': 'now', 'interval_type': 'after_sub', 'template_ref': 'mail.template,%i' % cls.env['ir.model.data']._xmlid_to_res_id('event.event_subscription')}), (0, 0, { # 1 days before event 'interval_nbr': 1, 'interval_unit': 'days', 'interval_type': 'before_event', 'template_ref': 'mail.template,%i' % cls.env['ir.model.data']._xmlid_to_res_id('event.event_reminder')}), ], }) # Mock dates to have reproducible computed fields based on time cls.reference_now = datetime(2020, 1, 31, 10, 0, 0) cls.reference_beg = datetime(2020, 2, 1, 8, 30, 0) cls.reference_end = datetime(2020, 2, 4, 18, 45, 0) cls.event_0 = cls.env['event.event'].create({ 'date_begin': cls.reference_beg, 'date_end': cls.reference_end, 'date_tz': 'Europe/Brussels', 'name': 'TestEvent', }) @tagged('event_event') class TestEventData(TestEventInternalsCommon): @users('user_eventmanager') def test_event_date_computation(self): event = self.event_0.with_user(self.env.user) with freeze_time(self.reference_now): event.write({ 'registration_ids': [(0, 0, {'partner_id': self.event_customer.id, 'name': 'test_reg'})], 'date_begin': datetime(2020, 1, 31, 15, 0, 0), 'date_end': datetime(2020, 4, 5, 18, 0, 0), }) registration = event.registration_ids[0] self.assertEqual(registration.get_date_range_str(), u'today') event.date_begin = datetime(2020, 2, 1, 15, 0, 0) self.assertEqual(registration.get_date_range_str(), u'tomorrow') event.date_begin = datetime(2020, 2, 2, 6, 0, 0) self.assertEqual(registration.get_date_range_str(), u'in 2 days') event.date_begin = datetime(2020, 2, 20, 17, 0, 0) self.assertEqual(registration.get_date_range_str(), u'next month') event.date_begin = datetime(2020, 3, 1, 10, 0, 0) self.assertEqual(registration.get_date_range_str(), u'on Mar 1, 2020') # Is actually 8:30 to 20:00 in Mexico event.write({ 'date_begin': datetime(2020, 1, 31, 14, 30, 0), 'date_end': datetime(2020, 2, 1, 2, 0, 0), 'date_tz': 'Mexico/General' }) self.assertTrue(event.is_one_day) @freeze_time('2020-1-31 10:00:00') @users('user_eventmanager') def test_event_date_timezone(self): event = self.event_0.with_user(self.env.user) # Is actually 8:30 to 20:00 in Mexico event.write({ 'date_begin': datetime(2020, 1, 31, 14, 30, 0), 'date_end': datetime(2020, 2, 1, 2, 0, 0), 'date_tz': 'Mexico/General' }) self.assertTrue(event.is_one_day) self.assertFalse(event.is_ongoing) @users('user_eventmanager') @mute_logger('odoo.models.unlink') def test_event_configuration_from_type(self): """ Test data computation of event coming from its event.type template. """ self.assertEqual(self.env.user.tz, 'Europe/Brussels') # ------------------------------------------------------------ # STARTING DATA # ------------------------------------------------------------ event_type = self.env['event.type'].browse(self.event_type_complex.id) event = self.env['event.event'].create({ 'name': 'Event Update Type', 'date_begin': FieldsDatetime.to_string(datetime.today() + timedelta(days=1)), 'date_end': FieldsDatetime.to_string(datetime.today() + timedelta(days=15)), 'event_mail_ids': False, }) self.assertEqual(event.date_tz, self.env.user.tz) self.assertFalse(event.seats_limited) self.assertEqual(event.event_mail_ids, self.env['event.mail']) self.assertEqual(event.event_ticket_ids, self.env['event.event.ticket']) registration = self._create_registrations(event, 1) # ------------------------------------------------------------ # FILL SYNC TEST # ------------------------------------------------------------ # change template to a one with mails -> fill event as it is void event_type.write({ 'event_type_mail_ids': [(5, 0), (0, 0, { 'interval_nbr': 1, 'interval_unit': 'days', 'interval_type': 'before_event', 'template_ref': 'mail.template,%i' % self.env['ir.model.data']._xmlid_to_res_id('event.event_reminder')}) ], 'event_type_ticket_ids': [(5, 0), (0, 0, {'name': 'TestRegistration'})], }) event.write({'event_type_id': event_type.id}) self.assertEqual(event.date_tz, 'Europe/Paris') self.assertTrue(event.seats_limited) self.assertEqual(event.seats_max, event_type.seats_max) # check 2many fields being populated self.assertEqual(len(event.event_mail_ids), 1) self.assertEqual(event.event_mail_ids.interval_nbr, 1) self.assertEqual(event.event_mail_ids.interval_unit, 'days') self.assertEqual(event.event_mail_ids.interval_type, 'before_event') self.assertEqual(event.event_mail_ids.template_ref, self.env.ref('event.event_reminder')) self.assertEqual(len(event.event_ticket_ids), 1) # update template, unlink from event -> should not impact event event_type.write({'has_seats_limitation': False}) self.assertEqual(event_type.seats_max, 0) self.assertTrue(event.seats_limited) self.assertEqual(event.seats_max, 30) # original template value event.write({'event_type_id': False}) self.assertEqual(event.event_type_id, self.env["event.type"]) # set template back -> update event event.write({'event_type_id': event_type.id}) self.assertFalse(event.seats_limited) self.assertEqual(event.seats_max, 0) self.assertEqual(len(event.event_ticket_ids), 1) event_ticket1 = event.event_ticket_ids[0] self.assertEqual(event_ticket1.name, 'TestRegistration') @users('user_eventmanager') def test_event_configuration_mails_from_type(self): """ Test data computation (related to mails) of event coming from its event.type template. This test uses pretty low level Form data checks, as manipulations in a non-saved Form are required to highlight an undesired behavior when switching event_type templates : event_mail_ids not linked to a registration were generated and kept when switching between different templates in the Form, which could rapidly lead to a substantial amount of undesired lines. """ # setup test records event_type_default = self.env['event.type'].create({ 'name': 'Type Default', 'event_type_mail_ids': False, }) event_type_mails = self.env['event.type'].create({ 'name': 'Type Mails', 'event_type_mail_ids': [ Command.clear(), Command.create({ 'notification_type': 'mail', 'interval_nbr': 77, 'interval_unit': 'days', 'interval_type': 'after_event', 'template_ref': 'mail.template,%i' % self.env['ir.model.data']._xmlid_to_res_id('event.event_reminder'), }) ], }) event = self.env['event.event'].create({ 'name': 'Event', 'date_begin': FieldsDatetime.to_string(datetime.today() + timedelta(days=1)), 'date_end': FieldsDatetime.to_string(datetime.today() + timedelta(days=15)), 'event_type_id': event_type_default.id }) event.write({ 'event_mail_ids': [ Command.clear(), Command.create({ 'notification_type': 'mail', 'interval_unit': 'now', 'interval_type': 'after_sub', 'template_ref': 'mail.template,%i' % self.env['ir.model.data']._xmlid_to_res_id('event.event_subscription'), }) ] }) mail = event.event_mail_ids[0] registration = self._create_registrations(event, 1) self.assertEqual(registration.state, 'open') # verify that mail is linked to the registration self.assertEqual( set(mail.mapped('mail_registration_ids.registration_id.id')), set([registration.id]) ) # start test scenario event_form = Form(event) # verify that mail is linked to the event in the form self.assertEqual( set(map(lambda m: m.get('id', None), event_form.event_mail_ids._records)), set([mail.id]) ) # switch to an event_type with a mail template which should be computed event_form.event_type_id = event_type_mails # verify that 2 mails were computed self.assertEqual(len(event_form.event_mail_ids._records), 2) # verify that the mail linked to the registration was kept self.assertTrue(filter(lambda m: m.get('id', None) == mail.id, event_form.event_mail_ids._records)) # since the other computed event.mail is to be created from an event.type.mail template, # verify that its attributes are the correct ones computed_mail = next(filter(lambda m: m.get('id', None) != mail.id, event_form.event_mail_ids._records), {}) self.assertEqual(computed_mail.get('interval_nbr', None), 77) self.assertEqual(computed_mail.get('interval_unit', None), 'days') self.assertEqual(computed_mail.get('interval_type', None), 'after_event') # switch back to an event type without a mail template event_form.event_type_id = event_type_default # verify that the mail linked to the registration was kept, and the other removed self.assertEqual( set(map(lambda m: m.get('id', None), event_form.event_mail_ids._records)), set([mail.id]) ) @users('user_eventmanager') def test_event_configuration_note_from_type(self): event_type = self.env['event.type'].browse(self.event_type_complex.id) event = self.env['event.event'].create({ 'name': 'Event Update Type Note', 'date_begin': FieldsDatetime.to_string(datetime.today() + timedelta(days=1)), 'date_end': FieldsDatetime.to_string(datetime.today() + timedelta(days=15)), }) # verify that note is not propagated if the event type contains blank html event.write({'note': '
Event Note
'}) event_type.write({'note': 'Event Note
') # verify that note is correctly propagated if it contains non empty html event.write({'event_type_id': False}) event_type.write({'note': 'Event Type Note
'}) event.write({'event_type_id': event_type.id}) self.assertEqual(event.note, 'Event Type Note
') @users('user_eventmanager') def test_event_configuration_tickets_from_type(self): """ Test data computation (related to tickets) of event coming from its event.type template. This test uses pretty low level Form data checks, as manipulations in a non-saved Form are required to highlight an undesired behavior when switching event_type templates : event_ticket_ids not linked to a registration were generated and kept when switching between different templates in the Form, which could rapidly lead to a substantial amount of undesired lines. """ # setup test records event_type_default = self.env['event.type'].create({ 'name': 'Type Default', }) event_type_tickets = self.env['event.type'].create({ 'name': 'Type Tickets', }) event_type_tickets.write({ 'event_type_ticket_ids': [ Command.clear(), Command.create({ 'name': 'Default Ticket', 'seats_max': 10, }) ] }) event = self.env['event.event'].create({ 'name': 'Event', 'date_begin': FieldsDatetime.to_string(datetime.today() + timedelta(days=1)), 'date_end': FieldsDatetime.to_string(datetime.today() + timedelta(days=15)), 'event_type_id': event_type_default.id }) event.write({ 'event_ticket_ids': [ Command.clear(), Command.create({ 'name': 'Registration Ticket', 'seats_max': 10, }) ] }) ticket = event.event_ticket_ids[0] registration = self._create_registrations(event, 1) # link the ticket to the registration registration.write({'event_ticket_id': ticket.id}) # start test scenario event_form = Form(event) # verify that the ticket is linked to the event in the form self.assertEqual( set(map(lambda m: m.get('name', None), event_form.event_ticket_ids._records)), set(['Registration Ticket']) ) # switch to an event_type with a ticket template which should be computed event_form.event_type_id = event_type_tickets # verify that both tickets are computed self.assertEqual( set(map(lambda m: m.get('name', None), event_form.event_ticket_ids._records)), set(['Registration Ticket', 'Default Ticket']) ) # switch back to an event_type without default tickets event_form.event_type_id = event_type_default # verify that the ticket linked to the registration was kept, and the other removed self.assertEqual( set(map(lambda m: m.get('name', None), event_form.event_ticket_ids._records)), set(['Registration Ticket']) ) @users('user_eventmanager') def test_event_mail_default_config(self): event = self.env['event.event'].create({ 'name': 'Event Update Type', 'date_begin': FieldsDatetime.to_string(datetime.today() + timedelta(days=1)), 'date_end': FieldsDatetime.to_string(datetime.today() + timedelta(days=15)), }) self.assertEqual(event.date_tz, self.env.user.tz) self.assertFalse(event.seats_limited) #Event Communications: when no event type, default configuration self.assertEqual(len(event.event_mail_ids), 3) self.assertEqual(event.event_mail_ids[0].interval_unit, 'now') self.assertEqual(event.event_mail_ids[0].interval_type, 'after_sub') self.assertEqual(event.event_mail_ids[0].template_ref, self.env.ref('event.event_subscription')) self.assertEqual(event.event_mail_ids[1].interval_nbr, 1) self.assertEqual(event.event_mail_ids[1].interval_unit, 'hours') self.assertEqual(event.event_mail_ids[1].interval_type, 'before_event') self.assertEqual(event.event_mail_ids[1].template_ref, self.env.ref('event.event_reminder')) self.assertEqual(event.event_mail_ids[2].interval_nbr, 3) self.assertEqual(event.event_mail_ids[2].interval_unit, 'days') self.assertEqual(event.event_mail_ids[2].interval_type, 'before_event') self.assertEqual(event.event_mail_ids[2].template_ref, self.env.ref('event.event_reminder')) event.write({ 'event_mail_ids': False }) self.assertEqual(event.event_mail_ids, self.env['event.mail']) def test_event_mail_filter_template_on_event(self): """Test that the mail template are filtered to show only those which are related to the event registration model. This is important to be able to show only relevant mail templates on the related field "template_ref". """ self.env['mail.template'].search([('model', '=', 'event.registration')]).unlink() self.env['mail.template'].create({'model_id': self.env['ir.model']._get('event.registration').id, 'name': 'test template'}) self.env['mail.template'].create({'model_id': self.env['ir.model']._get('res.partner').id, 'name': 'test template'}) templates = self.env['mail.template'].with_context(filter_template_on_event=True).name_search('test template') self.assertEqual(len(templates), 1, 'Should return only mail templates related to the event registration model') @freeze_time('2020-1-31 10:00:00') @users('user_eventmanager') def test_event_registrable(self): """Test if `_compute_event_registrations_open` works properly.""" event = self.event_0.with_user(self.env.user) event.write({ 'date_begin': datetime(2020, 1, 30, 8, 0, 0), 'date_end': datetime(2020, 1, 31, 8, 0, 0), }) self.assertFalse(event.event_registrations_open) event.write({ 'date_end': datetime(2020, 2, 4, 8, 0, 0), }) self.assertTrue(event.event_registrations_open) # ticket without dates boundaries -> ok ticket = self.env['event.event.ticket'].create({ 'name': 'TestTicket', 'event_id': event.id, }) self.assertTrue(event.event_registrations_open) # even with valid tickets, date limits registrations event.write({ 'date_begin': datetime(2020, 1, 28, 15, 0, 0), 'date_end': datetime(2020, 1, 30, 15, 0, 0), }) self.assertFalse(event.event_registrations_open) # no more seats available registration = self.env['event.registration'].create({ 'name': 'Albert Test', 'event_id': event.id, }) event.write({ 'date_end': datetime(2020, 2, 1, 15, 0, 0), 'seats_max': 1, 'seats_limited': True, }) self.assertEqual(event.seats_available, 0) self.assertFalse(event.event_registrations_open) # seats available are back registration.unlink() self.assertEqual(event.seats_available, 1) self.assertTrue(event.event_registrations_open) # but tickets are expired ticket.write({'end_sale_datetime': datetime(2020, 1, 30, 15, 0, 0)}) self.assertTrue(ticket.is_expired) self.assertFalse(event.event_registrations_open) @freeze_time('2020-1-31 10:00:00') @users('user_eventmanager') def test_event_ongoing(self): event_1 = self.env['event.event'].create({ 'name': 'Test Event 1', 'date_begin': datetime(2020, 1, 25, 8, 0, 0), 'date_end': datetime(2020, 2, 1, 18, 0, 0), }) self.assertTrue(event_1.is_ongoing) ongoing_events = self.env['event.event'].search([('is_ongoing', '=', True)]) self.assertIn(event_1, ongoing_events) event_1.update({'date_begin': datetime(2020, 2, 1, 9, 0, 0)}) self.assertFalse(event_1.is_ongoing) ongoing_events = self.env['event.event'].search([('is_ongoing', '=', True)]) self.assertNotIn(event_1, ongoing_events) event_2 = self.env['event.event'].create({ 'name': 'Test Event 2', 'date_begin': datetime(2020, 1, 25, 8, 0, 0), 'date_end': datetime(2020, 1, 28, 8, 0, 0), }) self.assertFalse(event_2.is_ongoing) finished_or_upcoming_events = self.env['event.event'].search([('is_ongoing', '=', False)]) self.assertIn(event_2, finished_or_upcoming_events) event_2.update({'date_end': datetime(2020, 2, 2, 8, 0, 1)}) self.assertTrue(event_2.is_ongoing) finished_or_upcoming_events = self.env['event.event'].search([('is_ongoing', '=', False)]) self.assertNotIn(event_2, finished_or_upcoming_events) @users('user_eventmanager') def test_event_seats(self): event_type = self.event_type_complex.with_user(self.env.user) event = self.env['event.event'].create({ 'name': 'Event Update Type', 'event_type_id': event_type.id, 'date_begin': FieldsDatetime.to_string(datetime.today() + timedelta(days=1)), 'date_end': FieldsDatetime.to_string(datetime.today() + timedelta(days=15)), }) self.assertEqual(event.address_id, self.env.user.company_id.partner_id) # seats: coming from event type configuration self.assertTrue(event.seats_limited) self.assertEqual(event.seats_available, event.event_type_id.seats_max) self.assertEqual(event.seats_reserved, 0) self.assertEqual(event.seats_used, 0) self.assertEqual(event.seats_taken, 0) # create registration in order to check the seats computation reg_open_multiple = self.env['event.registration'].create([{ 'event_id': event.id, 'name': 'reg_open', } for _ in range(5)]) self.assertEqual(set(reg_open_multiple.mapped('state')), {'open'}) reg_open = reg_open_multiple[0] reg_draft = self.env['event.registration'].create({ 'event_id': event.id, 'name': 'reg_draft', }) reg_draft.write({'state': 'draft'}) reg_done = self.env['event.registration'].create({ 'event_id': event.id, 'name': 'reg_done', }) reg_done.write({'state': 'done'}) self.assertEqual(event.seats_available, event.event_type_id.seats_max - 6) self.assertEqual(event.seats_reserved, 5) self.assertEqual(event.seats_used, 1) self.assertEqual(event.seats_taken, 6) # ------------------------------------------------------------ # SEATS AVAILABILITY AND (UN-)ARCHIVING REGISTRATIONS # ------------------------------------------------------------ # Archiving and seats availability reg_open.action_archive() self.assertEqual(event.seats_reserved, 4) self.assertEqual(event.seats_available, event.event_type_id.seats_max - 5) self.assertEqual(event.seats_taken, 5) reg_draft.action_archive() self.assertEqual(event.seats_available, event.event_type_id.seats_max - 5) self.assertEqual(event.seats_taken, 5) # Un-archiving confirmed seats requires available seat(s) reg_open.action_unarchive() self.assertEqual(event.seats_reserved, 5) self.assertEqual(event.seats_available, event.event_type_id.seats_max - 6) self.assertEqual(event.seats_taken, 6) reg_draft.action_unarchive() self.assertEqual(event.seats_available, event.event_type_id.seats_max - 6) self.assertEqual(event.seats_taken, 6) reg_open.action_archive() self.assertEqual(event.seats_reserved, 4) # It is not possible to set a seats_max value below number of current # confirmed registrations. (4 "reserved" + 1 "used") with self.assertRaises(exceptions.ValidationError): event.write({'seats_max': 4}) event.write({'seats_max': 5}) self.assertEqual(event.seats_available, 0) # It is not possible to unarchive a confirmed seat if the event is fully booked with self.assertRaises(exceptions.ValidationError): reg_open.action_unarchive() # raising the limit allows it event.write({'seats_max': 6}) self.assertEqual(reg_open.state, "open") reg_open.action_unarchive() # It is not possible to confirm a draft reservation if the event is # fully booked with self.assertRaises(exceptions.ValidationError): reg_draft.write({'state': 'open'}) # It is not possible to create an open registration (default value) # when the event is full new_open_registration = { 'event_id': event.id, 'name': 'reg_open', } with self.assertRaises(exceptions.ValidationError): self.env['event.registration'].create(new_open_registration) # If the seats limitation is removed, it becomes possible of course event.write({'seats_limited': 0}) self.env['event.registration'].create(new_open_registration) reg_draft.write({'state': 'open'}) @tagged('event_registration') class TestEventRegistrationData(TestEventInternalsCommon): @users('user_eventmanager') def test_registration_partner_sync(self): """ Test registration computed fields about partner """ test_email = '"Nibbler In Space"