2011-03-26 14:10:41 +01:00
|
|
|
/**
|
2011-05-30 16:53:11 +02:00
|
|
|
* The MessageHandler handles all Messages that comes from Socket.IO and controls the sessions
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
2011-08-11 16:26:41 +02:00
|
|
|
* Copyright 2009 Google Inc., 2011 Peter 'Pita' Martischka (Primary Technology Ltd)
|
2011-03-26 14:10:41 +01:00
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS-IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
|
|
|
|
2012-02-28 21:19:10 +01:00
|
|
|
|
2011-12-04 16:50:02 +01:00
|
|
|
var ERR = require("async-stacktrace");
|
2011-05-16 14:10:53 +02:00
|
|
|
var async = require("async");
|
2011-07-27 19:52:23 +02:00
|
|
|
var padManager = require("../db/PadManager");
|
2012-02-26 17:48:17 +01:00
|
|
|
var Changeset = require("ep_etherpad-lite/static/js/Changeset");
|
2012-03-18 09:05:46 +01:00
|
|
|
var AttributePool = require("ep_etherpad-lite/static/js/AttributePool");
|
2012-04-08 21:21:30 +02:00
|
|
|
var AttributeManager = require("ep_etherpad-lite/static/js/AttributeManager");
|
2011-07-27 19:52:23 +02:00
|
|
|
var authorManager = require("../db/AuthorManager");
|
|
|
|
var readOnlyManager = require("../db/ReadOnlyManager");
|
|
|
|
var settings = require('../utils/Settings');
|
2011-08-13 23:07:21 +02:00
|
|
|
var securityManager = require("../db/SecurityManager");
|
2012-03-01 19:00:58 +01:00
|
|
|
var plugins = require("ep_etherpad-lite/static/js/pluginfw/plugins.js");
|
2011-08-17 18:28:30 +02:00
|
|
|
var log4js = require('log4js');
|
|
|
|
var messageLogger = log4js.getLogger("message");
|
2012-04-07 01:05:25 +02:00
|
|
|
var _ = require('underscore');
|
2012-06-26 23:10:42 +02:00
|
|
|
var hooks = require("ep_etherpad-lite/static/js/pluginfw/hooks.js");
|
2011-03-26 14:10:41 +01:00
|
|
|
|
2011-03-26 22:29:33 +01:00
|
|
|
/**
|
|
|
|
* A associative array that saves which sessions belong to a pad
|
|
|
|
*/
|
2011-03-26 14:10:41 +01:00
|
|
|
var pad2sessions = {};
|
|
|
|
|
2011-03-26 22:29:33 +01:00
|
|
|
/**
|
2012-04-23 16:18:14 +02:00
|
|
|
* A associative array that saves informations about a session
|
2011-03-26 22:29:33 +01:00
|
|
|
* key = sessionId
|
2012-04-23 16:18:14 +02:00
|
|
|
* values = padId, readonlyPadId, readonly, author, rev
|
|
|
|
* padId = the real padId of the pad
|
|
|
|
* readonlyPadId = The readonly pad id of the pad
|
|
|
|
* readonly = Wether the client has only read access (true) or read/write access (false)
|
2011-03-26 22:29:33 +01:00
|
|
|
* rev = That last revision that was send to this client
|
|
|
|
* author = the author name of this session
|
|
|
|
*/
|
2011-03-26 14:10:41 +01:00
|
|
|
var sessioninfos = {};
|
|
|
|
|
2011-03-26 22:29:33 +01:00
|
|
|
/**
|
|
|
|
* Saves the Socket class we need to send and recieve data from the client
|
|
|
|
*/
|
2011-03-26 14:10:41 +01:00
|
|
|
var socketio;
|
|
|
|
|
2011-03-26 22:29:33 +01:00
|
|
|
/**
|
|
|
|
* This Method is called by server.js to tell the message handler on which socket it should send
|
|
|
|
* @param socket_io The Socket
|
|
|
|
*/
|
2011-03-26 14:10:41 +01:00
|
|
|
exports.setSocketIO = function(socket_io)
|
|
|
|
{
|
|
|
|
socketio=socket_io;
|
|
|
|
}
|
|
|
|
|
2011-03-26 22:29:33 +01:00
|
|
|
/**
|
|
|
|
* Handles the connection of a new user
|
|
|
|
* @param client the new client
|
|
|
|
*/
|
2011-03-26 14:10:41 +01:00
|
|
|
exports.handleConnect = function(client)
|
2011-08-17 18:28:30 +02:00
|
|
|
{
|
2012-04-23 16:18:14 +02:00
|
|
|
//Initalize sessioninfos for this new session
|
2011-06-23 15:08:18 +02:00
|
|
|
sessioninfos[client.id]={};
|
2011-03-26 14:10:41 +01:00
|
|
|
}
|
|
|
|
|
2011-08-16 21:02:30 +02:00
|
|
|
/**
|
|
|
|
* Kicks all sessions from a pad
|
|
|
|
* @param client the new client
|
|
|
|
*/
|
|
|
|
exports.kickSessionsFromPad = function(padID)
|
|
|
|
{
|
|
|
|
//skip if there is nobody on this pad
|
|
|
|
if(!pad2sessions[padID])
|
|
|
|
return;
|
|
|
|
|
|
|
|
//disconnect everyone from this pad
|
|
|
|
for(var i in pad2sessions[padID])
|
|
|
|
{
|
|
|
|
socketio.sockets.sockets[pad2sessions[padID][i]].json.send({disconnect:"deleted"});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-03-26 22:29:33 +01:00
|
|
|
/**
|
|
|
|
* Handles the disconnection of a user
|
|
|
|
* @param client the client that leaves
|
|
|
|
*/
|
2011-03-26 14:10:41 +01:00
|
|
|
exports.handleDisconnect = function(client)
|
2011-08-17 18:28:30 +02:00
|
|
|
{
|
2011-03-26 22:29:33 +01:00
|
|
|
//save the padname of this session
|
2012-04-23 16:18:14 +02:00
|
|
|
var sessionPad=sessioninfos[client.id].padId;
|
2011-03-26 14:10:41 +01:00
|
|
|
|
2011-05-23 21:11:57 +02:00
|
|
|
//if this connection was already etablished with a handshake, send a disconnect message to the others
|
2011-07-31 16:05:04 +02:00
|
|
|
if(sessioninfos[client.id] && sessioninfos[client.id].author)
|
2011-05-16 14:10:53 +02:00
|
|
|
{
|
2011-06-23 15:08:18 +02:00
|
|
|
var author = sessioninfos[client.id].author;
|
2011-05-23 21:11:57 +02:00
|
|
|
|
|
|
|
//get the author color out of the db
|
|
|
|
authorManager.getAuthorColorId(author, function(err, color)
|
2011-05-16 14:10:53 +02:00
|
|
|
{
|
2011-12-04 16:50:02 +01:00
|
|
|
ERR(err);
|
2011-05-23 21:11:57 +02:00
|
|
|
|
|
|
|
//prepare the notification for the other users on the pad, that this user left
|
|
|
|
var messageToTheOtherUsers = {
|
|
|
|
"type": "COLLABROOM",
|
|
|
|
"data": {
|
|
|
|
type: "USER_LEAVE",
|
|
|
|
userInfo: {
|
|
|
|
"ip": "127.0.0.1",
|
|
|
|
"colorId": color,
|
|
|
|
"userAgent": "Anonymous",
|
|
|
|
"userId": author
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
//Go trough all user that are still on the pad, and send them the USER_LEAVE message
|
|
|
|
for(i in pad2sessions[sessionPad])
|
2011-05-16 14:10:53 +02:00
|
|
|
{
|
2012-03-14 01:32:16 +01:00
|
|
|
var socket = socketio.sockets.sockets[pad2sessions[sessionPad][i]];
|
|
|
|
if(socket !== undefined){
|
|
|
|
socket.json.send(messageToTheOtherUsers);
|
|
|
|
}
|
|
|
|
|
2011-03-29 00:07:35 +02:00
|
|
|
}
|
2011-05-23 21:11:57 +02:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
//Go trough all sessions of this pad, search and destroy the entry of this client
|
|
|
|
for(i in pad2sessions[sessionPad])
|
|
|
|
{
|
2011-06-23 15:08:18 +02:00
|
|
|
if(pad2sessions[sessionPad][i] == client.id)
|
2011-03-26 14:10:41 +01:00
|
|
|
{
|
2011-11-19 21:07:39 +01:00
|
|
|
pad2sessions[sessionPad].splice(i, 1);
|
2011-05-23 21:11:57 +02:00
|
|
|
break;
|
2011-03-26 14:10:41 +01:00
|
|
|
}
|
2011-05-23 21:11:57 +02:00
|
|
|
}
|
|
|
|
|
2012-04-23 16:18:14 +02:00
|
|
|
//Delete the sessioninfos entrys of this session
|
2011-06-23 15:08:18 +02:00
|
|
|
delete sessioninfos[client.id];
|
2011-03-26 14:10:41 +01:00
|
|
|
}
|
|
|
|
|
2011-03-26 22:29:33 +01:00
|
|
|
/**
|
|
|
|
* Handles a message from a user
|
|
|
|
* @param client the client that send this message
|
|
|
|
* @param message the message from the client
|
|
|
|
*/
|
2011-03-26 14:10:41 +01:00
|
|
|
exports.handleMessage = function(client, message)
|
2011-03-26 22:29:33 +01:00
|
|
|
{
|
2012-06-26 23:10:42 +02:00
|
|
|
_.map(hooks.callAll( "handleMessage", { client: client, message: message }), function ( newmessage ) {
|
|
|
|
if ( newmessage || newmessage === null ) {
|
|
|
|
message = newmessage;
|
|
|
|
}
|
|
|
|
});
|
2011-03-26 14:10:41 +01:00
|
|
|
if(message == null)
|
|
|
|
{
|
2011-08-20 19:56:38 +02:00
|
|
|
messageLogger.warn("Message is null!");
|
|
|
|
return;
|
2011-03-26 14:10:41 +01:00
|
|
|
}
|
|
|
|
if(!message.type)
|
|
|
|
{
|
2011-08-20 19:56:38 +02:00
|
|
|
messageLogger.warn("Message has no type attribute!");
|
|
|
|
return;
|
2011-03-26 14:10:41 +01:00
|
|
|
}
|
2012-06-27 00:28:18 +02:00
|
|
|
|
|
|
|
var finalHandler = function () {
|
|
|
|
//Check what type of message we get and delegate to the other methodes
|
|
|
|
if(message.type == "CLIENT_READY") {
|
|
|
|
handleClientReady(client, message);
|
|
|
|
} else if(message.type == "CHANGESET_REQ") {
|
|
|
|
handleChangesetRequest(client, message);
|
|
|
|
} else if(message.type == "COLLABROOM") {
|
|
|
|
if (sessioninfos[client.id].readonly) {
|
|
|
|
messageLogger.warn("Dropped message, COLLABROOM for readonly pad");
|
|
|
|
} else if (message.data.type == "USER_CHANGES") {
|
|
|
|
handleUserChanges(client, message);
|
|
|
|
} else if (message.data.type == "USERINFO_UPDATE") {
|
|
|
|
handleUserInfoUpdate(client, message);
|
|
|
|
} else if (message.data.type == "CHAT_MESSAGE") {
|
|
|
|
handleChatMessage(client, message);
|
|
|
|
} else if (message.data.type == "SAVE_REVISION") {
|
|
|
|
handleSaveRevisionMessage(client, message);
|
|
|
|
} else if (message.data.type == "CLIENT_MESSAGE" &&
|
|
|
|
message.data.payload.type == "suggestUserName") {
|
|
|
|
handleSuggestUserName(client, message);
|
|
|
|
} else {
|
|
|
|
messageLogger.warn("Dropped message, unknown COLLABROOM Data Type " + message.data.type);
|
|
|
|
}
|
2012-04-23 16:18:14 +02:00
|
|
|
} else {
|
2012-06-27 00:28:18 +02:00
|
|
|
messageLogger.warn("Dropped message, unknown Message Type " + message.type);
|
2012-04-23 16:18:14 +02:00
|
|
|
}
|
2012-06-27 00:28:18 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
if (message && message.padId) {
|
|
|
|
async.series([
|
|
|
|
//check permissions
|
|
|
|
function(callback)
|
|
|
|
{
|
|
|
|
// Note: message.sessionID is an entirely different kind of
|
|
|
|
// session from the sessions we use here! Beware! FIXME: Call
|
|
|
|
// our "sessions" "connections".
|
|
|
|
// FIXME: Use a hook instead
|
|
|
|
// FIXME: Allow to override readwrite access with readonly
|
|
|
|
securityManager.checkAccess(message.padId, message.sessionID, message.token, message.password, function(err, statusObject)
|
|
|
|
{
|
|
|
|
if(ERR(err, callback)) return;
|
|
|
|
|
|
|
|
//access was granted
|
|
|
|
if(statusObject.accessStatus == "grant")
|
|
|
|
{
|
|
|
|
callback();
|
|
|
|
}
|
|
|
|
//no access, send the client a message that tell him why
|
|
|
|
else
|
|
|
|
{
|
|
|
|
client.json.send({accessStatus: statusObject.accessStatus})
|
|
|
|
}
|
|
|
|
});
|
|
|
|
},
|
|
|
|
finalHandler
|
|
|
|
]);
|
2012-06-27 01:07:08 +02:00
|
|
|
} else {
|
|
|
|
finalHandler();
|
2011-03-26 14:10:41 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-04-23 12:52:30 +02:00
|
|
|
|
2012-02-29 20:40:14 +01:00
|
|
|
/**
|
|
|
|
* Handles a save revision message
|
|
|
|
* @param client the client that send this message
|
|
|
|
* @param message the message from the client
|
|
|
|
*/
|
|
|
|
function handleSaveRevisionMessage(client, message){
|
2012-04-23 16:18:14 +02:00
|
|
|
var padId = sessioninfos[client.id].padId;
|
2012-02-29 20:40:14 +01:00
|
|
|
var userId = sessioninfos[client.id].author;
|
|
|
|
|
|
|
|
padManager.getPad(padId, function(err, pad)
|
|
|
|
{
|
|
|
|
if(ERR(err)) return;
|
|
|
|
|
|
|
|
pad.addSavedRevision(pad.head, userId);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2011-07-14 17:15:38 +02:00
|
|
|
/**
|
|
|
|
* Handles a Chat Message
|
|
|
|
* @param client the client that send this message
|
|
|
|
* @param message the message from the client
|
|
|
|
*/
|
|
|
|
function handleChatMessage(client, message)
|
|
|
|
{
|
|
|
|
var time = new Date().getTime();
|
|
|
|
var userId = sessioninfos[client.id].author;
|
|
|
|
var text = message.data.text;
|
2012-04-23 16:18:14 +02:00
|
|
|
var padId = sessioninfos[client.id].padId;
|
2011-07-14 17:15:38 +02:00
|
|
|
|
|
|
|
var pad;
|
|
|
|
var userName;
|
|
|
|
|
|
|
|
async.series([
|
|
|
|
//get the pad
|
|
|
|
function(callback)
|
|
|
|
{
|
|
|
|
padManager.getPad(padId, function(err, _pad)
|
|
|
|
{
|
2011-12-04 16:50:02 +01:00
|
|
|
if(ERR(err, callback)) return;
|
2011-07-14 17:15:38 +02:00
|
|
|
pad = _pad;
|
2011-12-04 16:50:02 +01:00
|
|
|
callback();
|
2011-07-14 17:15:38 +02:00
|
|
|
});
|
|
|
|
},
|
|
|
|
function(callback)
|
|
|
|
{
|
|
|
|
authorManager.getAuthorName(userId, function(err, _userName)
|
|
|
|
{
|
2011-12-04 16:50:02 +01:00
|
|
|
if(ERR(err, callback)) return;
|
2011-07-14 17:15:38 +02:00
|
|
|
userName = _userName;
|
2011-12-04 16:50:02 +01:00
|
|
|
callback();
|
2011-07-14 17:15:38 +02:00
|
|
|
});
|
|
|
|
},
|
|
|
|
//save the chat message and broadcast it
|
|
|
|
function(callback)
|
|
|
|
{
|
|
|
|
//save the chat message
|
|
|
|
pad.appendChatMessage(text, userId, time);
|
|
|
|
|
|
|
|
var msg = {
|
|
|
|
type: "COLLABROOM",
|
|
|
|
data: {
|
|
|
|
type: "CHAT_MESSAGE",
|
|
|
|
userId: userId,
|
|
|
|
userName: userName,
|
|
|
|
time: time,
|
|
|
|
text: text
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
//broadcast the chat message to everyone on the pad
|
|
|
|
for(var i in pad2sessions[padId])
|
|
|
|
{
|
|
|
|
socketio.sockets.sockets[pad2sessions[padId][i]].json.send(msg);
|
|
|
|
}
|
|
|
|
|
|
|
|
callback();
|
|
|
|
}
|
|
|
|
], function(err)
|
|
|
|
{
|
2011-12-04 16:50:02 +01:00
|
|
|
ERR(err);
|
2011-07-14 17:15:38 +02:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-06-30 15:19:30 +02:00
|
|
|
/**
|
|
|
|
* Handles a handleSuggestUserName, that means a user have suggest a userName for a other user
|
|
|
|
* @param client the client that send this message
|
|
|
|
* @param message the message from the client
|
|
|
|
*/
|
|
|
|
function handleSuggestUserName(client, message)
|
|
|
|
{
|
|
|
|
//check if all ok
|
|
|
|
if(message.data.payload.newName == null)
|
|
|
|
{
|
2011-11-09 17:19:00 +01:00
|
|
|
messageLogger.warn("Dropped message, suggestUserName Message has no newName!");
|
2011-08-17 18:28:30 +02:00
|
|
|
return;
|
2011-06-30 15:19:30 +02:00
|
|
|
}
|
|
|
|
if(message.data.payload.unnamedId == null)
|
|
|
|
{
|
2011-11-09 17:19:00 +01:00
|
|
|
messageLogger.warn("Dropped message, suggestUserName Message has no unnamedId!");
|
2011-08-17 18:28:30 +02:00
|
|
|
return;
|
2011-06-30 15:19:30 +02:00
|
|
|
}
|
|
|
|
|
2012-04-23 16:18:14 +02:00
|
|
|
var padId = sessioninfos[client.id].padId;
|
2011-06-30 15:19:30 +02:00
|
|
|
|
|
|
|
//search the author and send him this message
|
|
|
|
for(var i in pad2sessions[padId])
|
|
|
|
{
|
|
|
|
if(sessioninfos[pad2sessions[padId][i]].author == message.data.payload.unnamedId)
|
|
|
|
{
|
2011-07-06 22:43:28 +02:00
|
|
|
socketio.sockets.sockets[pad2sessions[padId][i]].send(message);
|
2011-06-30 15:19:30 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-03-26 22:29:33 +01:00
|
|
|
/**
|
|
|
|
* Handles a USERINFO_UPDATE, that means that a user have changed his color or name. Anyway, we get both informations
|
|
|
|
* @param client the client that send this message
|
|
|
|
* @param message the message from the client
|
|
|
|
*/
|
2011-03-26 14:10:41 +01:00
|
|
|
function handleUserInfoUpdate(client, message)
|
|
|
|
{
|
2011-03-26 22:29:33 +01:00
|
|
|
//check if all ok
|
2011-03-26 14:10:41 +01:00
|
|
|
if(message.data.userInfo.colorId == null)
|
|
|
|
{
|
2011-11-09 17:19:00 +01:00
|
|
|
messageLogger.warn("Dropped message, USERINFO_UPDATE Message has no colorId!");
|
2011-08-17 18:28:30 +02:00
|
|
|
return;
|
2011-03-26 14:10:41 +01:00
|
|
|
}
|
|
|
|
|
2011-03-26 22:29:33 +01:00
|
|
|
//Find out the author name of this session
|
2011-06-23 15:08:18 +02:00
|
|
|
var author = sessioninfos[client.id].author;
|
2011-03-26 14:10:41 +01:00
|
|
|
|
2011-03-26 22:29:33 +01:00
|
|
|
//Tell the authorManager about the new attributes
|
2011-03-26 14:10:41 +01:00
|
|
|
authorManager.setAuthorColorId(author, message.data.userInfo.colorId);
|
|
|
|
authorManager.setAuthorName(author, message.data.userInfo.name);
|
2011-03-27 14:13:46 +02:00
|
|
|
|
2012-04-23 16:18:14 +02:00
|
|
|
var padId = sessioninfos[client.id].padId;
|
2011-03-27 14:13:46 +02:00
|
|
|
|
|
|
|
//set a null name, when there is no name set. cause the client wants it null
|
|
|
|
if(message.data.userInfo.name == null)
|
|
|
|
{
|
|
|
|
message.data.userInfo.name = null;
|
|
|
|
}
|
|
|
|
|
|
|
|
//The Client don't know about a USERINFO_UPDATE, it can handle only new user_newinfo, so change the message type
|
|
|
|
message.data.type = "USER_NEWINFO";
|
|
|
|
|
|
|
|
//Send the other clients on the pad the update message
|
2011-06-30 15:19:30 +02:00
|
|
|
for(var i in pad2sessions[padId])
|
2011-03-27 14:13:46 +02:00
|
|
|
{
|
2011-06-23 15:08:18 +02:00
|
|
|
if(pad2sessions[padId][i] != client.id)
|
2011-03-27 14:13:46 +02:00
|
|
|
{
|
2011-07-06 19:34:39 +02:00
|
|
|
socketio.sockets.sockets[pad2sessions[padId][i]].json.send(message);
|
2011-03-27 14:13:46 +02:00
|
|
|
}
|
|
|
|
}
|
2011-03-26 14:10:41 +01:00
|
|
|
}
|
|
|
|
|
2011-03-26 22:29:33 +01:00
|
|
|
/**
|
|
|
|
* Handles a USERINFO_UPDATE, that means that a user have changed his color or name. Anyway, we get both informations
|
|
|
|
* This Method is nearly 90% copied out of the Etherpad Source Code. So I can't tell you what happens here exactly
|
|
|
|
* Look at https://github.com/ether/pad/blob/master/etherpad/src/etherpad/collab/collab_server.js in the function applyUserChanges()
|
|
|
|
* @param client the client that send this message
|
|
|
|
* @param message the message from the client
|
|
|
|
*/
|
2011-03-26 14:10:41 +01:00
|
|
|
function handleUserChanges(client, message)
|
|
|
|
{
|
2011-03-26 22:29:33 +01:00
|
|
|
//check if all ok
|
2011-03-26 14:10:41 +01:00
|
|
|
if(message.data.baseRev == null)
|
|
|
|
{
|
2011-11-09 17:19:00 +01:00
|
|
|
messageLogger.warn("Dropped message, USER_CHANGES Message has no baseRev!");
|
2011-08-17 18:28:30 +02:00
|
|
|
return;
|
2011-03-26 14:10:41 +01:00
|
|
|
}
|
|
|
|
if(message.data.apool == null)
|
|
|
|
{
|
2011-11-09 17:19:00 +01:00
|
|
|
messageLogger.warn("Dropped message, USER_CHANGES Message has no apool!");
|
2011-08-17 18:28:30 +02:00
|
|
|
return;
|
2011-03-26 14:10:41 +01:00
|
|
|
}
|
|
|
|
if(message.data.changeset == null)
|
|
|
|
{
|
2011-11-09 17:19:00 +01:00
|
|
|
messageLogger.warn("Dropped message, USER_CHANGES Message has no changeset!");
|
2011-08-17 18:28:30 +02:00
|
|
|
return;
|
2011-03-26 14:10:41 +01:00
|
|
|
}
|
2012-04-23 16:18:14 +02:00
|
|
|
|
2011-03-26 22:29:33 +01:00
|
|
|
//get all Vars we need
|
2011-03-26 14:10:41 +01:00
|
|
|
var baseRev = message.data.baseRev;
|
2012-03-18 09:05:46 +01:00
|
|
|
var wireApool = (new AttributePool()).fromJsonable(message.data.apool);
|
2011-03-26 14:10:41 +01:00
|
|
|
var changeset = message.data.changeset;
|
2011-05-17 17:33:54 +02:00
|
|
|
|
|
|
|
var r, apool, pad;
|
2011-03-26 14:10:41 +01:00
|
|
|
|
2011-05-17 17:33:54 +02:00
|
|
|
async.series([
|
|
|
|
//get the pad
|
|
|
|
function(callback)
|
2011-03-26 14:10:41 +01:00
|
|
|
{
|
2012-04-23 16:18:14 +02:00
|
|
|
padManager.getPad(sessioninfos[client.id].padId, function(err, value)
|
2011-03-26 14:10:41 +01:00
|
|
|
{
|
2011-12-04 16:50:02 +01:00
|
|
|
if(ERR(err, callback)) return;
|
2011-05-17 17:33:54 +02:00
|
|
|
pad = value;
|
2011-12-04 16:50:02 +01:00
|
|
|
callback();
|
2011-05-17 17:33:54 +02:00
|
|
|
});
|
|
|
|
},
|
|
|
|
//create the changeset
|
|
|
|
function(callback)
|
|
|
|
{
|
|
|
|
//ex. _checkChangesetAndPool
|
|
|
|
|
|
|
|
//Copied from Etherpad, don't know what it does exactly
|
2011-08-20 19:51:43 +02:00
|
|
|
try
|
|
|
|
{
|
|
|
|
//this looks like a changeset check, it throws errors sometimes
|
|
|
|
Changeset.checkRep(changeset);
|
|
|
|
|
|
|
|
Changeset.eachAttribNumber(changeset, function(n) {
|
|
|
|
if (! wireApool.getAttrib(n)) {
|
|
|
|
throw "Attribute pool is missing attribute "+n+" for changeset "+changeset;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
//there is an error in this changeset, so just refuse it
|
|
|
|
catch(e)
|
|
|
|
{
|
|
|
|
console.warn("Can't apply USER_CHANGES "+changeset+", cause it faild checkRep");
|
|
|
|
client.json.send({disconnect:"badChangeset"});
|
|
|
|
return;
|
|
|
|
}
|
2011-05-17 17:33:54 +02:00
|
|
|
|
|
|
|
//ex. adoptChangesetAttribs
|
|
|
|
|
|
|
|
//Afaik, it copies the new attributes from the changeset, to the global Attribute Pool
|
|
|
|
changeset = Changeset.moveOpsToNewPool(changeset, wireApool, pad.pool);
|
|
|
|
|
|
|
|
//ex. applyUserChanges
|
|
|
|
apool = pad.pool;
|
|
|
|
r = baseRev;
|
|
|
|
|
|
|
|
//https://github.com/caolan/async#whilst
|
|
|
|
async.whilst(
|
|
|
|
function() { return r < pad.getHeadRevisionNumber(); },
|
|
|
|
function(callback)
|
|
|
|
{
|
|
|
|
r++;
|
|
|
|
|
|
|
|
pad.getRevisionChangeset(r, function(err, c)
|
|
|
|
{
|
2011-12-04 16:50:02 +01:00
|
|
|
if(ERR(err, callback)) return;
|
|
|
|
|
|
|
|
changeset = Changeset.follow(c, changeset, false, apool);
|
|
|
|
callback(null);
|
2011-05-17 17:33:54 +02:00
|
|
|
});
|
|
|
|
},
|
|
|
|
//use the callback of the series function
|
|
|
|
callback
|
|
|
|
);
|
|
|
|
},
|
|
|
|
//do correction changesets, and send it to all users
|
|
|
|
function (callback)
|
|
|
|
{
|
|
|
|
var prevText = pad.text();
|
2011-07-30 14:33:16 +02:00
|
|
|
|
|
|
|
if (Changeset.oldLen(changeset) != prevText.length)
|
|
|
|
{
|
2011-07-31 19:25:51 +02:00
|
|
|
console.warn("Can't apply USER_CHANGES "+changeset+" with oldLen " + Changeset.oldLen(changeset) + " to document of length " + prevText.length);
|
2011-07-30 14:33:16 +02:00
|
|
|
client.json.send({disconnect:"badChangeset"});
|
|
|
|
callback();
|
|
|
|
return;
|
2011-03-26 14:10:41 +01:00
|
|
|
}
|
2011-05-17 17:33:54 +02:00
|
|
|
|
2011-06-23 15:08:18 +02:00
|
|
|
var thisAuthor = sessioninfos[client.id].author;
|
2011-05-17 17:33:54 +02:00
|
|
|
|
|
|
|
pad.appendRevision(changeset, thisAuthor);
|
|
|
|
|
|
|
|
var correctionChangeset = _correctMarkersInPad(pad.atext, pad.pool);
|
|
|
|
if (correctionChangeset) {
|
|
|
|
pad.appendRevision(correctionChangeset);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (pad.text().lastIndexOf("\n\n") != pad.text().length-2) {
|
|
|
|
var nlChangeset = Changeset.makeSplice(pad.text(), pad.text().length-1, 0, "\n");
|
|
|
|
pad.appendRevision(nlChangeset);
|
2011-03-26 14:10:41 +01:00
|
|
|
}
|
2011-05-17 17:33:54 +02:00
|
|
|
|
2011-07-21 21:13:58 +02:00
|
|
|
exports.updatePadClients(pad, callback);
|
|
|
|
}
|
|
|
|
], function(err)
|
|
|
|
{
|
2011-12-04 16:50:02 +01:00
|
|
|
ERR(err);
|
2011-07-21 21:13:58 +02:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
exports.updatePadClients = function(pad, callback)
|
|
|
|
{
|
2011-08-19 22:35:31 +02:00
|
|
|
//skip this step if noone is on this pad
|
|
|
|
if(!pad2sessions[pad.id])
|
|
|
|
{
|
|
|
|
callback();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2011-07-21 21:13:58 +02:00
|
|
|
//go trough all sessions on this pad
|
|
|
|
async.forEach(pad2sessions[pad.id], function(session, callback)
|
|
|
|
{
|
2012-04-23 14:20:17 +02:00
|
|
|
|
2011-07-21 21:13:58 +02:00
|
|
|
//https://github.com/caolan/async#whilst
|
|
|
|
//send them all new changesets
|
|
|
|
async.whilst(
|
2012-04-23 14:20:17 +02:00
|
|
|
function (){ return sessioninfos[session].rev < pad.getHeadRevisionNumber()},
|
2011-07-21 21:13:58 +02:00
|
|
|
function(callback)
|
2012-04-23 14:20:17 +02:00
|
|
|
{
|
|
|
|
var author, revChangeset, currentTime;
|
|
|
|
var r = sessioninfos[session].rev + 1;
|
2011-07-21 21:13:58 +02:00
|
|
|
|
|
|
|
async.parallel([
|
|
|
|
function (callback)
|
2011-05-17 17:33:54 +02:00
|
|
|
{
|
2011-07-21 21:13:58 +02:00
|
|
|
pad.getRevisionAuthor(r, function(err, value)
|
2011-05-17 17:33:54 +02:00
|
|
|
{
|
2011-12-04 16:50:02 +01:00
|
|
|
if(ERR(err, callback)) return;
|
2011-07-21 21:13:58 +02:00
|
|
|
author = value;
|
2011-12-04 16:50:02 +01:00
|
|
|
callback();
|
2011-05-17 17:33:54 +02:00
|
|
|
});
|
|
|
|
},
|
2011-07-21 21:13:58 +02:00
|
|
|
function (callback)
|
|
|
|
{
|
|
|
|
pad.getRevisionChangeset(r, function(err, value)
|
|
|
|
{
|
2011-12-04 16:50:02 +01:00
|
|
|
if(ERR(err, callback)) return;
|
2011-07-21 21:13:58 +02:00
|
|
|
revChangeset = value;
|
2011-12-04 16:50:02 +01:00
|
|
|
callback();
|
2011-07-21 21:13:58 +02:00
|
|
|
});
|
2012-04-23 14:20:17 +02:00
|
|
|
},
|
|
|
|
function (callback)
|
|
|
|
{
|
|
|
|
pad.getRevisionDate(r, function(err, date)
|
|
|
|
{
|
|
|
|
if(ERR(err, callback)) return;
|
|
|
|
currentTime = date;
|
|
|
|
callback();
|
|
|
|
});
|
2011-07-21 21:13:58 +02:00
|
|
|
}
|
|
|
|
], function(err)
|
|
|
|
{
|
2011-12-04 16:50:02 +01:00
|
|
|
if(ERR(err, callback)) return;
|
2012-01-19 13:02:46 +01:00
|
|
|
// next if session has not been deleted
|
|
|
|
if(sessioninfos[session] == null)
|
|
|
|
{
|
|
|
|
callback(null);
|
|
|
|
return;
|
|
|
|
}
|
2011-07-21 21:13:58 +02:00
|
|
|
if(author == sessioninfos[session].author)
|
|
|
|
{
|
|
|
|
socketio.sockets.sockets[session].json.send({"type":"COLLABROOM","data":{type:"ACCEPT_COMMIT", newRev:r}});
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
var forWire = Changeset.prepareForWire(revChangeset, pad.pool);
|
2012-04-23 14:20:17 +02:00
|
|
|
var wireMsg = {"type":"COLLABROOM",
|
|
|
|
"data":{type:"NEW_CHANGES",
|
|
|
|
newRev:r,
|
|
|
|
changeset: forWire.translated,
|
|
|
|
apool: forWire.pool,
|
|
|
|
author: author,
|
|
|
|
currentTime: currentTime,
|
|
|
|
timeDelta: currentTime - sessioninfos[session].time
|
|
|
|
}};
|
2011-07-21 21:13:58 +02:00
|
|
|
|
|
|
|
socketio.sockets.sockets[session].json.send(wireMsg);
|
|
|
|
}
|
2012-04-23 14:20:17 +02:00
|
|
|
|
|
|
|
if(sessioninfos[session] != null)
|
|
|
|
{
|
|
|
|
sessioninfos[session].time = currentTime;
|
|
|
|
sessioninfos[session].rev = r;
|
|
|
|
}
|
2011-05-17 17:33:54 +02:00
|
|
|
|
2011-07-21 21:13:58 +02:00
|
|
|
callback(null);
|
|
|
|
});
|
|
|
|
},
|
|
|
|
callback
|
|
|
|
);
|
|
|
|
},callback);
|
2011-03-26 14:10:41 +01:00
|
|
|
}
|
|
|
|
|
2011-03-26 22:29:33 +01:00
|
|
|
/**
|
|
|
|
* Copied from the Etherpad Source Code. Don't know what this methode does excatly...
|
|
|
|
*/
|
2011-03-26 14:10:41 +01:00
|
|
|
function _correctMarkersInPad(atext, apool) {
|
|
|
|
var text = atext.text;
|
|
|
|
|
|
|
|
// collect char positions of line markers (e.g. bullets) in new atext
|
|
|
|
// that aren't at the start of a line
|
|
|
|
var badMarkers = [];
|
|
|
|
var iter = Changeset.opIterator(atext.attribs);
|
|
|
|
var offset = 0;
|
|
|
|
while (iter.hasNext()) {
|
|
|
|
var op = iter.next();
|
2012-04-07 01:05:25 +02:00
|
|
|
|
|
|
|
var hasMarker = _.find(AttributeManager.lineAttributes, function(attribute){
|
|
|
|
return Changeset.opAttributeValue(op, attribute, apool);
|
|
|
|
}) !== undefined;
|
|
|
|
|
|
|
|
if (hasMarker) {
|
2011-03-26 14:10:41 +01:00
|
|
|
for(var i=0;i<op.chars;i++) {
|
|
|
|
if (offset > 0 && text.charAt(offset-1) != '\n') {
|
|
|
|
badMarkers.push(offset);
|
|
|
|
}
|
|
|
|
offset++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
offset += op.chars;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (badMarkers.length == 0) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
// create changeset that removes these bad markers
|
|
|
|
offset = 0;
|
|
|
|
var builder = Changeset.builder(text.length);
|
|
|
|
badMarkers.forEach(function(pos) {
|
|
|
|
builder.keepText(text.substring(offset, pos));
|
|
|
|
builder.remove(1);
|
|
|
|
offset = pos+1;
|
|
|
|
});
|
|
|
|
return builder.toString();
|
|
|
|
}
|
|
|
|
|
2011-03-26 22:29:33 +01:00
|
|
|
/**
|
|
|
|
* Handles a CLIENT_READY. A CLIENT_READY is the first message from the client to the server. The Client sends his token
|
|
|
|
* and the pad it wants to enter. The Server answers with the inital values (clientVars) of the pad
|
|
|
|
* @param client the client that send this message
|
|
|
|
* @param message the message from the client
|
|
|
|
*/
|
2011-03-26 14:10:41 +01:00
|
|
|
function handleClientReady(client, message)
|
|
|
|
{
|
2011-03-26 22:29:33 +01:00
|
|
|
//check if all ok
|
2011-03-26 14:10:41 +01:00
|
|
|
if(!message.token)
|
|
|
|
{
|
2011-11-19 20:21:23 +01:00
|
|
|
messageLogger.warn("Dropped message, CLIENT_READY Message has no token!");
|
2011-08-17 18:28:30 +02:00
|
|
|
return;
|
2011-03-26 14:10:41 +01:00
|
|
|
}
|
|
|
|
if(!message.padId)
|
|
|
|
{
|
2011-11-19 20:21:23 +01:00
|
|
|
messageLogger.warn("Dropped message, CLIENT_READY Message has no padId!");
|
2011-08-17 18:28:30 +02:00
|
|
|
return;
|
2011-03-26 14:10:41 +01:00
|
|
|
}
|
|
|
|
if(!message.protocolVersion)
|
|
|
|
{
|
2011-11-19 20:21:23 +01:00
|
|
|
messageLogger.warn("Dropped message, CLIENT_READY Message has no protocolVersion!");
|
2011-08-17 18:28:30 +02:00
|
|
|
return;
|
2011-03-26 14:10:41 +01:00
|
|
|
}
|
2011-06-20 12:44:04 +02:00
|
|
|
if(message.protocolVersion != 2)
|
2011-03-26 14:10:41 +01:00
|
|
|
{
|
2011-11-19 20:21:23 +01:00
|
|
|
messageLogger.warn("Dropped message, CLIENT_READY Message has a unknown protocolVersion '" + message.protocolVersion + "'!");
|
2011-08-17 18:28:30 +02:00
|
|
|
return;
|
2011-03-26 14:10:41 +01:00
|
|
|
}
|
|
|
|
|
2011-05-16 14:10:53 +02:00
|
|
|
var author;
|
|
|
|
var authorName;
|
|
|
|
var authorColorId;
|
2011-05-17 17:33:54 +02:00
|
|
|
var pad;
|
2011-06-30 13:03:20 +02:00
|
|
|
var historicalAuthorData = {};
|
2012-04-23 12:52:30 +02:00
|
|
|
var currentTime;
|
2011-07-14 17:15:38 +02:00
|
|
|
var chatMessages;
|
2012-04-23 16:18:14 +02:00
|
|
|
var padIds;
|
2011-05-16 14:10:53 +02:00
|
|
|
|
2011-05-17 17:33:54 +02:00
|
|
|
async.series([
|
2012-04-23 16:18:14 +02:00
|
|
|
// Get ro/rw id:s
|
|
|
|
function (callback) {
|
|
|
|
readOnlyManager.getIds(message.padId, function(err, value) {
|
|
|
|
if(ERR(err, callback)) return;
|
|
|
|
padIds = value;
|
|
|
|
callback();
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
2011-08-13 23:07:21 +02:00
|
|
|
//check permissions
|
2011-05-16 14:10:53 +02:00
|
|
|
function(callback)
|
2011-03-27 13:49:04 +02:00
|
|
|
{
|
2012-04-23 16:18:14 +02:00
|
|
|
// Note: message.sessionID is an entierly different kind of
|
|
|
|
// session from the sessions we use here! Beware! FIXME: Call
|
|
|
|
// our "sessions" "connections".
|
|
|
|
// FIXME: Use a hook instead
|
|
|
|
// FIXME: Allow to override readwrite access with readonly
|
|
|
|
securityManager.checkAccess (padIds.padId, message.sessionID, message.token, message.password, function(err, statusObject)
|
2011-03-27 13:49:04 +02:00
|
|
|
{
|
2011-12-04 16:50:02 +01:00
|
|
|
if(ERR(err, callback)) return;
|
2011-05-16 14:10:53 +02:00
|
|
|
|
2011-08-13 23:07:21 +02:00
|
|
|
//access was granted
|
|
|
|
if(statusObject.accessStatus == "grant")
|
|
|
|
{
|
|
|
|
author = statusObject.authorID;
|
|
|
|
callback();
|
|
|
|
}
|
|
|
|
//no access, send the client a message that tell him why
|
|
|
|
else
|
|
|
|
{
|
2011-08-15 19:26:20 +02:00
|
|
|
client.json.send({accessStatus: statusObject.accessStatus})
|
2011-08-13 23:07:21 +02:00
|
|
|
}
|
|
|
|
});
|
|
|
|
},
|
|
|
|
//get all authordata of this new user
|
|
|
|
function(callback)
|
|
|
|
{
|
|
|
|
async.parallel([
|
|
|
|
//get colorId
|
|
|
|
function(callback)
|
|
|
|
{
|
|
|
|
authorManager.getAuthorColorId(author, function(err, value)
|
2011-05-16 14:10:53 +02:00
|
|
|
{
|
2011-12-04 16:50:02 +01:00
|
|
|
if(ERR(err, callback)) return;
|
2011-08-13 23:07:21 +02:00
|
|
|
authorColorId = value;
|
2011-12-04 16:50:02 +01:00
|
|
|
callback();
|
2011-08-13 23:07:21 +02:00
|
|
|
});
|
|
|
|
},
|
|
|
|
//get author name
|
|
|
|
function(callback)
|
|
|
|
{
|
|
|
|
authorManager.getAuthorName(author, function(err, value)
|
2011-05-16 14:10:53 +02:00
|
|
|
{
|
2011-12-04 16:50:02 +01:00
|
|
|
if(ERR(err, callback)) return;
|
2011-08-13 23:07:21 +02:00
|
|
|
authorName = value;
|
2011-12-04 16:50:02 +01:00
|
|
|
callback();
|
2011-08-13 23:07:21 +02:00
|
|
|
});
|
|
|
|
},
|
|
|
|
function(callback)
|
|
|
|
{
|
2012-04-23 16:18:14 +02:00
|
|
|
padManager.getPad(padIds.padId, function(err, value)
|
2011-05-17 17:33:54 +02:00
|
|
|
{
|
2011-12-04 16:50:02 +01:00
|
|
|
if(ERR(err, callback)) return;
|
2011-08-13 23:07:21 +02:00
|
|
|
pad = value;
|
2011-12-04 16:50:02 +01:00
|
|
|
callback();
|
2011-08-13 23:07:21 +02:00
|
|
|
});
|
|
|
|
}
|
|
|
|
], callback);
|
2011-03-26 14:10:41 +01:00
|
|
|
},
|
2011-07-14 17:15:38 +02:00
|
|
|
//these db requests all need the pad object
|
2011-05-16 14:10:53 +02:00
|
|
|
function(callback)
|
2011-06-30 13:03:20 +02:00
|
|
|
{
|
|
|
|
var authors = pad.getAllAuthors();
|
|
|
|
|
2011-07-14 17:15:38 +02:00
|
|
|
async.parallel([
|
2012-04-23 12:52:30 +02:00
|
|
|
//get timestamp of latest revission needed for timeslider
|
|
|
|
function(callback)
|
|
|
|
{
|
|
|
|
pad.getRevisionDate(pad.getHeadRevisionNumber(), function(err, date)
|
|
|
|
{
|
|
|
|
if(ERR(err, callback)) return;
|
|
|
|
currentTime = date;
|
|
|
|
callback();
|
|
|
|
});
|
|
|
|
},
|
2011-07-14 17:15:38 +02:00
|
|
|
//get all author data out of the database
|
|
|
|
function(callback)
|
2011-06-30 13:03:20 +02:00
|
|
|
{
|
2011-07-14 17:15:38 +02:00
|
|
|
async.forEach(authors, function(authorId, callback)
|
|
|
|
{
|
|
|
|
authorManager.getAuthor(authorId, function(err, author)
|
|
|
|
{
|
2011-12-04 16:50:02 +01:00
|
|
|
if(ERR(err, callback)) return;
|
2011-07-31 17:16:59 +02:00
|
|
|
delete author.timestamp;
|
2011-07-14 17:15:38 +02:00
|
|
|
historicalAuthorData[authorId] = author;
|
2011-12-04 16:50:02 +01:00
|
|
|
callback();
|
2011-07-14 17:15:38 +02:00
|
|
|
});
|
|
|
|
}, callback);
|
|
|
|
},
|
|
|
|
//get the latest chat messages
|
|
|
|
function(callback)
|
|
|
|
{
|
2011-08-17 21:04:28 +02:00
|
|
|
pad.getLastChatMessages(100, function(err, _chatMessages)
|
2011-07-14 17:15:38 +02:00
|
|
|
{
|
2011-12-04 16:50:02 +01:00
|
|
|
if(ERR(err, callback)) return;
|
2011-07-14 17:15:38 +02:00
|
|
|
chatMessages = _chatMessages;
|
2011-12-04 16:50:02 +01:00
|
|
|
callback();
|
2011-07-14 17:15:38 +02:00
|
|
|
});
|
|
|
|
}
|
|
|
|
], callback);
|
|
|
|
|
2011-06-30 13:03:20 +02:00
|
|
|
},
|
|
|
|
function(callback)
|
2011-05-16 14:10:53 +02:00
|
|
|
{
|
2011-06-04 19:02:22 +02:00
|
|
|
//Check if this author is already on the pad, if yes, kick the other sessions!
|
2012-04-23 16:18:14 +02:00
|
|
|
if(pad2sessions[padIds.padId])
|
2011-05-16 14:10:53 +02:00
|
|
|
{
|
2012-04-23 16:18:14 +02:00
|
|
|
for(var i in pad2sessions[padIds.padId])
|
2011-05-16 14:10:53 +02:00
|
|
|
{
|
2012-04-23 16:18:14 +02:00
|
|
|
if(sessioninfos[pad2sessions[padIds.padId][i]] && sessioninfos[pad2sessions[padIds.padId][i]].author == author)
|
2011-05-16 14:10:53 +02:00
|
|
|
{
|
2012-04-23 16:18:14 +02:00
|
|
|
var socket = socketio.sockets.sockets[pad2sessions[padIds.padId][i]];
|
2012-03-14 01:32:16 +01:00
|
|
|
if(socket) socket.json.send({disconnect:"userdup"});
|
2011-05-16 14:10:53 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-04-23 16:18:14 +02:00
|
|
|
//Save in sessioninfos that this session belonges to this pad
|
2011-06-23 15:08:18 +02:00
|
|
|
var sessionId=String(client.id);
|
2012-04-23 16:18:14 +02:00
|
|
|
sessioninfos[sessionId].padId = padIds.padId;
|
|
|
|
sessioninfos[sessionId].readOnlyPadId = padIds.readOnlyPadId;
|
|
|
|
sessioninfos[sessionId].readonly = padIds.readonly;
|
2011-05-16 14:10:53 +02:00
|
|
|
|
|
|
|
//check if there is already a pad2sessions entry, if not, create one
|
2012-04-23 16:18:14 +02:00
|
|
|
if(!pad2sessions[padIds.padId])
|
2011-05-16 14:10:53 +02:00
|
|
|
{
|
2012-04-23 16:18:14 +02:00
|
|
|
pad2sessions[padIds.padId] = [];
|
2011-05-16 14:10:53 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
//Saves in pad2sessions that this session belongs to this pad
|
2012-04-23 16:18:14 +02:00
|
|
|
pad2sessions[padIds.padId].push(sessionId);
|
2011-05-16 14:10:53 +02:00
|
|
|
|
|
|
|
//prepare all values for the wire
|
2011-05-19 00:04:11 +02:00
|
|
|
var atext = Changeset.cloneAText(pad.atext);
|
2011-05-16 17:28:58 +02:00
|
|
|
var attribsForWire = Changeset.prepareForWire(atext.attribs, pad.pool);
|
2011-05-16 14:10:53 +02:00
|
|
|
var apool = attribsForWire.pool.toJsonable();
|
|
|
|
atext.attribs = attribsForWire.translated;
|
|
|
|
|
2012-04-23 16:18:14 +02:00
|
|
|
// Warning: never ever send padIds.padId to the client. If the
|
|
|
|
// client is read only you would open a security hole 1 swedish
|
|
|
|
// mile wide...
|
2011-05-16 14:10:53 +02:00
|
|
|
var clientVars = {
|
|
|
|
"accountPrivs": {
|
|
|
|
"maxRevisions": 100
|
|
|
|
},
|
|
|
|
"initialRevisionList": [],
|
|
|
|
"initialOptions": {
|
|
|
|
"guestPolicy": "deny"
|
|
|
|
},
|
2012-04-23 12:58:04 +02:00
|
|
|
"savedRevisions": pad.getSavedRevisions(),
|
2011-05-16 14:10:53 +02:00
|
|
|
"collab_client_vars": {
|
|
|
|
"initialAttributedText": atext,
|
2011-07-31 17:14:55 +02:00
|
|
|
"clientIp": "127.0.0.1",
|
2011-05-16 14:10:53 +02:00
|
|
|
//"clientAgent": "Anonymous Agent",
|
|
|
|
"padId": message.padId,
|
2011-06-30 13:03:20 +02:00
|
|
|
"historicalAuthorData": historicalAuthorData,
|
2011-05-16 14:10:53 +02:00
|
|
|
"apool": apool,
|
|
|
|
"rev": pad.getHeadRevisionNumber(),
|
2012-04-23 12:52:30 +02:00
|
|
|
"globalPadId": message.padId,
|
|
|
|
"time": currentTime,
|
2011-05-16 14:10:53 +02:00
|
|
|
},
|
2011-08-16 22:29:51 +02:00
|
|
|
"colorPalette": ["#ffc7c7", "#fff1c7", "#e3ffc7", "#c7ffd5", "#c7ffff", "#c7d5ff", "#e3c7ff", "#ffc7f1", "#ff8f8f", "#ffe38f", "#c7ff8f", "#8fffab", "#8fffff", "#8fabff", "#c78fff", "#ff8fe3", "#d97979", "#d9c179", "#a9d979", "#79d991", "#79d9d9", "#7991d9", "#a979d9", "#d979c1", "#d9a9a9", "#d9cda9", "#c1d9a9", "#a9d9b5", "#a9d9d9", "#a9b5d9", "#c1a9d9", "#d9a9cd", "#4c9c82", "#12d1ad", "#2d8e80", "#7485c3", "#a091c7", "#3185ab", "#6818b4", "#e6e76d", "#a42c64", "#f386e5", "#4ecc0c", "#c0c236", "#693224", "#b5de6a", "#9b88fd", "#358f9b", "#496d2f", "#e267fe", "#d23056", "#1a1a64", "#5aa335", "#d722bb", "#86dc6c", "#b5a714", "#955b6a", "#9f2985", "#4b81c8", "#3d6a5b", "#434e16", "#d16084", "#af6a0e", "#8c8bd8"],
|
2011-07-31 17:14:55 +02:00
|
|
|
"clientIp": "127.0.0.1",
|
2011-05-16 14:10:53 +02:00
|
|
|
"userIsGuest": true,
|
|
|
|
"userColor": authorColorId,
|
2011-03-26 14:10:41 +01:00
|
|
|
"padId": message.padId,
|
2011-05-16 14:10:53 +02:00
|
|
|
"initialTitle": "Pad: " + message.padId,
|
|
|
|
"opts": {},
|
2011-07-14 17:15:38 +02:00
|
|
|
"chatHistory": chatMessages,
|
2012-04-23 16:18:14 +02:00
|
|
|
"numConnectedUsers": pad2sessions[padIds.padId].length,
|
2011-05-16 14:10:53 +02:00
|
|
|
"isProPad": false,
|
2012-04-23 16:18:14 +02:00
|
|
|
"readOnlyId": padIds.readOnlyPadId,
|
2012-04-23 16:41:41 +02:00
|
|
|
"readonly": padIds.readonly,
|
2011-05-16 14:10:53 +02:00
|
|
|
"serverTimestamp": new Date().getTime(),
|
|
|
|
"globalPadId": message.padId,
|
|
|
|
"userId": author,
|
|
|
|
"cookiePrefsToSet": {
|
|
|
|
"fullWidth": false,
|
|
|
|
"hideSidebar": false
|
|
|
|
},
|
2011-12-18 06:18:35 +01:00
|
|
|
"abiwordAvailable": settings.abiwordAvailable(),
|
2012-02-29 16:05:44 +01:00
|
|
|
"plugins": {
|
|
|
|
"plugins": plugins.plugins,
|
|
|
|
"parts": plugins.parts,
|
2012-04-23 12:52:30 +02:00
|
|
|
},
|
|
|
|
"initialChangesets": [] // FIXME: REMOVE THIS SHIT
|
|
|
|
|
2011-03-26 14:10:41 +01:00
|
|
|
}
|
2012-02-29 16:05:44 +01:00
|
|
|
|
2011-05-16 14:10:53 +02:00
|
|
|
//Add a username to the clientVars if one avaiable
|
|
|
|
if(authorName != null)
|
|
|
|
{
|
|
|
|
clientVars.userName = authorName;
|
|
|
|
}
|
|
|
|
|
2011-12-04 22:01:49 +01:00
|
|
|
if(sessioninfos[client.id] !== undefined)
|
2011-11-26 00:24:10 +01:00
|
|
|
{
|
2011-12-04 22:01:49 +01:00
|
|
|
//This is a reconnect, so we don't have to send the client the ClientVars again
|
|
|
|
if(message.reconnect == true)
|
|
|
|
{
|
|
|
|
//Save the revision in sessioninfos, we take the revision from the info the client send to us
|
|
|
|
sessioninfos[client.id].rev = message.client_rev;
|
|
|
|
}
|
|
|
|
//This is a normal first connect
|
|
|
|
else
|
|
|
|
{
|
|
|
|
//Send the clientVars to the Client
|
2012-04-23 12:52:30 +02:00
|
|
|
client.json.send({type: "CLIENT_VARS", data: clientVars});
|
2011-12-04 22:01:49 +01:00
|
|
|
//Save the revision in sessioninfos
|
|
|
|
sessioninfos[client.id].rev = pad.getHeadRevisionNumber();
|
|
|
|
}
|
|
|
|
|
|
|
|
//Save the revision and the author id in sessioninfos
|
|
|
|
sessioninfos[client.id].author = author;
|
2011-11-26 00:24:10 +01:00
|
|
|
}
|
2011-05-16 14:10:53 +02:00
|
|
|
|
|
|
|
//prepare the notification for the other users on the pad, that this user joined
|
|
|
|
var messageToTheOtherUsers = {
|
2011-03-26 14:10:41 +01:00
|
|
|
"type": "COLLABROOM",
|
|
|
|
"data": {
|
|
|
|
type: "USER_NEWINFO",
|
|
|
|
userInfo: {
|
|
|
|
"ip": "127.0.0.1",
|
2011-05-16 14:10:53 +02:00
|
|
|
"colorId": authorColorId,
|
2011-03-26 14:10:41 +01:00
|
|
|
"userAgent": "Anonymous",
|
2011-05-16 14:10:53 +02:00
|
|
|
"userId": author
|
2011-03-26 14:10:41 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
2011-05-16 14:10:53 +02:00
|
|
|
|
|
|
|
//Add the authorname of this new User, if avaiable
|
|
|
|
if(authorName != null)
|
|
|
|
{
|
|
|
|
messageToTheOtherUsers.data.userInfo.name = authorName;
|
|
|
|
}
|
|
|
|
|
|
|
|
//Run trough all sessions of this pad
|
2012-04-23 16:18:14 +02:00
|
|
|
async.forEach(pad2sessions[padIds.padId], function(sessionID, callback)
|
2011-05-16 14:10:53 +02:00
|
|
|
{
|
2012-02-17 11:40:29 +01:00
|
|
|
var author, socket, sessionAuthorName, sessionAuthorColorId;
|
|
|
|
|
|
|
|
//Since sessioninfos might change while being enumerated, check if the
|
|
|
|
//sessionID is still assigned to a valid session
|
|
|
|
if(sessioninfos[sessionID] !== undefined &&
|
|
|
|
socketio.sockets.sockets[sessionID] !== undefined){
|
|
|
|
author = sessioninfos[sessionID].author;
|
|
|
|
socket = socketio.sockets.sockets[sessionID];
|
|
|
|
}else {
|
|
|
|
// If the sessionID is not valid, callback();
|
|
|
|
callback();
|
|
|
|
return;
|
|
|
|
}
|
2011-05-16 14:10:53 +02:00
|
|
|
async.series([
|
|
|
|
//get the authorname & colorId
|
|
|
|
function(callback)
|
|
|
|
{
|
|
|
|
async.parallel([
|
|
|
|
function(callback)
|
|
|
|
{
|
2012-02-17 11:40:29 +01:00
|
|
|
authorManager.getAuthorColorId(author, function(err, value)
|
2011-05-16 14:10:53 +02:00
|
|
|
{
|
2011-12-04 16:50:02 +01:00
|
|
|
if(ERR(err, callback)) return;
|
2011-05-16 14:10:53 +02:00
|
|
|
sessionAuthorColorId = value;
|
2011-12-04 16:50:02 +01:00
|
|
|
callback();
|
2011-05-16 14:10:53 +02:00
|
|
|
})
|
|
|
|
},
|
|
|
|
function(callback)
|
|
|
|
{
|
2012-02-17 11:40:29 +01:00
|
|
|
authorManager.getAuthorName(author, function(err, value)
|
2011-05-16 14:10:53 +02:00
|
|
|
{
|
2011-12-04 16:50:02 +01:00
|
|
|
if(ERR(err, callback)) return;
|
2011-05-16 14:10:53 +02:00
|
|
|
sessionAuthorName = value;
|
2011-12-04 16:50:02 +01:00
|
|
|
callback();
|
2011-05-16 14:10:53 +02:00
|
|
|
})
|
|
|
|
}
|
|
|
|
],callback);
|
|
|
|
},
|
|
|
|
function (callback)
|
|
|
|
{
|
|
|
|
//Jump over, if this session is the connection session
|
2012-02-17 11:40:29 +01:00
|
|
|
if(sessionID != client.id)
|
2011-05-16 14:10:53 +02:00
|
|
|
{
|
|
|
|
//Send this Session the Notification about the new user
|
2012-02-17 11:40:29 +01:00
|
|
|
socket.json.send(messageToTheOtherUsers);
|
2011-05-16 14:10:53 +02:00
|
|
|
|
|
|
|
//Send the new User a Notification about this other user
|
|
|
|
var messageToNotifyTheClientAboutTheOthers = {
|
|
|
|
"type": "COLLABROOM",
|
|
|
|
"data": {
|
|
|
|
type: "USER_NEWINFO",
|
|
|
|
userInfo: {
|
|
|
|
"ip": "127.0.0.1",
|
|
|
|
"colorId": sessionAuthorColorId,
|
|
|
|
"name": sessionAuthorName,
|
|
|
|
"userAgent": "Anonymous",
|
2012-02-17 11:40:29 +01:00
|
|
|
"userId": author
|
2011-05-16 14:10:53 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
2011-06-23 15:08:18 +02:00
|
|
|
client.json.send(messageToNotifyTheClientAboutTheOthers);
|
2011-05-16 14:10:53 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
], callback);
|
|
|
|
}, callback);
|
2011-03-26 14:10:41 +01:00
|
|
|
}
|
2011-05-16 14:10:53 +02:00
|
|
|
],function(err)
|
|
|
|
{
|
2011-12-04 16:50:02 +01:00
|
|
|
ERR(err);
|
2011-05-16 14:10:53 +02:00
|
|
|
});
|
2011-03-26 14:10:41 +01:00
|
|
|
}
|
2012-04-23 12:52:30 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Handles a request for a rough changeset, the timeslider client needs it
|
|
|
|
*/
|
|
|
|
function handleChangesetRequest(client, message)
|
|
|
|
{
|
|
|
|
//check if all ok
|
|
|
|
if(message.data == null)
|
|
|
|
{
|
|
|
|
messageLogger.warn("Dropped message, changeset request has no data!");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if(message.padId == null)
|
|
|
|
{
|
|
|
|
messageLogger.warn("Dropped message, changeset request has no padId!");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if(message.data.granularity == null)
|
|
|
|
{
|
|
|
|
messageLogger.warn("Dropped message, changeset request has no granularity!");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if(message.data.start == null)
|
|
|
|
{
|
|
|
|
messageLogger.warn("Dropped message, changeset request has no start!");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if(message.data.requestID == null)
|
|
|
|
{
|
|
|
|
messageLogger.warn("Dropped message, changeset request has no requestID!");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
var granularity = message.data.granularity;
|
|
|
|
var start = message.data.start;
|
|
|
|
var end = start + (100 * granularity);
|
|
|
|
var padId = message.padId;
|
|
|
|
|
|
|
|
//build the requested rough changesets and send them back
|
|
|
|
getChangesetInfo(padId, start, end, granularity, function(err, changesetInfo)
|
|
|
|
{
|
|
|
|
ERR(err);
|
|
|
|
|
|
|
|
var data = changesetInfo;
|
|
|
|
data.requestID = message.data.requestID;
|
|
|
|
|
|
|
|
client.json.send({type: "CHANGESET_REQ", data: data});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Tries to rebuild the getChangestInfo function of the original Etherpad
|
|
|
|
* https://github.com/ether/pad/blob/master/etherpad/src/etherpad/control/pad/pad_changeset_control.js#L144
|
|
|
|
*/
|
|
|
|
function getChangesetInfo(padId, startNum, endNum, granularity, callback)
|
|
|
|
{
|
|
|
|
var forwardsChangesets = [];
|
|
|
|
var backwardsChangesets = [];
|
|
|
|
var timeDeltas = [];
|
|
|
|
var apool = new AttributePool();
|
|
|
|
var pad;
|
|
|
|
var composedChangesets = {};
|
|
|
|
var revisionDate = [];
|
|
|
|
var lines;
|
|
|
|
|
|
|
|
async.series([
|
|
|
|
//get the pad from the database
|
|
|
|
function(callback)
|
|
|
|
{
|
|
|
|
padManager.getPad(padId, function(err, _pad)
|
|
|
|
{
|
|
|
|
if(ERR(err, callback)) return;
|
|
|
|
pad = _pad;
|
|
|
|
callback();
|
|
|
|
});
|
|
|
|
},
|
|
|
|
function(callback)
|
|
|
|
{
|
|
|
|
//calculate the last full endnum
|
|
|
|
var lastRev = pad.getHeadRevisionNumber();
|
|
|
|
if (endNum > lastRev+1) {
|
|
|
|
endNum = lastRev+1;
|
|
|
|
}
|
|
|
|
endNum = Math.floor(endNum / granularity)*granularity;
|
|
|
|
|
|
|
|
var compositesChangesetNeeded = [];
|
|
|
|
var revTimesNeeded = [];
|
|
|
|
|
|
|
|
//figure out which composite Changeset and revTimes we need, to load them in bulk
|
|
|
|
var compositeStart = startNum;
|
|
|
|
while (compositeStart < endNum)
|
|
|
|
{
|
|
|
|
var compositeEnd = compositeStart + granularity;
|
|
|
|
|
|
|
|
//add the composite Changeset we needed
|
|
|
|
compositesChangesetNeeded.push({start: compositeStart, end: compositeEnd});
|
|
|
|
|
|
|
|
//add the t1 time we need
|
|
|
|
revTimesNeeded.push(compositeStart == 0 ? 0 : compositeStart - 1);
|
|
|
|
//add the t2 time we need
|
|
|
|
revTimesNeeded.push(compositeEnd - 1);
|
|
|
|
|
|
|
|
compositeStart += granularity;
|
|
|
|
}
|
|
|
|
|
|
|
|
//get all needed db values parallel
|
|
|
|
async.parallel([
|
|
|
|
function(callback)
|
|
|
|
{
|
|
|
|
//get all needed composite Changesets
|
|
|
|
async.forEach(compositesChangesetNeeded, function(item, callback)
|
|
|
|
{
|
|
|
|
composePadChangesets(padId, item.start, item.end, function(err, changeset)
|
|
|
|
{
|
|
|
|
if(ERR(err, callback)) return;
|
|
|
|
composedChangesets[item.start + "/" + item.end] = changeset;
|
|
|
|
callback();
|
|
|
|
});
|
|
|
|
}, callback);
|
|
|
|
},
|
|
|
|
function(callback)
|
|
|
|
{
|
|
|
|
//get all needed revision Dates
|
|
|
|
async.forEach(revTimesNeeded, function(revNum, callback)
|
|
|
|
{
|
|
|
|
pad.getRevisionDate(revNum, function(err, revDate)
|
|
|
|
{
|
|
|
|
if(ERR(err, callback)) return;
|
|
|
|
revisionDate[revNum] = Math.floor(revDate/1000);
|
|
|
|
callback();
|
|
|
|
});
|
|
|
|
}, callback);
|
|
|
|
},
|
|
|
|
//get the lines
|
|
|
|
function(callback)
|
|
|
|
{
|
|
|
|
getPadLines(padId, startNum-1, function(err, _lines)
|
|
|
|
{
|
|
|
|
if(ERR(err, callback)) return;
|
|
|
|
lines = _lines;
|
|
|
|
callback();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
], callback);
|
|
|
|
},
|
|
|
|
//doesn't know what happens here excatly :/
|
|
|
|
function(callback)
|
|
|
|
{
|
|
|
|
var compositeStart = startNum;
|
|
|
|
|
|
|
|
while (compositeStart < endNum)
|
|
|
|
{
|
|
|
|
if (compositeStart + granularity > endNum)
|
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
var compositeEnd = compositeStart + granularity;
|
|
|
|
|
|
|
|
var forwards = composedChangesets[compositeStart + "/" + compositeEnd];
|
|
|
|
var backwards = Changeset.inverse(forwards, lines.textlines, lines.alines, pad.apool());
|
|
|
|
|
|
|
|
Changeset.mutateAttributionLines(forwards, lines.alines, pad.apool());
|
|
|
|
Changeset.mutateTextLines(forwards, lines.textlines);
|
|
|
|
|
|
|
|
var forwards2 = Changeset.moveOpsToNewPool(forwards, pad.apool(), apool);
|
|
|
|
var backwards2 = Changeset.moveOpsToNewPool(backwards, pad.apool(), apool);
|
|
|
|
|
|
|
|
var t1, t2;
|
|
|
|
if (compositeStart == 0)
|
|
|
|
{
|
|
|
|
t1 = revisionDate[0];
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
t1 = revisionDate[compositeStart - 1];
|
|
|
|
}
|
|
|
|
|
|
|
|
t2 = revisionDate[compositeEnd - 1];
|
|
|
|
|
|
|
|
timeDeltas.push(t2 - t1);
|
|
|
|
forwardsChangesets.push(forwards2);
|
|
|
|
backwardsChangesets.push(backwards2);
|
|
|
|
|
|
|
|
compositeStart += granularity;
|
|
|
|
}
|
|
|
|
|
|
|
|
callback();
|
|
|
|
}
|
|
|
|
], function(err)
|
|
|
|
{
|
|
|
|
if(ERR(err, callback)) return;
|
|
|
|
|
|
|
|
callback(null, {forwardsChangesets: forwardsChangesets,
|
|
|
|
backwardsChangesets: backwardsChangesets,
|
|
|
|
apool: apool.toJsonable(),
|
|
|
|
actualEndNum: endNum,
|
|
|
|
timeDeltas: timeDeltas,
|
|
|
|
start: startNum,
|
|
|
|
granularity: granularity });
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Tries to rebuild the getPadLines function of the original Etherpad
|
|
|
|
* https://github.com/ether/pad/blob/master/etherpad/src/etherpad/control/pad/pad_changeset_control.js#L263
|
|
|
|
*/
|
|
|
|
function getPadLines(padId, revNum, callback)
|
|
|
|
{
|
|
|
|
var atext;
|
|
|
|
var result = {};
|
|
|
|
var pad;
|
|
|
|
|
|
|
|
async.series([
|
|
|
|
//get the pad from the database
|
|
|
|
function(callback)
|
|
|
|
{
|
|
|
|
padManager.getPad(padId, function(err, _pad)
|
|
|
|
{
|
|
|
|
if(ERR(err, callback)) return;
|
|
|
|
pad = _pad;
|
|
|
|
callback();
|
|
|
|
});
|
|
|
|
},
|
|
|
|
//get the atext
|
|
|
|
function(callback)
|
|
|
|
{
|
|
|
|
if(revNum >= 0)
|
|
|
|
{
|
|
|
|
pad.getInternalRevisionAText(revNum, function(err, _atext)
|
|
|
|
{
|
|
|
|
if(ERR(err, callback)) return;
|
|
|
|
atext = _atext;
|
|
|
|
callback();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
atext = Changeset.makeAText("\n");
|
|
|
|
callback(null);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
function(callback)
|
|
|
|
{
|
|
|
|
result.textlines = Changeset.splitTextLines(atext.text);
|
|
|
|
result.alines = Changeset.splitAttributionLines(atext.attribs, atext.text);
|
|
|
|
callback(null);
|
|
|
|
}
|
|
|
|
], function(err)
|
|
|
|
{
|
|
|
|
if(ERR(err, callback)) return;
|
|
|
|
callback(null, result);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Tries to rebuild the composePadChangeset function of the original Etherpad
|
|
|
|
* https://github.com/ether/pad/blob/master/etherpad/src/etherpad/control/pad/pad_changeset_control.js#L241
|
|
|
|
*/
|
|
|
|
function composePadChangesets(padId, startNum, endNum, callback)
|
|
|
|
{
|
|
|
|
var pad;
|
|
|
|
var changesets = [];
|
|
|
|
var changeset;
|
|
|
|
|
|
|
|
async.series([
|
|
|
|
//get the pad from the database
|
|
|
|
function(callback)
|
|
|
|
{
|
|
|
|
padManager.getPad(padId, function(err, _pad)
|
|
|
|
{
|
|
|
|
if(ERR(err, callback)) return;
|
|
|
|
pad = _pad;
|
|
|
|
callback();
|
|
|
|
});
|
|
|
|
},
|
|
|
|
//fetch all changesets we need
|
|
|
|
function(callback)
|
|
|
|
{
|
|
|
|
var changesetsNeeded=[];
|
|
|
|
|
|
|
|
//create a array for all changesets, we will
|
|
|
|
//replace the values with the changeset later
|
|
|
|
for(var r=startNum;r<endNum;r++)
|
|
|
|
{
|
|
|
|
changesetsNeeded.push(r);
|
|
|
|
}
|
|
|
|
|
|
|
|
//get all changesets
|
|
|
|
async.forEach(changesetsNeeded, function(revNum,callback)
|
|
|
|
{
|
|
|
|
pad.getRevisionChangeset(revNum, function(err, value)
|
|
|
|
{
|
|
|
|
if(ERR(err, callback)) return;
|
|
|
|
changesets[revNum] = value;
|
|
|
|
callback();
|
|
|
|
});
|
|
|
|
},callback);
|
|
|
|
},
|
|
|
|
//compose Changesets
|
|
|
|
function(callback)
|
|
|
|
{
|
|
|
|
changeset = changesets[startNum];
|
|
|
|
var pool = pad.apool();
|
|
|
|
|
|
|
|
for(var r=startNum+1;r<endNum;r++)
|
|
|
|
{
|
|
|
|
var cs = changesets[r];
|
|
|
|
changeset = Changeset.compose(changeset, cs, pool);
|
|
|
|
}
|
|
|
|
|
|
|
|
callback(null);
|
|
|
|
}
|
|
|
|
],
|
|
|
|
//return err and changeset
|
|
|
|
function(err)
|
|
|
|
{
|
|
|
|
if(ERR(err, callback)) return;
|
|
|
|
callback(null, changeset);
|
|
|
|
});
|
|
|
|
}
|
2012-06-29 20:26:12 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the number of users in a pad
|
|
|
|
*/
|
|
|
|
exports.padUsersCount = function (padID, callback) {
|
|
|
|
if (!pad2sessions[padID] || typeof pad2sessions[padID] != typeof []) {
|
|
|
|
callback(null, {padUsersCount: 0});
|
|
|
|
} else {
|
|
|
|
callback(null, {padUsersCount: pad2sessions[padID].length});
|
|
|
|
}
|
|
|
|
}
|