mirror of https://github.com/movim/movim
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.
234 lines
6.8 KiB
234 lines
6.8 KiB
/**
|
|
* Movim Websocket
|
|
*
|
|
* This file define the websocket behaviour and handle its connection
|
|
*/
|
|
|
|
WebSocket.prototype.unregister = function() {
|
|
this.send(JSON.stringify({'func' : 'unregister'}));
|
|
};
|
|
|
|
WebSocket.prototype.register = function(host) {
|
|
this.send(JSON.stringify({'func' : 'register', 'host' : host}));
|
|
};
|
|
|
|
/**
|
|
* @brief Definition of the MovimWebsocket object
|
|
* @param string error
|
|
*/
|
|
|
|
var MovimWebsocket = {
|
|
connection: null,
|
|
attached: [],
|
|
registered: [],
|
|
attempts: 1,
|
|
pong: false,
|
|
closed: false,
|
|
|
|
launchAttached : function() {
|
|
// We hide the Websocket error
|
|
MovimUtils.hideElement(document.getElementById('error_websocket'));
|
|
|
|
for(var i = 0; i < MovimWebsocket.attached.length; i++) {
|
|
MovimWebsocket.attached[i]();
|
|
}
|
|
},
|
|
|
|
launchRegistered : function() {
|
|
for(var i = 0; i < MovimWebsocket.registered.length; i++) {
|
|
MovimWebsocket.registered[i]();
|
|
}
|
|
},
|
|
|
|
init : function() {
|
|
if(SECURE_WEBSOCKET) {
|
|
var uri = 'wss://' + BASE_HOST + '/ws/';
|
|
} else {
|
|
var uri = 'ws://' + BASE_HOST + '/ws/';
|
|
}
|
|
|
|
if(this.connection
|
|
&& this.connection.readyState == 1) {
|
|
this.connection.onclose = null;
|
|
this.connection.close();
|
|
}
|
|
|
|
var date = new Date();
|
|
|
|
this.connection = new WebSocket(uri +
|
|
'?path=' + MovimUtils.urlParts().page +
|
|
'&offset=' + date.getTimezoneOffset());
|
|
|
|
this.connection.onopen = function(e) {
|
|
console.log("Connection established!");
|
|
MovimWebsocket.attempts = 1;
|
|
MovimWebsocket.launchAttached();
|
|
setTimeout(function(){
|
|
MovimWebsocket.ping();
|
|
}, 10000);
|
|
};
|
|
|
|
this.connection.onmessage = function(e) {
|
|
var obj = JSON.parse(e.data);
|
|
|
|
if(obj != null) {
|
|
if(obj.func == 'registered') {
|
|
MovimWebsocket.launchRegistered();
|
|
}
|
|
|
|
if(obj.func == 'disconnected') {
|
|
MovimUtils.disconnect();
|
|
}
|
|
|
|
if(obj.func == 'pong') {
|
|
MovimWebsocket.pong = true;
|
|
}
|
|
|
|
if(obj.func == 'block') {
|
|
MovimWebsocket.clearAttached();
|
|
MovimUtils.addClass('body', 'disabled');
|
|
}
|
|
|
|
MovimWebsocket.handle(obj);
|
|
}
|
|
};
|
|
|
|
this.connection.onclose = function(e) {
|
|
console.log("Connection closed by the server or session closed");
|
|
|
|
if(e.code == 1008) {
|
|
// The server closed the connection and asked to keep it this way
|
|
this.closed = true;
|
|
MovimUtils.showElement(document.getElementById('error_websocket'));
|
|
MovimWebsocket.connection.close();
|
|
} if(e.code == 1006) {
|
|
MovimWebsocket.reconnect();
|
|
} else if(e.code == 1000) {
|
|
MovimUtils.disconnect();
|
|
}
|
|
};
|
|
|
|
this.connection.onerror = function(e) {
|
|
console.log("Connection error!");
|
|
|
|
// We show the Websocket error
|
|
MovimUtils.showElement(document.getElementById('error_websocket'));
|
|
MovimWebsocket.reconnect();
|
|
|
|
// We prevent the onclose launch
|
|
this.onclose = null;
|
|
};
|
|
},
|
|
|
|
send : function(widget, func, params) {
|
|
if(this.connection.readyState == 1) {
|
|
this.connection.send(
|
|
JSON.stringify(
|
|
{'func' : 'message', 'body' :
|
|
{
|
|
'widget' : widget,
|
|
'func' : func,
|
|
'params' : params
|
|
}
|
|
}
|
|
)
|
|
);
|
|
}
|
|
},
|
|
|
|
// A ping/pong system to handle socket errors for buggy browser (Chrome on Linux…)
|
|
ping : function() {
|
|
if(this.connection.readyState == 1 && !this.closed) {
|
|
this.connection.send(
|
|
JSON.stringify(
|
|
{'func' : 'ping'}
|
|
)
|
|
);
|
|
|
|
setTimeout(function(){
|
|
if(MovimWebsocket.pong == false) {
|
|
MovimWebsocket.connection.onerror();
|
|
} else {
|
|
MovimWebsocket.pong = false;
|
|
MovimWebsocket.ping();
|
|
}
|
|
}, 10000);
|
|
}
|
|
},
|
|
|
|
attach : function(func) {
|
|
if(typeof(func) === "function") {
|
|
this.attached.push(func);
|
|
}
|
|
},
|
|
|
|
register : function(func) {
|
|
if(typeof(func) === "function") {
|
|
this.registered.push(func);
|
|
}
|
|
},
|
|
|
|
clearAttached : function() {
|
|
this.attached = [];
|
|
},
|
|
|
|
handle : function(funcall) {
|
|
if(funcall.func != null && (typeof window[funcall.func] == 'function')) {
|
|
try {
|
|
window[funcall.func].apply(null, funcall.params);
|
|
} catch(err) {
|
|
console.log("Error caught: "
|
|
+ err.toString()
|
|
+ " - "
|
|
+ funcall.func
|
|
+ ":"
|
|
+ JSON.stringify(funcall.params)
|
|
);
|
|
}
|
|
} else if(funcall.func != null) {
|
|
var funcs = funcall.func.split('.');
|
|
var called = funcs[0];
|
|
if(typeof window[called] == 'object'
|
|
&& typeof window[funcs[0]][funcs[1]] != 'undefined') {
|
|
window[funcs[0]][funcs[1]].apply(null, funcall.params);
|
|
}
|
|
}
|
|
},
|
|
|
|
unregister : function(reload) {
|
|
this.connection.unregister();
|
|
},
|
|
|
|
reconnect : function() {
|
|
var interval = MovimWebsocket.generateInterval();
|
|
console.log("Try to reconnect");
|
|
setTimeout(function () {
|
|
// We've tried to reconnect so increment the attempts by 1
|
|
MovimWebsocket.attempts++;
|
|
|
|
// Connection has closed so try to reconnect every 10 seconds.
|
|
MovimWebsocket.init();
|
|
}, interval);
|
|
},
|
|
|
|
generateInterval :function() {
|
|
var maxInterval = (Math.pow(2, MovimWebsocket.attempts) - 1) * 1000;
|
|
|
|
if (maxInterval > 30*1000) {
|
|
maxInterval = 30*1000; // If the generated interval is more than 30 seconds, truncate it down to 30 seconds.
|
|
}
|
|
|
|
// generate the interval to a random number between 0 and the maxInterval determined from above
|
|
return Math.random() * maxInterval;
|
|
}
|
|
}
|
|
|
|
window.onbeforeunload = function() {
|
|
MovimWebsocket.connection.onclose = function () {}; // disable onclose handler first
|
|
MovimWebsocket.connection.close()
|
|
};
|
|
|
|
document.addEventListener("DOMContentLoaded", function(event) {
|
|
// And we start it
|
|
MovimWebsocket.init();
|
|
});
|