The communications platform that puts data protection first.
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
Rocket.Chat/app/lib/server/functions/createRoom.js

132 lines
3.4 KiB

import { Meteor } from 'meteor/meteor';
import _ from 'underscore';
import s from 'underscore.string';
import { Users, Rooms, Subscriptions } from '../../../models';
import { callbacks } from '../../../callbacks';
import { addUserRoles } from '../../../authorization';
import { getValidRoomName } from '../../../utils';
import { Apps } from '../../../apps/server';
import { createDirectRoom } from './createDirectRoom';
export const createRoom = function(type, name, owner, members = [], readOnly, extraData = {}, options = {}) {
callbacks.run('beforeCreateRoom', { type, name, owner, members, readOnly, extraData, options });
if (type === 'd') {
return createDirectRoom(members, extraData, options);
}
name = s.trim(name);
owner = s.trim(owner);
members = [].concat(members);
if (!name) {
throw new Meteor.Error('error-invalid-name', 'Invalid name', { function: 'RocketChat.createRoom' });
}
owner = Users.findOneByUsernameIgnoringCase(owner, { fields: { username: 1 } });
if (!owner) {
throw new Meteor.Error('error-invalid-user', 'Invalid user', { function: 'RocketChat.createRoom' });
}
if (!_.contains(members, owner.username)) {
members.push(owner.username);
}
if (extraData.broadcast) {
readOnly = true;
delete extraData.reactWhenReadOnly;
}
const now = new Date();
const validRoomNameOptions = {};
if (options.nameValidationRegex) {
validRoomNameOptions.nameValidationRegex = options.nameValidationRegex;
}
let room = {
name: getValidRoomName(name, null, validRoomNameOptions),
fname: name,
t: type,
msgs: 0,
usersCount: 0,
u: {
_id: owner._id,
username: owner.username,
},
...extraData,
ts: now,
ro: readOnly === true,
};
if (Apps && Apps.isLoaded()) {
const prevent = Promise.await(Apps.getBridges().getListenerBridge().roomEvent('IPreRoomCreatePrevent', room));
if (prevent) {
throw new Meteor.Error('error-app-prevented-creation', 'A Rocket.Chat App prevented the room creation.');
}
let result;
result = Promise.await(Apps.getBridges().getListenerBridge().roomEvent('IPreRoomCreateExtend', room));
result = Promise.await(Apps.getBridges().getListenerBridge().roomEvent('IPreRoomCreateModify', result));
if (typeof result === 'object') {
room = Object.assign(room, result);
}
}
if (type === 'c') {
callbacks.run('beforeCreateChannel', owner, room);
}
room = Rooms.createWithFullRoomData(room);
for (const username of members) {
const member = Users.findOneByUsername(username, { fields: { username: 1, 'settings.preferences': 1 } });
if (!member) {
continue;
}
const extra = options.subscriptionExtra || {};
extra.open = true;
if (room.prid) {
extra.prid = room.prid;
}
if (username === owner.username) {
extra.ls = now;
}
Subscriptions.createWithRoomAndUser(room, member, extra);
}
addUserRoles(owner._id, ['owner'], room._id);
if (type === 'c') {
Meteor.defer(() => {
callbacks.run('afterCreateChannel', owner, room);
});
} else if (type === 'p') {
Meteor.defer(() => {
callbacks.run('afterCreatePrivateGroup', owner, room);
});
}
Meteor.defer(() => {
callbacks.run('afterCreateRoom', owner, room);
});
if (Apps && Apps.isLoaded()) {
// This returns a promise, but it won't mutate anything about the message
// so, we don't really care if it is successful or fails
Apps.getBridges().getListenerBridge().roomEvent('IPostRoomCreate', room);
}
return {
rid: room._id, // backwards compatible
...room,
};
};