2014-05-07 20:47:25 +01:00
|
|
|
/**
|
2017-01-11 15:24:33 +00:00
|
|
|
* Copyright JS Foundation and other contributors, http://js.foundation
|
2014-05-07 20:47:25 +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.
|
|
|
|
**/
|
2015-06-30 23:42:03 +01:00
|
|
|
|
2014-08-08 00:01:35 +01:00
|
|
|
RED.comms = (function() {
|
2015-06-30 23:42:03 +01:00
|
|
|
|
2014-05-07 20:47:25 +01:00
|
|
|
var errornotification = null;
|
2014-12-08 16:25:38 +00:00
|
|
|
var clearErrorTimer = null;
|
2016-04-10 17:28:28 +01:00
|
|
|
var connectCountdownTimer = null;
|
|
|
|
var connectCountdown = 10;
|
2014-05-07 20:47:25 +01:00
|
|
|
var subscriptions = {};
|
2014-05-08 14:15:54 +01:00
|
|
|
var ws;
|
2014-12-10 14:16:07 +00:00
|
|
|
var pendingAuth = false;
|
2015-09-23 17:15:28 +01:00
|
|
|
var reconnectAttempts = 0;
|
2016-04-10 18:49:39 +01:00
|
|
|
var active = false;
|
2015-06-30 23:42:03 +01:00
|
|
|
|
2014-05-07 20:47:25 +01:00
|
|
|
function connectWS() {
|
2016-04-10 18:49:39 +01:00
|
|
|
active = true;
|
2015-06-17 22:09:05 +01:00
|
|
|
var path = location.hostname;
|
|
|
|
var port = location.port;
|
|
|
|
if (port.length !== 0) {
|
|
|
|
path = path+":"+port;
|
|
|
|
}
|
|
|
|
path = path+document.location.pathname;
|
2014-05-07 20:47:25 +01:00
|
|
|
path = path+(path.slice(-1) == "/"?"":"/")+"comms";
|
|
|
|
path = "ws"+(document.location.protocol=="https:"?"s":"")+"://"+path;
|
2015-06-30 23:42:03 +01:00
|
|
|
|
2014-12-10 14:16:07 +00:00
|
|
|
var auth_tokens = RED.settings.get("auth-tokens");
|
|
|
|
pendingAuth = (auth_tokens!=null);
|
2015-06-30 23:42:03 +01:00
|
|
|
|
2014-12-10 14:16:07 +00:00
|
|
|
function completeConnection() {
|
|
|
|
for (var t in subscriptions) {
|
|
|
|
if (subscriptions.hasOwnProperty(t)) {
|
|
|
|
ws.send(JSON.stringify({subscribe:t}));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2015-06-30 23:42:03 +01:00
|
|
|
|
2014-05-08 14:15:54 +01:00
|
|
|
ws = new WebSocket(path);
|
2014-05-07 20:47:25 +01:00
|
|
|
ws.onopen = function() {
|
2015-09-23 17:15:28 +01:00
|
|
|
reconnectAttempts = 0;
|
2014-05-07 20:47:25 +01:00
|
|
|
if (errornotification) {
|
2014-12-08 16:25:38 +00:00
|
|
|
clearErrorTimer = setTimeout(function() {
|
|
|
|
errornotification.close();
|
|
|
|
errornotification = null;
|
|
|
|
},1000);
|
2014-05-07 20:47:25 +01:00
|
|
|
}
|
2014-12-10 14:16:07 +00:00
|
|
|
if (pendingAuth) {
|
2014-11-12 13:21:59 +00:00
|
|
|
ws.send(JSON.stringify({auth:auth_tokens.access_token}));
|
2014-12-10 14:16:07 +00:00
|
|
|
} else {
|
|
|
|
completeConnection();
|
2014-05-08 14:15:54 +01:00
|
|
|
}
|
2014-05-07 20:47:25 +01:00
|
|
|
}
|
|
|
|
ws.onmessage = function(event) {
|
|
|
|
var msg = JSON.parse(event.data);
|
2016-04-10 18:49:39 +01:00
|
|
|
if (pendingAuth && msg.auth) {
|
|
|
|
if (msg.auth === "ok") {
|
|
|
|
pendingAuth = false;
|
|
|
|
completeConnection();
|
|
|
|
} else if (msg.auth === "fail") {
|
|
|
|
// anything else is an error...
|
|
|
|
active = false;
|
|
|
|
RED.user.login({updateMenu:true},function() {
|
|
|
|
connectWS();
|
|
|
|
})
|
|
|
|
}
|
2014-12-10 14:16:07 +00:00
|
|
|
} else if (msg.topic) {
|
2014-05-08 14:15:54 +01:00
|
|
|
for (var t in subscriptions) {
|
2014-08-08 00:01:35 +01:00
|
|
|
if (subscriptions.hasOwnProperty(t)) {
|
|
|
|
var re = new RegExp("^"+t.replace(/([\[\]\?\(\)\\\\$\^\*\.|])/g,"\\$1").replace(/\+/g,"[^/]+").replace(/\/#$/,"(\/.*)?")+"$");
|
|
|
|
if (re.test(msg.topic)) {
|
|
|
|
var subscribers = subscriptions[t];
|
|
|
|
if (subscribers) {
|
|
|
|
for (var i=0;i<subscribers.length;i++) {
|
|
|
|
subscribers[i](msg.topic,msg.data);
|
|
|
|
}
|
2014-05-08 14:15:54 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2014-05-07 20:47:25 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
ws.onclose = function() {
|
2016-04-10 18:49:39 +01:00
|
|
|
if (!active) {
|
|
|
|
return;
|
|
|
|
}
|
2016-04-10 17:28:28 +01:00
|
|
|
if (clearErrorTimer) {
|
2014-12-08 16:25:38 +00:00
|
|
|
clearTimeout(clearErrorTimer);
|
|
|
|
clearErrorTimer = null;
|
2014-05-07 20:47:25 +01:00
|
|
|
}
|
2015-09-23 17:15:28 +01:00
|
|
|
reconnectAttempts++;
|
2016-04-10 17:28:28 +01:00
|
|
|
if (reconnectAttempts < 10) {
|
|
|
|
setTimeout(connectWS,1000);
|
|
|
|
if (reconnectAttempts > 5 && errornotification == null) {
|
|
|
|
errornotification = RED.notify(RED._("notification.errors.lostConnection"),"error",true);
|
|
|
|
}
|
|
|
|
} else if (reconnectAttempts < 20) {
|
|
|
|
setTimeout(connectWS,2000);
|
|
|
|
} else {
|
|
|
|
connectCountdown = 60;
|
|
|
|
connectCountdownTimer = setInterval(function() {
|
|
|
|
connectCountdown--;
|
|
|
|
if (connectCountdown === 0) {
|
|
|
|
errornotification.update(RED._("notification.errors.lostConnection"));
|
|
|
|
clearInterval(connectCountdownTimer);
|
|
|
|
connectWS();
|
|
|
|
} else {
|
|
|
|
var msg = RED._("notification.errors.lostConnectionReconnect",{time: connectCountdown})+' <a href="#">'+ RED._("notification.errors.lostConnectionTry")+'</a>';
|
|
|
|
errornotification.update(msg);
|
|
|
|
$(errornotification).find("a").click(function(e) {
|
|
|
|
e.preventDefault();
|
|
|
|
errornotification.update(RED._("notification.errors.lostConnection"));
|
|
|
|
clearInterval(connectCountdownTimer);
|
|
|
|
connectWS();
|
|
|
|
})
|
|
|
|
}
|
|
|
|
},1000);
|
|
|
|
}
|
|
|
|
|
2014-05-07 20:47:25 +01:00
|
|
|
}
|
|
|
|
}
|
2015-06-30 23:42:03 +01:00
|
|
|
|
2014-05-07 20:47:25 +01:00
|
|
|
function subscribe(topic,callback) {
|
|
|
|
if (subscriptions[topic] == null) {
|
|
|
|
subscriptions[topic] = [];
|
|
|
|
}
|
|
|
|
subscriptions[topic].push(callback);
|
2014-05-08 14:15:54 +01:00
|
|
|
if (ws && ws.readyState == 1) {
|
|
|
|
ws.send(JSON.stringify({subscribe:topic}));
|
|
|
|
}
|
2014-05-07 20:47:25 +01:00
|
|
|
}
|
2015-06-30 23:42:03 +01:00
|
|
|
|
2014-08-28 00:35:07 +01:00
|
|
|
function unsubscribe(topic,callback) {
|
2015-02-14 22:25:25 +01:00
|
|
|
if (subscriptions[topic]) {
|
|
|
|
for (var i=0;i<subscriptions[topic].length;i++) {
|
|
|
|
if (subscriptions[topic][i] === callback) {
|
|
|
|
subscriptions[topic].splice(i,1);
|
2014-08-28 00:35:07 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2015-02-14 22:25:25 +01:00
|
|
|
if (subscriptions[topic].length === 0) {
|
|
|
|
delete subscriptions[topic];
|
2014-08-28 00:35:07 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2015-06-30 23:42:03 +01:00
|
|
|
|
2014-05-07 20:47:25 +01:00
|
|
|
return {
|
2014-05-08 14:15:54 +01:00
|
|
|
connect: connectWS,
|
2014-08-28 00:35:07 +01:00
|
|
|
subscribe: subscribe,
|
|
|
|
unsubscribe:unsubscribe
|
2014-05-07 20:47:25 +01:00
|
|
|
}
|
2014-08-08 00:01:35 +01:00
|
|
|
})();
|