node-red/public/red/comms.js

117 lines
4.0 KiB
JavaScript
Raw Normal View History

2014-05-07 21:47:25 +02:00
/**
* Copyright 2014 IBM Corp.
*
* 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.
**/
2014-08-08 01:01:35 +02:00
RED.comms = (function() {
2014-05-07 21:47:25 +02:00
var errornotification = null;
var clearErrorTimer = null;
2014-05-07 21:47:25 +02:00
var subscriptions = {};
2014-05-08 15:15:54 +02:00
var ws;
2014-12-10 15:16:07 +01:00
var pendingAuth = false;
2014-05-07 21:47:25 +02:00
function connectWS() {
var path = location.hostname+":"+location.port+document.location.pathname;
path = path+(path.slice(-1) == "/"?"":"/")+"comms";
path = "ws"+(document.location.protocol=="https:"?"s":"")+"://"+path;
2014-12-10 15:16:07 +01:00
var auth_tokens = RED.settings.get("auth-tokens");
pendingAuth = (auth_tokens!=null);
function completeConnection() {
for (var t in subscriptions) {
if (subscriptions.hasOwnProperty(t)) {
ws.send(JSON.stringify({subscribe:t}));
}
}
}
2014-05-08 15:15:54 +02:00
ws = new WebSocket(path);
2014-05-07 21:47:25 +02:00
ws.onopen = function() {
if (errornotification) {
clearErrorTimer = setTimeout(function() {
errornotification.close();
errornotification = null;
},1000);
2014-05-07 21:47:25 +02:00
}
2014-12-10 15:16:07 +01:00
if (pendingAuth) {
2014-11-12 14:21:59 +01:00
ws.send(JSON.stringify({auth:auth_tokens.access_token}));
2014-12-10 15:16:07 +01:00
} else {
completeConnection();
2014-05-08 15:15:54 +02:00
}
2014-05-07 21:47:25 +02:00
}
ws.onmessage = function(event) {
var msg = JSON.parse(event.data);
2014-12-10 15:16:07 +01:00
if (pendingAuth && msg.auth == "ok") {
pendingAuth = false;
completeConnection();
} else if (msg.topic) {
2014-05-08 15:15:54 +02:00
for (var t in subscriptions) {
2014-08-08 01:01:35 +02: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 15:15:54 +02:00
}
}
}
2014-05-07 21:47:25 +02:00
}
}
};
ws.onclose = function() {
if (errornotification == null) {
errornotification = RED.notify("<b>Error</b>: Lost connection to server","error",true);
} else if (clearErrorTimer) {
clearTimeout(clearErrorTimer);
clearErrorTimer = null;
2014-05-07 21:47:25 +02:00
}
setTimeout(connectWS,1000);
}
}
2014-05-08 15:15:54 +02:00
2014-05-07 21:47:25 +02:00
function subscribe(topic,callback) {
if (subscriptions[topic] == null) {
subscriptions[topic] = [];
}
subscriptions[topic].push(callback);
2014-05-08 15:15:54 +02:00
if (ws && ws.readyState == 1) {
ws.send(JSON.stringify({subscribe:topic}));
}
2014-05-07 21:47:25 +02:00
}
function unsubscribe(topic,callback) {
if (subscriptions[topic]) {
for (var i=0;i<subscriptions[topic].length;i++) {
if (subscriptions[topic][i] === callback) {
subscriptions[topic].splice(i,1);
break;
}
}
if (subscriptions[topic].length === 0) {
delete subscriptions[topic];
}
}
}
2014-05-08 15:15:54 +02:00
2014-05-07 21:47:25 +02:00
return {
2014-05-08 15:15:54 +02:00
connect: connectWS,
subscribe: subscribe,
unsubscribe:unsubscribe
2014-05-07 21:47:25 +02:00
}
2014-08-08 01:01:35 +02:00
})();