@ -1,6 +1,5 @@
'use strict' ;
'use strict' ;
var async = require ( 'async' ) ;
var validator = require ( 'validator' ) ;
var validator = require ( 'validator' ) ;
var db = require ( '../database' ) ;
var db = require ( '../database' ) ;
@ -10,34 +9,23 @@ var privileges = require('../privileges');
var meta = require ( '../meta' ) ;
var meta = require ( '../meta' ) ;
module . exports = function ( Messaging ) {
module . exports = function ( Messaging ) {
Messaging . getRoomData = function ( roomId , callback ) {
Messaging . getRoomData = async ( roomId ) => {
async . waterfall ( [
const data = await db . getObject ( 'chat:room:' + roomId ) ;
function ( next ) {
if ( ! data ) {
db . getObject ( 'chat:room:' + roomId , next ) ;
throw new Error ( '[[error:no-chat-room]]' ) ;
} ,
}
function ( data , next ) {
if ( ! data ) {
modifyRoomData ( [ data ] ) ;
return callback ( new Error ( '[[error:no-chat-room]]' ) ) ;
return data ;
}
modifyRoomData ( [ data ] ) ;
next ( null , data ) ;
} ,
] , callback ) ;
} ;
} ;
Messaging . getRoomsData = function ( roomIds , callback ) {
Messaging . getRoomsData = async ( roomIds ) => {
var keys = roomIds . map ( function ( roomId ) {
const roomData = await db . getObjects ( roomIds . map ( function ( roomId ) {
return 'chat:room:' + roomId ;
return 'chat:room:' + roomId ;
} ) ;
} ) ) ;
async . waterfall ( [
function ( next ) {
modifyRoomData ( roomData ) ;
db . getObjects ( keys , next ) ;
return roomData ;
} ,
function ( roomData , next ) {
modifyRoomData ( roomData ) ;
next ( null , roomData ) ;
} ,
] , callback ) ;
} ;
} ;
function modifyRoomData ( rooms ) {
function modifyRoomData ( rooms ) {
@ -52,284 +40,204 @@ module.exports = function (Messaging) {
} ) ;
} ) ;
}
}
Messaging . newRoom = function ( uid , toUids , callback ) {
Messaging . newRoom = async ( uid , toUids ) => {
var roomId ;
const now = Date . now ( ) ;
var now = Date . now ( ) ;
const roomId = await db . incrObjectField ( 'global' , 'nextChatRoomId' ) ;
async . waterfall ( [
const room = {
function ( next ) {
owner : uid ,
db . incrObjectField ( 'global' , 'nextChatRoomId' , next ) ;
roomId : roomId ,
} ,
} ;
function ( _roomId , next ) {
roomId = _roomId ;
var room = {
owner : uid ,
roomId : roomId ,
} ;
db . setObject ( 'chat:room:' + roomId , room , next ) ;
} ,
function ( next ) {
db . sortedSetAdd ( 'chat:room:' + roomId + ':uids' , now , uid , next ) ;
} ,
function ( next ) {
Messaging . addUsersToRoom ( uid , toUids , roomId , next ) ;
} ,
function ( next ) {
Messaging . addRoomToUsers ( roomId , [ uid ] . concat ( toUids ) , now , next ) ;
} ,
function ( next ) {
next ( null , roomId ) ;
} ,
] , callback ) ;
} ;
Messaging . isUserInRoom = function ( uid , roomId , callback ) {
await Promise . all ( [
async . waterfall ( [
db . setObject ( 'chat:room:' + roomId , room ) ,
function ( next ) {
db . sortedSetAdd ( 'chat:room:' + roomId + ':uids' , now , uid ) ,
db . isSortedSetMember ( 'chat:room:' + roomId + ':uids' , uid , next ) ;
] ) ;
} ,
await Promise . all ( [
function ( inRoom , next ) {
Messaging . addUsersToRoom ( uid , toUids , roomId ) ,
plugins . fireHook ( 'filter:messaging.isUserInRoom' , { uid : uid , roomId : roomId , inRoom : inRoom } , next ) ;
Messaging . addRoomToUsers ( roomId , [ uid ] . concat ( toUids ) , now ) ,
} ,
] ) ;
function ( data , next ) {
next ( null , data . inRoom ) ;
} ,
] , callback ) ;
} ;
Messaging . roomExists = function ( roomId , callback ) {
return roomId ;
db . exists ( 'chat:room:' + roomId + ':uids' , callback ) ;
} ;
} ;
Messaging . getUserCountInRoom = function ( roomId , callback ) {
Messaging . isUserInRoom = async ( uid , roomId ) => {
db . sortedSetCard ( 'chat:room:' + roomId + ':uids' , callback ) ;
const inRoom = db . isSortedSetMember ( 'chat:room:' + roomId + ':uids' , uid ) ;
const data = await plugins . fireHook ( 'filter:messaging.isUserInRoom' , { uid : uid , roomId : roomId , inRoom : inRoom } ) ;
return data . inRoom ;
} ;
} ;
Messaging . isRoomOwner = function ( uid , roomId , callback ) {
Messaging . roomExists = async roomId => db . exists ( 'chat:room:' + roomId + ':uids' ) ;
async . waterfall ( [
function ( next ) {
Messaging . getUserCountInRoom = async roomId => db . sortedSetCard ( 'chat:room:' + roomId + ':uids' ) ;
db . getObjectField ( 'chat:room:' + roomId , 'owner' , next ) ;
} ,
Messaging . isRoomOwner = async ( uid , roomId ) => {
function ( owner , next ) {
const owner = await db . getObjectField ( 'chat:room:' + roomId , 'owner' ) ;
next ( null , parseInt ( uid , 10 ) === parseInt ( owner , 10 ) ) ;
return parseInt ( uid , 10 ) === parseInt ( owner , 10 ) ;
} ,
] , callback ) ;
} ;
} ;
Messaging . addUsersToRoom = function ( uid , uids , roomId , callback ) {
Messaging . addUsersToRoom = async function ( uid , uids , roomId ) {
async . waterfall ( [
const now = Date . now ( ) ;
function ( next ) {
const timestamps = uids . map ( ( ) => now ) ;
Messaging . isUserInRoom ( uid , roomId , next ) ;
const inRoom = await Messaging . isUserInRoom ( uid , roomId ) ;
} ,
if ( ! inRoom ) {
function ( inRoom , next ) {
throw new Error ( '[[error:cant-add-users-to-chat-room]]' ) ;
if ( ! inRoom ) {
}
return next ( new Error ( '[[error:cant-add-users-to-chat-room]]' ) ) ;
}
await db . sortedSetAdd ( 'chat:room:' + roomId + ':uids' , timestamps , uids ) ;
const now = Date . now ( ) ;
const [ userCount , roomData ] = await Promise . all ( [
const timestamps = uids . map ( ( ) => now ) ;
db . sortedSetCard ( 'chat:room:' + roomId + ':uids' ) ,
db . sortedSetAdd ( 'chat:room:' + roomId + ':uids' , timestamps , uids , next ) ;
db . getObject ( 'chat:room:' + roomId ) ,
} ,
] ) ;
function ( next ) {
async . parallel ( {
if ( ! roomData . hasOwnProperty ( 'groupChat' ) && userCount > 2 ) {
userCount : async . apply ( db . sortedSetCard , 'chat:room:' + roomId + ':uids' ) ,
await db . setObjectField ( 'chat:room:' + roomId , 'groupChat' , 1 ) ;
roomData : async . apply ( db . getObject , 'chat:room:' + roomId ) ,
}
} , next ) ;
} ,
await Promise . all ( uids . map ( uid => Messaging . addSystemMessage ( 'user-join' , uid , roomId ) ) ) ;
function ( results , next ) {
if ( ! results . roomData . hasOwnProperty ( 'groupChat' ) && results . userCount > 2 ) {
return db . setObjectField ( 'chat:room:' + roomId , 'groupChat' , 1 , next ) ;
}
next ( ) ;
} ,
] , callback ) ;
} ;
} ;
Messaging . removeUsersFromRoom = function ( uid , uids , roomId , callback ) {
Messaging . removeUsersFromRoom = async ( uid , uids , roomId ) => {
async . waterfall ( [
const [ isOwner , userCount ] = await Promise . all ( [
function ( next ) {
Messaging . isRoomOwner ( uid , roomId ) ,
async . parallel ( {
Messaging . getUserCountInRoom ( roomId ) ,
isOwner : async . apply ( Messaging . isRoomOwner , uid , roomId ) ,
] ) ;
userCount : async . apply ( Messaging . getUserCountInRoom , roomId ) ,
} , next ) ;
if ( ! isOwner ) {
} ,
throw new Error ( '[[error:cant-remove-users-from-chat-room]]' ) ;
function ( results , next ) {
}
if ( ! results . isOwner ) {
if ( userCount === 2 ) {
return next ( new Error ( '[[error:cant-remove-users-from-chat-room]]' ) ) ;
throw new Error ( '[[error:cant-remove-last-user]]' ) ;
}
}
if ( results . userCount === 2 ) {
return next ( new Error ( '[[error:cant-remove-last-user]]' ) ) ;
await Messaging . leaveRoom ( uids , roomId ) ;
}
Messaging . leaveRoom ( uids , roomId , next ) ;
} ,
] , callback ) ;
} ;
} ;
Messaging . leaveRoom = function ( uids , roomId , callback ) {
Messaging . leaveRoom = async ( uids , roomId ) => {
async . waterfall ( [
const keys = uids
function ( next ) {
. map ( function ( uid ) {
db . sortedSetRemove ( 'chat:room:' + roomId + ':uids' , uids , next ) ;
return 'uid:' + uid + ':chat:rooms' ;
} ,
} )
function ( next ) {
. concat ( uids . map ( function ( uid ) {
var keys = uids . map ( function ( uid ) {
return 'uid:' + uid + ':chat:rooms:unread' ;
return 'uid:' + uid + ':chat:rooms' ;
} ) ) ;
} ) ;
keys = keys . concat ( uids . map ( function ( uid ) {
await Promise . all ( [
return 'uid:' + uid + ':chat:rooms:unread' ;
db . sortedSetRemove ( 'chat:room:' + roomId + ':uids' , uids ) ,
} ) ) ;
db . sortedSetsRemove ( keys , roomId ) ,
db . sortedSetsRemove ( keys , roomId , next ) ;
] ) ;
} ,
function ( next ) {
await Promise . all ( uids . map ( uid => Messaging . addSystemMessage ( 'user-leave' , uid , roomId ) ) ) ;
updateOwner ( roomId , next ) ;
await updateOwner ( roomId ) ;
} ,
] , callback ) ;
} ;
} ;
Messaging . leaveRooms = function ( uid , roomIds , callback ) {
Messaging . leaveRooms = async ( uid , roomIds ) => {
async . waterfall ( [
const roomKeys = roomIds . map ( roomId => 'chat:room:' + roomId + ':uids' ) ;
function ( next ) {
await Promise . all ( [
const roomKeys = roomIds . map ( roomId => 'chat:room:' + roomId + ':uids' ) ;
db . sortedSetsRemove ( roomKeys , uid ) ,
db . sortedSetsRemove ( roomKeys , uid , next ) ;
db . sortedSetRemove ( [
} ,
'uid:' + uid + ':chat:rooms' ,
function ( next ) {
'uid:' + uid + ':chat:rooms:unread' ,
db . sortedSetRemove ( [
] , roomIds ) ,
'uid:' + uid + ':chat:rooms' ,
] ) ;
'uid:' + uid + ':chat:rooms:unread' ,
] , roomIds , next ) ;
await Promise . all (
} ,
roomIds . map ( roomId => updateOwner ( roomId ) )
function ( next ) {
. concat ( roomIds . map ( roomId => Messaging . addSystemMessage ( 'user-leave' , uid , roomId ) ) )
async . eachSeries ( roomIds , updateOwner , next ) ;
) ;
} ,
] , callback ) ;
} ;
} ;
function updateOwner ( roomId , callback ) {
async function updateOwner ( roomId ) {
async . waterfall ( [
const uids = await db . getSortedSetRange ( 'chat:room:' + roomId + ':uids' , 0 , 0 ) ;
function ( next ) {
const newOwner = uids [ 0 ] || 0 ;
db . getSortedSetRange ( 'chat:room:' + roomId + ':uids' , 0 , 0 , next ) ;
await db . setObjectField ( 'chat:room:' + roomId , 'owner' , newOwner ) ;
} ,
function ( uids , next ) {
var newOwner = uids [ 0 ] || 0 ;
db . setObjectField ( 'chat:room:' + roomId , 'owner' , newOwner , next ) ;
} ,
] , callback ) ;
}
}
Messaging . getUidsInRoom = function ( roomId , start , stop , callback ) {
Messaging . getUidsInRoom = async ( roomId , start , stop ) => db . getSortedSetRevRange ( 'chat:room:' + roomId + ':uids' , start , stop ) ;
db . getSortedSetRevRange ( 'chat:room:' + roomId + ':uids' , start , stop , callback ) ;
} ;
Messaging . getUsersInRoom = async ( roomId , start , stop ) => {
const uids = await Messaging . getUidsInRoom ( roomId , start , stop ) ;
const [ users , ownerId ] = await Promise . all ( [
user . getUsersFields ( uids , [ 'uid' , 'username' , 'picture' , 'status' ] ) ,
db . getObjectField ( 'chat:room:' + roomId , 'owner' ) ,
] ) ;
Messaging . getUsersInRoom = function ( roomId , start , stop , callback ) {
return users . map ( function ( user ) {
async . waterfall ( [
user . isOwner = parseInt ( user . uid , 10 ) === parseInt ( ownerId , 10 ) ;
function ( next ) {
return user ;
Messaging . getUidsInRoom ( roomId , start , stop , next ) ;
} ) ;
} ,
function ( uids , next ) {
user . getUsersFields ( uids , [ 'uid' , 'username' , 'picture' , 'status' ] , next ) ;
} ,
function ( users , next ) {
db . getObjectField ( 'chat:room:' + roomId , 'owner' , function ( err , ownerId ) {
next ( err , users . map ( function ( user ) {
user . isOwner = parseInt ( user . uid , 10 ) === parseInt ( ownerId , 10 ) ;
return user ;
} ) ) ;
} ) ;
} ,
] , callback ) ;
} ;
} ;
Messaging . renameRoom = function ( uid , roomId , newName , callback ) {
Messaging . renameRoom = async function ( uid , roomId , newName ) {
if ( ! newName ) {
if ( ! newName ) {
return callback ( new Error ( '[[error:invalid-name]]' ) ) ;
throw new Error ( '[[error:invalid-name]]' ) ;
}
}
newName = newName . trim ( ) ;
newName = newName . trim ( ) ;
if ( newName . length > 75 ) {
if ( newName . length > 75 ) {
return callback ( new Error ( '[[error:chat-room-name-too-long]]' ) ) ;
throw new Error ( '[[error:chat-room-name-too-long]]' ) ;
}
}
async . waterfall ( [
function ( next ) {
plugins . fireHook ( 'filter:chat.renameRoom' , {
uid : uid ,
roomId : roomId ,
newName : newName ,
} , next ) ;
} ,
function ( result , next ) {
Messaging . isRoomOwner ( uid , roomId , next ) ;
} ,
function ( isOwner , next ) {
if ( ! isOwner ) {
return next ( new Error ( '[[error:no-privileges]]' ) ) ;
}
db . setObjectField ( 'chat:room:' + roomId , 'roomName' , newName , next ) ;
} ,
async . apply ( plugins . fireHook , 'action:chat.renameRoom' , {
roomId : roomId ,
newName : newName ,
} ) ,
] , callback ) ;
} ;
Messaging . canReply = function ( roomId , uid , callback ) {
const payload = await plugins . fireHook ( 'filter:chat.renameRoom' , {
async . waterfall ( [
uid : uid ,
function ( next ) {
roomId : roomId ,
db . isSortedSetMember ( 'chat:room:' + roomId + ':uids' , uid , next ) ;
newName : newName ,
} ,
} ) ;
function ( inRoom , next ) {
const isOwner = await Messaging . isRoomOwner ( payload . uid , payload . roomId ) ;
plugins . fireHook ( 'filter:messaging.canReply' , { uid : uid , roomId : roomId , inRoom : inRoom , canReply : inRoom } , next ) ;
if ( ! isOwner ) {
} ,
throw new Error ( '[[error:no-privileges]]' ) ;
function ( data , next ) {
}
next ( null , data . canReply ) ;
} ,
await db . setObjectField ( 'chat:room:' + payload . roomId , 'roomName' , payload . newName ) ;
] , callback ) ;
await Messaging . addSystemMessage ( 'room-rename, ' + payload . newName . replace ( ',' , '%2C' ) , payload . uid , payload . roomId ) ;
plugins . fireHook ( 'action:chat.renameRoom' , {
roomId : payload . roomId ,
newName : payload . newName ,
} ) ;
} ;
} ;
Messaging . loadRoom = function ( uid , data , callback ) {
Messaging . canReply = async ( roomId , uid ) => {
async . waterfall ( [
const inRoom = db . isSortedSetMember ( 'chat:room:' + roomId + ':uids' , uid ) ;
function ( next ) {
const data = await plugins . fireHook ( 'filter:messaging.canReply' , { uid : uid , roomId : roomId , inRoom : inRoom , canReply : inRoom } ) ;
privileges . global . can ( 'chat' , uid , next ) ;
return data . canReply ;
} ,
} ;
function ( canChat , next ) {
if ( ! canChat ) {
return next ( new Error ( '[[error:no-privileges]]' ) ) ;
}
Messaging . isUserInRoom ( uid , data . roomId , next ) ;
Messaging . loadRoom = async ( uid , data ) => {
} ,
const canChat = await privileges . global . can ( 'chat' , uid ) ;
function ( inRoom , next ) {
if ( ! canChat ) {
if ( ! inRoom ) {
throw new Error ( '[[error:no-privileges]]' ) ;
return callback ( null , null ) ;
}
}
const inRoom = await Messaging . isUserInRoom ( uid , data . roomId ) ;
if ( ! inRoom ) {
return null ;
}
async . parallel ( {
const [ roomData , canReply , users , messages , isAdminOrGlobalMod ] = await Promise . all ( [
roomData : async . apply ( Messaging . getRoomData , data . roomId ) ,
Messaging . getRoomData ( data . roomId ) ,
canReply : async . apply ( Messaging . canReply , data . roomId , uid ) ,
Messaging . canReply ( data . roomId , uid ) ,
users : async . apply ( Messaging . getUsersInRoom , data . roomId , 0 , - 1 ) ,
Messaging . getUsersInRoom ( data . roomId , 0 , - 1 ) ,
messages : async . apply ( Messaging . getMessages , {
Messaging . getMessages ( {
callerUid : uid ,
callerUid : uid ,
uid : data . uid || uid ,
uid : data . uid || uid ,
roomId : data . roomId ,
roomId : data . roomId ,
isNew : false ,
isNew : false ,
} ) ,
} ) ,
isAdminOrGlobalMod : function ( next ) {
user . isAdminOrGlobalMod ( uid ) ,
user . isAdminOrGlobalMod ( uid , next ) ;
] ) ;
} ,
} , next ) ;
var room = roomData ;
} ,
room . messages = messages ;
function ( results , next ) {
room . isOwner = parseInt ( room . owner , 10 ) === parseInt ( uid , 10 ) ;
var room = results . roomData ;
room . users = users . filter ( function ( user ) {
room . messages = results . messages ;
return user && parseInt ( user . uid , 10 ) && parseInt ( user . uid , 10 ) !== uid ;
room . isOwner = parseInt ( room . owner , 10 ) === parseInt ( uid , 10 ) ;
} ) ;
room . users = results . users . filter ( function ( user ) {
room . canReply = canReply ;
return user && parseInt ( user . uid , 10 ) && parseInt ( user . uid , 10 ) !== uid ;
room . groupChat = room . hasOwnProperty ( 'groupChat' ) ? room . groupChat : users . length > 2 ;
} ) ;
room . usernames = Messaging . generateUsernames ( users , uid ) ;
room . canReply = results . canReply ;
room . maximumUsersInChatRoom = meta . config . maximumUsersInChatRoom ;
room . groupChat = room . hasOwnProperty ( 'groupChat' ) ? room . groupChat : results . users . length > 2 ;
room . maximumChatMessageLength = meta . config . maximumChatMessageLength ;
room . usernames = Messaging . generateUsernames ( results . users , uid ) ;
room . showUserInput = ! room . maximumUsersInChatRoom || room . maximumUsersInChatRoom > 2 ;
room . maximumUsersInChatRoom = meta . config . maximumUsersInChatRoom ;
room . isAdminOrGlobalMod = isAdminOrGlobalMod ;
room . maximumChatMessageLength = meta . config . maximumChatMessageLength ;
room . showUserInput = ! room . maximumUsersInChatRoom || room . maximumUsersInChatRoom > 2 ;
return room ;
room . isAdminOrGlobalMod = results . isAdminOrGlobalMod ;
next ( null , room ) ;
} ,
] , callback ) ;
} ;
} ;
} ;
} ;