import { AppsEngineException } from '@rocket.chat/apps-engine/definition/exceptions'; import { Meteor } from 'meteor/meteor'; import type { IUser } from '@rocket.chat/core-typings'; import { Subscriptions, Users, Rooms } from '@rocket.chat/models'; import { Message, Team } from '@rocket.chat/core-services'; import { AppEvents, Apps } from '../../../../ee/server/apps'; import { callbacks } from '../../../../lib/callbacks'; import { roomCoordinator } from '../../../../server/lib/rooms/roomCoordinator'; import { RoomMemberActions } from '../../../../definition/IRoomTypeConfig'; export const addUserToRoom = async function ( rid: string, user: Pick | string, inviter?: Pick, silenced?: boolean, ): Promise { const now = new Date(); const room = await Rooms.findOneById(rid); if (!room) { throw new Meteor.Error('error-invalid-room', 'Invalid room', { method: 'addUserToRoom', }); } const userToBeAdded = typeof user !== 'string' ? user : await Users.findOneByUsername(user.replace('@', '')); const roomDirectives = roomCoordinator.getRoomDirectives(room.t); if (!userToBeAdded) { throw new Meteor.Error('user-not-found'); } if ( !(await roomDirectives.allowMemberAction(room, RoomMemberActions.JOIN, userToBeAdded._id)) && !(await roomDirectives.allowMemberAction(room, RoomMemberActions.INVITE, userToBeAdded._id)) ) { return; } try { await callbacks.run('federation.beforeAddUserToARoom', { user, inviter }, room); } catch (error) { throw new Meteor.Error((error as any)?.message); } await callbacks.run('beforeAddedToRoom', { user: userToBeAdded, inviter: userToBeAdded }); // Check if user is already in room const subscription = await Subscriptions.findOneByRoomIdAndUserId(rid, userToBeAdded._id); if (subscription || !userToBeAdded) { return; } try { await Apps.triggerEvent(AppEvents.IPreRoomUserJoined, room, userToBeAdded, inviter); } catch (error: any) { if (error.name === AppsEngineException.name) { throw new Meteor.Error('error-app-prevented', error.message); } throw error; } if (room.t === 'c' || room.t === 'p' || room.t === 'l') { // Add a new event, with an optional inviter await callbacks.run('beforeAddedToRoom', { user: userToBeAdded, inviter }, room); // Keep the current event await callbacks.run('beforeJoinRoom', userToBeAdded, room); } await Subscriptions.createWithRoomAndUser(room, userToBeAdded as IUser, { ts: now, open: true, alert: true, unread: 1, userMentions: 1, groupMentions: 0, }); if (!userToBeAdded.username) { throw new Meteor.Error('error-invalid-user', 'Cannot add an user to a room without a username'); } if (!silenced) { if (inviter) { const extraData = { ts: now, u: { _id: inviter._id, username: inviter.username, }, }; if (room.teamMain) { await Message.saveSystemMessage('added-user-to-team', rid, userToBeAdded.username, userToBeAdded, extraData); } else { await Message.saveSystemMessage('au', rid, userToBeAdded.username, userToBeAdded, extraData); } } else if (room.prid) { await Message.saveSystemMessage('ut', rid, userToBeAdded.username, userToBeAdded, { ts: now }); } else if (room.teamMain) { await Message.saveSystemMessage('ujt', rid, userToBeAdded.username, userToBeAdded, { ts: now }); } else { await Message.saveSystemMessage('uj', rid, userToBeAdded.username, userToBeAdded, { ts: now }); } } if (room.t === 'c' || room.t === 'p') { process.nextTick(async function () { // Add a new event, with an optional inviter await callbacks.run('afterAddedToRoom', { user: userToBeAdded, inviter }, room); // Keep the current event await callbacks.run('afterJoinRoom', userToBeAdded, room); void Apps.triggerEvent(AppEvents.IPostRoomUserJoined, room, userToBeAdded, inviter); }); } if (room.teamMain && room.teamId) { // if user is joining to main team channel, create a membership await Team.addMember(inviter || userToBeAdded, userToBeAdded._id, room.teamId); } return true; };