# Part of Odoo. See LICENSE file for full copyright and licensing details. from dateutil.relativedelta import relativedelta from unittest.mock import patch from odoo import fields from odoo.addons.mail.tests.common import MailCommon from odoo.addons.mail.tools.discuss import Store from odoo.tests.common import tagged, users from odoo.tools import mute_logger @tagged("RTC", "post_install", "-at_install") class TestChannelRTC(MailCommon): @users('employee') @mute_logger('odoo.models.unlink') def test_01_join_call(self): """Join call should remove existing sessions, remove invitation, create a new session, and return data.""" self.maxDiff = None channel = self.env['discuss.channel'].channel_create(name='Test Channel', group_id=self.env.ref('base.group_user').id) channel_member = channel.sudo().channel_member_ids.filtered(lambda channel_member: channel_member.partner_id == self.user_employee.partner_id) channel_member._rtc_join_call() self._reset_bus() with self.assertBus( [ # update sessions (self.cr.dbname, "discuss.channel", channel.id), # end of previous session (self.cr.dbname, "res.partner", self.user_employee.partner_id.id), # update sessions (self.cr.dbname, "discuss.channel", channel.id), ], [ { "type": "discuss.channel.rtc.session/ended", "payload": {"sessionId": channel_member.rtc_session_ids.id}, }, { "type": "mail.record/insert", "payload": { "discuss.channel": [ { "id": channel.id, "rtcSessions": [("DELETE", [channel_member.rtc_session_ids.id])], }, ], }, }, { "type": "mail.record/insert", "payload": { "discuss.channel": [ { "id": channel.id, "rtcSessions": [("ADD", [channel_member.rtc_session_ids.id + 1])], }, ], "discuss.channel.member": [ { "id": channel_member.id, "persona": {"id": channel_member.partner_id.id, "type": "partner"}, "thread": { "id": channel_member.channel_id.id, "model": "discuss.channel", }, }, ], "discuss.channel.rtc.session": [ { "channelMember": channel_member.id, "id": channel_member.rtc_session_ids.id + 1, }, ], "res.partner": [ { "id": channel_member.partner_id.id, "im_status": channel_member.partner_id.im_status, "name": channel_member.partner_id.name, }, ], }, }, ], ): store = Store() channel_member._rtc_join_call(store) res = store.get_result() self.assertEqual( res, { "discuss.channel": [ { "id": channel.id, "rtcSessions": [ ("ADD", [channel_member.rtc_session_ids.id]), ("DELETE", [channel_member.rtc_session_ids.id - 1]), ], }, ], "discuss.channel.member": [ { "id": channel_member.id, "persona": {"id": channel_member.partner_id.id, "type": "partner"}, "thread": { "id": channel_member.channel_id.id, "model": "discuss.channel", }, }, ], "discuss.channel.rtc.session": [ { "channelMember": channel_member.id, "id": channel_member.rtc_session_ids.id, }, ], "res.partner": [ { "id": channel_member.partner_id.id, "im_status": channel_member.partner_id.im_status, "name": channel_member.partner_id.name, }, ], "Rtc": { "iceServers": False, "selfSession": channel_member.rtc_session_ids.id, "serverInfo": None, }, }, ) @users('employee') @mute_logger('odoo.models.unlink') def test_10_start_call_in_chat_should_invite_all_members_to_call(self): test_user = self.env['res.users'].sudo().create({'name': "Test User", 'login': 'test'}) channel = self.env['discuss.channel'].channel_get(partners_to=(self.user_employee.partner_id + test_user.partner_id).ids) channel_member = channel.sudo().channel_member_ids.filtered(lambda channel_member: channel_member.partner_id == self.user_employee.partner_id) channel_member_test_user = channel.sudo().channel_member_ids.filtered(lambda channel_member: channel_member.partner_id == test_user.partner_id) channel_member._rtc_join_call() last_rtc_session_id = channel_member.rtc_session_ids.id channel_member._rtc_leave_call() self._reset_bus() with self.assertBus( [ # update new session (self.cr.dbname, "discuss.channel", channel.id), # message_post "started a live conference" (not asserted below) (self.cr.dbname, "discuss.channel", channel.id), # update new message separator (self.cr.dbname, "res.partner", self.user_employee.partner_id.id), # update of pin state (not asserted below) (self.cr.dbname, "discuss.channel", channel.id, "members"), # update of last interest (not asserted below) (self.cr.dbname, "discuss.channel", channel.id), # incoming invitation (self.cr.dbname, "res.partner", test_user.partner_id.id), # update list of invitations (self.cr.dbname, "discuss.channel", channel.id), ], [ { "type": "mail.record/insert", "payload": { "discuss.channel": [ { "id": channel.id, "rtcSessions": [("ADD", [last_rtc_session_id + 1])], }, ], "discuss.channel.member": [ { "id": channel_member.id, "persona": {"id": channel_member.partner_id.id, "type": "partner"}, "thread": { "id": channel_member.channel_id.id, "model": "discuss.channel", }, }, ], "discuss.channel.rtc.session": [ { "channelMember": channel_member.id, "id": last_rtc_session_id + 1, }, ], "res.partner": [ { "id": channel_member.partner_id.id, "im_status": channel_member.partner_id.im_status, "name": channel_member.partner_id.name, }, ], }, }, { "type": "mail.record/insert", "payload": { "discuss.channel": [ { "id": channel.id, "invitedMembers": [("ADD", [channel_member_test_user.id])], } ], "discuss.channel.member": [ { "id": channel_member_test_user.id, "persona": { "id": channel_member_test_user.partner_id.id, "type": "partner", }, "thread": { "id": channel_member_test_user.channel_id.id, "model": "discuss.channel", }, }, ], "res.partner": [ { "id": channel_member_test_user.partner_id.id, "im_status": channel_member_test_user.partner_id.im_status, "name": channel_member_test_user.partner_id.name, }, ], }, }, ], ): now = fields.Datetime.now() with patch.object(fields.Datetime, 'now', lambda: now + relativedelta(seconds=5)): channel_member._rtc_join_call() @users('employee') @mute_logger('odoo.models.unlink') def test_11_start_call_in_group_should_invite_all_members_to_call(self): test_user = self.env['res.users'].sudo().create({'name': "Test User", 'login': 'test'}) test_guest = self.env['mail.guest'].sudo().create({'name': "Test Guest"}) channel = self.env['discuss.channel'].create_group(partners_to=(self.user_employee.partner_id + test_user.partner_id).ids) channel.add_members(guest_ids=test_guest.ids) channel_member_test_user = channel.sudo().channel_member_ids.filtered(lambda channel_member: channel_member.partner_id == test_user.partner_id) channel_member_test_guest = channel.sudo().channel_member_ids.filtered(lambda channel_member: channel_member.guest_id == test_guest) channel_member = channel.sudo().channel_member_ids.filtered(lambda channel_member: channel_member.partner_id == self.user_employee.partner_id) channel_member._rtc_join_call() last_rtc_session_id = channel_member.rtc_session_ids.id channel_member._rtc_leave_call() self._reset_bus() with self.assertBus( [ # update new session (self.cr.dbname, "discuss.channel", channel.id), # message_post "started a live conference" (not asserted below) (self.cr.dbname, "discuss.channel", channel.id), # update new message separator (self.cr.dbname, "res.partner", self.user_employee.partner_id.id), # update of pin state (not asserted below) (self.cr.dbname, "discuss.channel", channel.id, "members"), # update of last interest (not asserted below) (self.cr.dbname, "discuss.channel", channel.id), # incoming invitation (self.cr.dbname, "res.partner", test_user.partner_id.id), # incoming invitation (self.cr.dbname, "mail.guest", test_guest.id), # update list of invitations (self.cr.dbname, "discuss.channel", channel.id), ], [ { "type": "mail.record/insert", "payload": { "discuss.channel": [ { "id": channel.id, "rtcSessions": [("ADD", [last_rtc_session_id + 1])], }, ], "discuss.channel.member": [ { "id": channel_member.id, "persona": {"id": channel_member.partner_id.id, "type": "partner"}, "thread": { "id": channel_member.channel_id.id, "model": "discuss.channel", }, }, ], "discuss.channel.rtc.session": [ { "channelMember": channel_member.id, "id": last_rtc_session_id + 1, }, ], "res.partner": [ { "id": channel_member.partner_id.id, "im_status": channel_member.partner_id.im_status, "name": channel_member.partner_id.name, }, ], }, }, { "type": "mail.record/insert", "payload": { "discuss.channel": [ { "id": channel.id, "rtcSessions": [("ADD", [last_rtc_session_id + 1])], }, ], "discuss.channel.member": [ { "id": channel_member.id, "persona": {"id": channel_member.partner_id.id, "type": "partner"}, "thread": { "id": channel_member.channel_id.id, "model": "discuss.channel", }, }, ], "discuss.channel.rtc.session": [ { "channelMember": channel_member.id, "id": last_rtc_session_id + 1, }, ], "res.partner": [ { "id": channel_member.partner_id.id, "im_status": channel_member.partner_id.im_status, "name": channel_member.partner_id.name, }, ], }, }, { "type": "mail.record/insert", "payload": { "discuss.channel": [ { "id": channel.id, "invitedMembers": [ ( "ADD", [channel_member_test_user.id, channel_member_test_guest.id], ) ], }, ], "discuss.channel.member": [ { "id": channel_member_test_user.id, "persona": { "id": channel_member_test_user.partner_id.id, "type": "partner", }, "thread": { "id": channel_member_test_user.channel_id.id, "model": "discuss.channel", }, }, { "id": channel_member_test_guest.id, "persona": { "id": channel_member_test_guest.guest_id.id, "type": "guest", }, "thread": { "id": channel_member_test_guest.channel_id.id, "model": "discuss.channel", }, }, ], "mail.guest": [ { "id": channel_member_test_guest.guest_id.id, "im_status": channel_member_test_guest.guest_id.im_status, "name": channel_member_test_guest.guest_id.name, }, ], "res.partner": [ { "id": channel_member_test_user.partner_id.id, "im_status": channel_member_test_user.partner_id.im_status, "name": channel_member_test_user.partner_id.name, }, ], }, }, ], ): now = fields.Datetime.now() with patch.object(fields.Datetime, 'now', lambda: now + relativedelta(seconds=5)): channel_member._rtc_join_call() @users('employee') @mute_logger('odoo.models.unlink') def test_20_join_call_should_cancel_pending_invitations(self): test_user = self.env['res.users'].sudo().create({'name': "Test User", 'login': 'test'}) test_guest = self.env['mail.guest'].sudo().create({'name': "Test Guest"}) channel = self.env['discuss.channel'].create_group(partners_to=(self.user_employee.partner_id + test_user.partner_id).ids) channel.add_members(guest_ids=test_guest.ids) channel_member = channel.sudo().channel_member_ids.filtered(lambda channel_member: channel_member.partner_id == self.user_employee.partner_id) channel_member._rtc_join_call() channel_member_test_user = channel.sudo().channel_member_ids.filtered(lambda channel_member: channel_member.partner_id == test_user.partner_id) self._reset_bus() with self.assertBus( [ # update invitation (self.cr.dbname, "res.partner", test_user.partner_id.id), # update list of invitations (self.cr.dbname, "discuss.channel", channel.id), # update sessions (self.cr.dbname, "discuss.channel", channel.id), ], [ { "type": "mail.record/insert", "payload": { "discuss.channel": [{"id": channel.id, "rtcInvitingSession": False}] }, }, { "type": "mail.record/insert", "payload": { "discuss.channel": [ { "id": channel.id, "invitedMembers": [("DELETE", [channel_member_test_user.id])], }, ], "discuss.channel.member": [ { "id": channel_member_test_user.id, "persona": { "id": channel_member_test_user.partner_id.id, "type": "partner", }, "thread": { "id": channel_member_test_user.channel_id.id, "model": "discuss.channel", }, }, ], "res.partner": [ { "id": channel_member_test_user.partner_id.id, "im_status": channel_member_test_user.partner_id.im_status, "name": channel_member_test_user.partner_id.name, }, ], }, }, { "type": "mail.record/insert", "payload": { "discuss.channel": [ { "id": channel.id, "rtcSessions": [("ADD", [channel_member.rtc_session_ids.id + 1])], }, ], "discuss.channel.member": [ { "id": channel_member_test_user.id, "persona": { "id": channel_member_test_user.partner_id.id, "type": "partner", }, "thread": { "id": channel_member_test_user.channel_id.id, "model": "discuss.channel", }, }, ], "discuss.channel.rtc.session": [ { "channelMember": channel_member_test_user.id, "id": channel_member.rtc_session_ids.id + 1, }, ], "res.partner": [ { "id": channel_member_test_user.partner_id.id, "im_status": channel_member_test_user.partner_id.im_status, "name": channel_member_test_user.partner_id.name, }, ], }, }, ], ): channel_member_test_user._rtc_join_call() channel_member_test_guest = channel.sudo().channel_member_ids.filtered(lambda channel_member: channel_member.guest_id == test_guest) self._reset_bus() with self.assertBus( [ # update invitation (self.cr.dbname, "mail.guest", test_guest.id), # update list of invitations (self.cr.dbname, "discuss.channel", channel.id), # update sessions (self.cr.dbname, "discuss.channel", channel.id), ], [ { "type": "mail.record/insert", "payload": { "discuss.channel": [{"id": channel.id, "rtcInvitingSession": False}] }, }, { "type": "mail.record/insert", "payload": { "discuss.channel": [ { "id": channel.id, "invitedMembers": [("DELETE", [channel_member_test_guest.id])], }, ], "discuss.channel.member": [ { "id": channel_member_test_guest.id, "persona": { "id": channel_member_test_guest.guest_id.id, "type": "guest", }, "thread": { "id": channel_member_test_guest.channel_id.id, "model": "discuss.channel", }, }, ], "mail.guest": [ { "id": channel_member_test_guest.guest_id.id, "im_status": channel_member_test_guest.guest_id.im_status, "name": channel_member_test_guest.guest_id.name, }, ], }, }, { "type": "mail.record/insert", "payload": { "discuss.channel": [ { "id": channel.id, "rtcSessions": [("ADD", [channel_member.rtc_session_ids.id + 2])], }, ], "discuss.channel.member": [ { "id": channel_member_test_guest.id, "persona": { "id": channel_member_test_guest.guest_id.id, "type": "guest", }, "thread": { "id": channel_member_test_guest.channel_id.id, "model": "discuss.channel", }, }, ], "discuss.channel.rtc.session": [ { "channelMember": channel_member_test_guest.id, "id": channel_member.rtc_session_ids.id + 2, }, ], "mail.guest": [ { "id": channel_member_test_guest.guest_id.id, "im_status": channel_member_test_guest.guest_id.im_status, "name": channel_member_test_guest.guest_id.name, }, ], }, }, ], ): channel_member_test_guest._rtc_join_call() @users('employee') @mute_logger('odoo.models.unlink') def test_21_leave_call_should_cancel_pending_invitations(self): test_user = self.env['res.users'].sudo().create({'name': "Test User", 'login': 'test'}) test_guest = self.env['mail.guest'].sudo().create({'name': "Test Guest"}) channel = self.env['discuss.channel'].create_group(partners_to=(self.user_employee.partner_id + test_user.partner_id).ids) channel.add_members(guest_ids=test_guest.ids) channel_member = channel.sudo().channel_member_ids.filtered(lambda channel_member: channel_member.partner_id == self.user_employee.partner_id) channel_member._rtc_join_call() channel_member_test_user = channel.sudo().channel_member_ids.filtered(lambda channel_member: channel_member.partner_id == test_user.partner_id) self._reset_bus() with self.assertBus( [ # update invitation (self.cr.dbname, "res.partner", test_user.partner_id.id), # update list of invitations (self.cr.dbname, "discuss.channel", channel.id), ], [ { "type": "mail.record/insert", "payload": { "discuss.channel": [{"id": channel.id, "rtcInvitingSession": False}] }, }, { "type": "mail.record/insert", "payload": { "discuss.channel": [ { "id": channel.id, "invitedMembers": [("DELETE", [channel_member_test_user.id])], }, ], "discuss.channel.member": [ { "id": channel_member_test_user.id, "persona": { "id": channel_member_test_user.partner_id.id, "type": "partner", }, "thread": { "id": channel_member_test_user.channel_id.id, "model": "discuss.channel", }, }, ], "res.partner": [ { "id": channel_member_test_user.partner_id.id, "im_status": channel_member_test_user.partner_id.im_status, "name": channel_member_test_user.partner_id.name, }, ], }, }, ], ): channel_member_test_user._rtc_leave_call() channel_member_test_guest = channel.sudo().channel_member_ids.filtered(lambda channel_member: channel_member.guest_id == test_guest) self._reset_bus() with self.assertBus( [ # update invitation (self.cr.dbname, "mail.guest", test_guest.id), # update list of invitations (self.cr.dbname, "discuss.channel", channel.id), ], [ { "type": "mail.record/insert", "payload": { "discuss.channel": [{"id": channel.id, "rtcInvitingSession": False}] }, }, { "type": "mail.record/insert", "payload": { "discuss.channel": [ { "id": channel.id, "invitedMembers": [("DELETE", [channel_member_test_guest.id])], }, ], "discuss.channel.member": [ { "id": channel_member_test_guest.id, "persona": { "id": channel_member_test_guest.guest_id.id, "type": "guest", }, "thread": { "id": channel_member_test_guest.channel_id.id, "model": "discuss.channel", }, }, ], "mail.guest": [ { "id": channel_member_test_guest.guest_id.id, "im_status": channel_member_test_guest.guest_id.im_status, "name": channel_member_test_guest.guest_id.name, }, ], }, }, ], ): channel_member_test_guest._rtc_leave_call() @users('employee') @mute_logger('odoo.models.unlink') def test_25_lone_call_participant_leaving_call_should_cancel_pending_invitations(self): test_user = self.env['res.users'].sudo().create({'name': "Test User", 'login': 'test'}) test_guest = self.env['mail.guest'].sudo().create({'name': "Test Guest"}) channel = self.env['discuss.channel'].create_group(partners_to=(self.user_employee.partner_id + test_user.partner_id).ids) channel.add_members(guest_ids=test_guest.ids) channel_member = channel.sudo().channel_member_ids.filtered(lambda channel_member: channel_member.partner_id == self.user_employee.partner_id) channel_member_test_user = channel.sudo().channel_member_ids.filtered(lambda channel_member: channel_member.partner_id == test_user.partner_id) channel_member_test_guest = channel.sudo().channel_member_ids.filtered(lambda channel_member: channel_member.guest_id == test_guest) channel_member._rtc_join_call() self._reset_bus() with self.assertBus( [ # update invitation (self.cr.dbname, "res.partner", test_user.partner_id.id), # update invitation (self.cr.dbname, "mail.guest", test_guest.id), # update list of invitations (self.cr.dbname, "discuss.channel", channel.id), # update sessions (self.cr.dbname, "discuss.channel", channel.id), # end session (self.cr.dbname, "res.partner", self.user_employee.partner_id.id), ], [ { "type": "discuss.channel.rtc.session/ended", "payload": {"sessionId": channel_member.rtc_session_ids.id}, }, { "type": "mail.record/insert", "payload": { "discuss.channel": [{"id": channel.id, "rtcInvitingSession": False}] }, }, { "type": "mail.record/insert", "payload": { "discuss.channel": [{"id": channel.id, "rtcInvitingSession": False}] }, }, { "type": "mail.record/insert", "payload": { "discuss.channel": [ { "id": channel.id, "invitedMembers": [ ( "DELETE", [channel_member_test_user.id, channel_member_test_guest.id], ) ], }, ], "discuss.channel.member": [ { "id": channel_member_test_user.id, "persona": { "id": channel_member_test_user.partner_id.id, "type": "partner", }, "thread": { "id": channel_member_test_user.channel_id.id, "model": "discuss.channel", }, }, { "id": channel_member_test_guest.id, "persona": { "id": channel_member_test_guest.guest_id.id, "type": "guest", }, "thread": { "id": channel_member_test_guest.channel_id.id, "model": "discuss.channel", }, }, ], "mail.guest": [ { "id": channel_member_test_guest.guest_id.id, "im_status": channel_member_test_guest.guest_id.im_status, "name": channel_member_test_guest.guest_id.name, }, ], "res.partner": [ { "id": channel_member_test_user.partner_id.id, "im_status": channel_member_test_user.partner_id.im_status, "name": channel_member_test_user.partner_id.name, }, ], }, }, { "type": "mail.record/insert", "payload": { "discuss.channel": [ { "id": channel.id, "rtcSessions": [("DELETE", [channel_member.rtc_session_ids.id])], }, ], }, }, ], ): channel_member._rtc_leave_call() @users('employee') @mute_logger('odoo.models.unlink') def test_30_add_members_while_in_call_should_invite_new_members_to_call(self): test_user = self.env['res.users'].sudo().create({'name': "Test User", 'login': 'test'}) test_guest = self.env['mail.guest'].sudo().create({'name': "Test Guest"}) channel = self.env['discuss.channel'].create_group(partners_to=self.user_employee.partner_id.ids) channel_member = channel.sudo().channel_member_ids.filtered(lambda member: member.partner_id == self.user_employee.partner_id) now = fields.Datetime.now() with patch.object(fields.Datetime, 'now', lambda: now + relativedelta(seconds=5)): channel_member._rtc_join_call() self._reset_bus() with self.mock_bus(): with patch.object(fields.Datetime, 'now', lambda: now + relativedelta(seconds=10)): channel.add_members(partner_ids=test_user.partner_id.ids, guest_ids=test_guest.ids, invite_to_rtc_call=True) channel_member_test_user = channel.sudo().channel_member_ids.filtered(lambda member: member.partner_id == test_user.partner_id) channel_member_test_guest = channel.sudo().channel_member_ids.filtered(lambda member: member.guest_id == test_guest) found_bus_notifs = self.assertBusNotifications( [ # discuss.channel/joined (self.cr.dbname, "res.partner", test_user.partner_id.id), # mail.record/insert - discuss.channel (last_interest_dt) (self.cr.dbname, "discuss.channel", channel.id), # mail.record/insert - discuss.channel.member (message_unread_counter, new_message_separator, …) (self.cr.dbname, "res.partner", self.user_employee.partner_id.id), # mail.record/insert - discuss.channel (is_pinned: true) (self.cr.dbname, "discuss.channel", channel.id, "members"), # discuss.channel/new_message (self.cr.dbname, "discuss.channel", channel.id), # discuss.channel/joined (self.cr.dbname, "mail.guest", test_guest.id), # mail.record/insert - discuss.channel.member (message_unread_counter, new_message_separator, …) (self.cr.dbname, "res.partner", self.user_employee.partner_id.id), # mail.record/insert - discuss.channel (is_pinned: true) (self.cr.dbname, "discuss.channel", channel.id, "members"), # discuss.channel/new_message (self.cr.dbname, "discuss.channel", channel.id), # mail.record/insert - discuss.channel (memberCount), discuss.channel.member (self.cr.dbname, "discuss.channel", channel.id), # mail.record/insert - discuss.channel (rtcInvitingSession), discuss.channel.member (self.cr.dbname, "res.partner", test_user.partner_id.id), # mail.record/insert - discuss.channel (rtcInvitingSession), discuss.channel.member (self.cr.dbname, "mail.guest", test_guest.id), # mail.record/insert - discuss.channel (invitedMembers), discuss.channel.member (self.cr.dbname, "discuss.channel", channel.id), ], message_items=[ { "type": "mail.record/insert", "payload": { "discuss.channel": [ { "id": channel.id, "rtcInvitingSession": channel_member.rtc_session_ids.id, }, ], "discuss.channel.member": [ { "id": channel_member.id, "persona": {"id": channel_member.partner_id.id, "type": "partner"}, "thread": { "id": channel_member.channel_id.id, "model": "discuss.channel", }, }, ], "discuss.channel.rtc.session": [ { "channelMember": channel_member.id, "id": channel_member.rtc_session_ids.id, "isCameraOn": channel_member.rtc_session_ids.is_camera_on, "isDeaf": channel_member.rtc_session_ids.is_deaf, "isScreenSharingOn": channel_member.rtc_session_ids.is_screen_sharing_on, "isSelfMuted": channel_member.rtc_session_ids.is_muted, }, ], "res.partner": [ { "id": channel_member.partner_id.id, "im_status": channel_member.partner_id.im_status, "name": channel_member.partner_id.name, }, ], }, }, { "type": "mail.record/insert", "payload": { "discuss.channel": [ { "id": channel.id, "rtcInvitingSession": channel_member.rtc_session_ids.id, }, ], "discuss.channel.member": [ { "id": channel_member.id, "persona": {"id": channel_member.partner_id.id, "type": "partner"}, "thread": { "id": channel_member.channel_id.id, "model": "discuss.channel", }, }, ], "discuss.channel.rtc.session": [ { "channelMember": channel_member.id, "id": channel_member.rtc_session_ids.id, "isCameraOn": channel_member.rtc_session_ids.is_camera_on, "isDeaf": channel_member.rtc_session_ids.is_deaf, "isScreenSharingOn": channel_member.rtc_session_ids.is_screen_sharing_on, "isSelfMuted": channel_member.rtc_session_ids.is_muted, }, ], "res.partner": [ { "id": channel_member.partner_id.id, "im_status": channel_member.partner_id.im_status, "name": channel_member.partner_id.name, }, ], }, }, { "type": "mail.record/insert", "payload": { "discuss.channel": [ { "id": channel.id, "invitedMembers": [ ( "ADD", [channel_member_test_user.id, channel_member_test_guest.id], ) ], } ], "discuss.channel.member": [ { "id": channel_member_test_user.id, "persona": { "id": channel_member_test_user.partner_id.id, "type": "partner", }, "thread": { "id": channel_member_test_user.channel_id.id, "model": "discuss.channel", }, }, { "id": channel_member_test_guest.id, "persona": { "id": channel_member_test_guest.guest_id.id, "type": "guest", }, "thread": { "id": channel_member_test_guest.channel_id.id, "model": "discuss.channel", }, }, ], "mail.guest": [ { "id": channel_member_test_guest.guest_id.id, "im_status": channel_member_test_guest.guest_id.im_status, "name": channel_member_test_guest.guest_id.name, }, ], "res.partner": [ { "id": channel_member_test_user.partner_id.id, "im_status": channel_member_test_user.partner_id.im_status, "name": channel_member_test_user.partner_id.name, }, ], }, }, ], ) self.assertEqual(self._new_bus_notifs, found_bus_notifs) @users('employee') @mute_logger('odoo.models.unlink') def test_40_leave_call_should_remove_existing_sessions_of_user_in_channel_and_return_data(self): channel = self.env['discuss.channel'].create_group(partners_to=self.user_employee.partner_id.ids) channel_member = channel.sudo().channel_member_ids.filtered(lambda channel_member: channel_member.partner_id == self.user_employee.partner_id) channel_member._rtc_join_call() self._reset_bus() with self.assertBus( [ # update list of sessions (self.cr.dbname, "discuss.channel", channel.id), # end session (self.cr.dbname, "res.partner", self.user_employee.partner_id.id), ], [ { "type": "discuss.channel.rtc.session/ended", "payload": {"sessionId": channel_member.rtc_session_ids.id}, }, { "type": "mail.record/insert", "payload": { "discuss.channel": [ { "id": channel.id, "rtcSessions": [("DELETE", [channel_member.rtc_session_ids.id])], }, ], }, }, ], ): now = fields.Datetime.now() with patch.object(fields.Datetime, 'now', lambda: now + relativedelta(seconds=5)): channel_member._rtc_leave_call() @users('employee') @mute_logger('odoo.models.unlink') def test_50_garbage_collect_should_remove_old_sessions_and_notify_data(self): self.env["discuss.channel.rtc.session"].sudo().search([]).unlink() # clean up before test channel = self.env['discuss.channel'].create_group(partners_to=self.user_employee.partner_id.ids) channel_member = channel.sudo().channel_member_ids.filtered(lambda channel_member: channel_member.partner_id == self.user_employee.partner_id) channel_member._rtc_join_call() channel_member.rtc_session_ids.flush_model() channel_member.rtc_session_ids._write({'write_date': fields.Datetime.now() - relativedelta(days=2)}) self._reset_bus() with self.assertBus( [ # update list of sessions (self.cr.dbname, "discuss.channel", channel.id), # session ended (self.cr.dbname, "res.partner", self.user_employee.partner_id.id), ], [ { "type": "discuss.channel.rtc.session/ended", "payload": {"sessionId": channel_member.rtc_session_ids.id}, }, { "type": "mail.record/insert", "payload": { "discuss.channel": [ { "id": channel.id, "rtcSessions": [("DELETE", [channel_member.rtc_session_ids.id])], }, ], }, }, ], ): self.env['discuss.channel.rtc.session'].sudo()._gc_inactive_sessions() self.assertFalse(channel_member.rtc_session_ids) @users('employee') @mute_logger('odoo.models.unlink') def test_51_action_disconnect_should_remove_selected_session_and_notify_data(self): channel = self.env['discuss.channel'].create_group(partners_to=self.user_employee.partner_id.ids) channel_member = channel.sudo().channel_member_ids.filtered(lambda channel_member: channel_member.partner_id == self.user_employee.partner_id) channel_member._rtc_join_call() self._reset_bus() with self.assertBus( [ # update list of sessions (self.cr.dbname, "discuss.channel", channel.id), # session ended (self.cr.dbname, "res.partner", self.user_employee.partner_id.id), ], [ { "type": "discuss.channel.rtc.session/ended", "payload": {"sessionId": channel_member.rtc_session_ids.id}, }, { "type": "mail.record/insert", "payload": { "discuss.channel": [ { "id": channel.id, "rtcSessions": [("DELETE", [channel_member.rtc_session_ids.id])], }, ], }, }, ], ): channel_member.rtc_session_ids.action_disconnect() self.assertFalse(channel_member.rtc_session_ids) @users('employee') @mute_logger('odoo.models.unlink') def test_60_rtc_sync_sessions_should_gc_and_return_outdated_and_active_sessions(self): channel = self.env['discuss.channel'].create_group(partners_to=self.user_employee.partner_id.ids) channel_member = channel.sudo().channel_member_ids.filtered(lambda channel_member: channel_member.partner_id == self.user_employee.partner_id) store = Store() channel_member._rtc_join_call(store) join_call_values = store.get_result() test_guest = self.env['mail.guest'].sudo().create({'name': "Test Guest"}) test_channel_member = self.env['discuss.channel.member'].create({ 'guest_id': test_guest.id, 'channel_id': channel.id, }) test_session = self.env['discuss.channel.rtc.session'].sudo().create({'channel_member_id': test_channel_member.id}) test_session.flush_model() test_session._write({'write_date': fields.Datetime.now() - relativedelta(days=2)}) unused_ids = [9998, 9999] self._reset_bus() with self.assertBus( [ # update list of sessions (self.cr.dbname, "discuss.channel", channel.id), # session ended (self.cr.dbname, "mail.guest", test_guest.id), ], [ { "type": "discuss.channel.rtc.session/ended", "payload": {"sessionId": test_session.id}, }, { "type": "mail.record/insert", "payload": { "discuss.channel": [ {"id": channel.id, "rtcSessions": [("DELETE", [test_session.id])]}, ], }, }, ], ): current_rtc_sessions, outdated_rtc_sessions = channel_member._rtc_sync_sessions( check_rtc_session_ids=[join_call_values["Rtc"]["selfSession"]] + unused_ids ) self.assertEqual(channel_member.rtc_session_ids, current_rtc_sessions) self.assertEqual(unused_ids, outdated_rtc_sessions.ids) self.assertFalse(outdated_rtc_sessions.exists())