2013-10-30 20:38:23 +00:00
|
|
|
/**
|
2014-05-29 18:31:36 +01:00
|
|
|
* Copyright 2013,2014 IBM Corp.
|
2013-10-30 20:38:23 +00: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.
|
|
|
|
**/
|
|
|
|
|
2014-05-29 18:31:36 +01:00
|
|
|
module.exports = function(RED) {
|
|
|
|
"use strict";
|
|
|
|
var XMPP = require('simple-xmpp');
|
|
|
|
|
|
|
|
try { var xmppkey = RED.settings.xmpp || require(process.env.NODE_RED_HOME+"/../xmppkeys.js"); }
|
|
|
|
catch(err) { }
|
2013-10-30 20:38:23 +00:00
|
|
|
|
2014-05-29 18:31:36 +01:00
|
|
|
function XMPPServerNode(n) {
|
2013-11-15 21:28:18 +00:00
|
|
|
RED.nodes.createNode(this,n);
|
|
|
|
this.server = n.server;
|
|
|
|
this.port = n.port;
|
2014-05-29 18:31:36 +01:00
|
|
|
this.nickname = n.nickname;
|
|
|
|
var credentials = RED.nodes.getCredentials(n.id);
|
|
|
|
if (credentials) {
|
|
|
|
this.username = credentials.user;
|
|
|
|
this.password = credentials.password;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
RED.nodes.registerType("xmpp-server",XMPPServerNode);
|
|
|
|
|
|
|
|
var querystring = require('querystring');
|
|
|
|
|
|
|
|
RED.httpAdmin.get('/xmpp-server/:id',function(req,res) {
|
|
|
|
var credentials = RED.nodes.getCredentials(req.params.id);
|
|
|
|
if (credentials) {
|
2014-06-28 23:36:36 +01:00
|
|
|
res.send(JSON.stringify({user:credentials.user,hasPassword:(credentials.password&&credentials.password!=="")}));
|
2014-05-29 18:31:36 +01:00
|
|
|
} else if (xmppkey && xmppkey.jid && xmppkey.password) {
|
|
|
|
RED.nodes.addCredentials(req.params.id,{user:xmppkey.jid, password:xmppkey.password, global:true});
|
|
|
|
credentials = RED.nodes.getCredentials(req.params.id);
|
2014-06-28 23:36:36 +01:00
|
|
|
res.send(JSON.stringify({user:credentials.user,global:credentials.global,hasPassword:(credentials.password&&credentials.password!=="")}));
|
2014-05-29 18:31:36 +01:00
|
|
|
} else {
|
|
|
|
res.send(JSON.stringify({}));
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
RED.httpAdmin.delete('/xmpp-server/:id',function(req,res) {
|
|
|
|
RED.nodes.deleteCredentials(req.params.id);
|
|
|
|
res.send(200);
|
|
|
|
});
|
|
|
|
|
|
|
|
RED.httpAdmin.post('/xmpp-server/:id',function(req,res) {
|
|
|
|
var body = "";
|
|
|
|
req.on('data', function(chunk) {
|
|
|
|
body+=chunk;
|
|
|
|
});
|
|
|
|
req.on('end', function(){
|
|
|
|
var newCreds = querystring.parse(body);
|
|
|
|
var credentials = RED.nodes.getCredentials(req.params.id)||{};
|
2014-06-28 23:36:36 +01:00
|
|
|
if (newCreds.user == null || newCreds.user === "") {
|
2014-05-29 18:31:36 +01:00
|
|
|
delete credentials.user;
|
|
|
|
} else {
|
|
|
|
credentials.user = newCreds.user;
|
|
|
|
}
|
2014-06-28 23:36:36 +01:00
|
|
|
if (newCreds.password === "") {
|
2014-05-29 18:31:36 +01:00
|
|
|
delete credentials.password;
|
|
|
|
} else {
|
|
|
|
credentials.password = newCreds.password||credentials.password;
|
|
|
|
}
|
|
|
|
RED.nodes.addCredentials(req.params.id,credentials);
|
|
|
|
res.send(200);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
function XmppInNode(n) {
|
|
|
|
RED.nodes.createNode(this,n);
|
|
|
|
this.server = n.server;
|
|
|
|
|
|
|
|
this.serverConfig = RED.nodes.getNode(this.server);
|
|
|
|
this.host = this.serverConfig.server;
|
|
|
|
this.port = this.serverConfig.port;
|
|
|
|
this.nick = this.serverConfig.nickname || "Node-RED";
|
|
|
|
this.userid = this.serverConfig.username;
|
|
|
|
this.password = this.serverConfig.password;
|
|
|
|
|
2013-11-15 21:28:18 +00:00
|
|
|
this.join = n.join || false;
|
|
|
|
this.sendAll = n.sendObject;
|
|
|
|
this.to = n.to || "";
|
|
|
|
var node = this;
|
|
|
|
|
2014-05-29 18:31:36 +01:00
|
|
|
var xmpp = new XMPP.SimpleXMPP();
|
2013-11-15 21:28:18 +00:00
|
|
|
|
|
|
|
xmpp.on('online', function() {
|
2014-05-29 18:31:36 +01:00
|
|
|
node.log('connected to '+node.host+":"+node.port);
|
2014-05-31 20:08:15 +01:00
|
|
|
node.status({fill:"green",shape:"dot",text:"connected"});
|
2014-05-29 18:31:36 +01:00
|
|
|
//xmpp.setPresence('online', node.nick+' online');
|
2013-11-15 21:28:18 +00:00
|
|
|
if (node.join) {
|
|
|
|
xmpp.join(node.to+'/'+node.nick);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
xmpp.on('chat', function(from, message) {
|
|
|
|
var msg = { topic:from, payload:message };
|
|
|
|
node.send([msg,null]);
|
|
|
|
});
|
|
|
|
|
|
|
|
xmpp.on('groupchat', function(conference, from, message, stamp) {
|
|
|
|
var msg = { topic:from, payload:message, room:conference };
|
|
|
|
if (from != node.nick) { node.send([msg,null]); }
|
|
|
|
});
|
|
|
|
|
|
|
|
//xmpp.on('chatstate', function(from, state) {
|
|
|
|
//console.log('%s is currently %s', from, state);
|
|
|
|
//var msg = { topic:from, payload:state };
|
|
|
|
//node.send([null,msg]);
|
|
|
|
//});
|
|
|
|
|
|
|
|
xmpp.on('buddy', function(jid, state, statusText) {
|
|
|
|
node.log(jid+" is "+state+" : "+statusText);
|
|
|
|
var msg = { topic:jid, payload: { presence:state, status:statusText} };
|
|
|
|
node.send([null,msg]);
|
|
|
|
});
|
|
|
|
|
|
|
|
xmpp.on('error', function(err) {
|
2014-05-29 18:31:36 +01:00
|
|
|
console.error("error",err);
|
2013-11-15 21:28:18 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
xmpp.on('close', function(err) {
|
|
|
|
node.log('connection closed');
|
2014-05-31 20:08:15 +01:00
|
|
|
node.status({fill:"red",shape:"ring",text:"not connected"});
|
2013-11-15 21:28:18 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
xmpp.on('subscribe', function(from) {
|
|
|
|
xmpp.acceptSubscription(from);
|
|
|
|
});
|
|
|
|
|
2014-05-29 18:31:36 +01:00
|
|
|
// Now actually make the connection
|
|
|
|
try {
|
|
|
|
xmpp.connect({
|
|
|
|
jid : node.userid,
|
|
|
|
password : node.password,
|
|
|
|
host : node.host,
|
|
|
|
port : node.port,
|
|
|
|
skipPresence : true,
|
|
|
|
reconnect : false
|
|
|
|
});
|
|
|
|
} catch(e) {
|
|
|
|
node.error("Bad xmpp configuration");
|
2014-05-31 20:08:15 +01:00
|
|
|
node.status({fill:"red",shape:"ring",text:"not connected"});
|
2014-05-29 18:31:36 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
node.on("close", function(done) {
|
|
|
|
//xmpp.setPresence('offline');
|
|
|
|
if (xmpp.conn) { xmpp.conn.end(); }
|
|
|
|
xmpp = null;
|
|
|
|
done();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
RED.nodes.registerType("xmpp in",XmppInNode);
|
|
|
|
|
|
|
|
function XmppOutNode(n) {
|
|
|
|
RED.nodes.createNode(this,n);
|
|
|
|
this.server = n.server;
|
|
|
|
|
|
|
|
this.serverConfig = RED.nodes.getNode(this.server);
|
|
|
|
this.host = this.serverConfig.server;
|
|
|
|
this.port = this.serverConfig.port;
|
|
|
|
this.nick = this.serverConfig.nickname || "Node-RED";
|
|
|
|
this.userid = this.serverConfig.username;
|
|
|
|
this.password = this.serverConfig.password;
|
|
|
|
|
|
|
|
this.join = n.join || false;
|
|
|
|
this.sendAll = n.sendObject;
|
|
|
|
this.to = n.to || "";
|
|
|
|
var node = this;
|
|
|
|
|
|
|
|
var xmpp = new XMPP.SimpleXMPP();
|
|
|
|
|
|
|
|
xmpp.on('online', function() {
|
|
|
|
node.log('connected to '+node.host+":"+node.port);
|
2014-05-31 20:08:15 +01:00
|
|
|
node.status({fill:"green",shape:"dot",text:"connected"});
|
2014-05-29 18:31:36 +01:00
|
|
|
xmpp.setPresence('online', node.nick+' online');
|
|
|
|
if (node.join) {
|
|
|
|
xmpp.join(node.to+'/'+node.nick);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
xmpp.on('error', function(err) {
|
|
|
|
console.error("error",err);
|
|
|
|
});
|
|
|
|
|
|
|
|
xmpp.on('close', function(err) {
|
|
|
|
node.log('connection closed');
|
2014-05-31 20:08:15 +01:00
|
|
|
node.status({fill:"red",shape:"ring",text:"not connected"});
|
2014-05-29 18:31:36 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
xmpp.on('subscribe', function(from) {
|
|
|
|
xmpp.acceptSubscription(from);
|
|
|
|
});
|
|
|
|
|
|
|
|
// Now actually make the connection
|
|
|
|
try {
|
|
|
|
xmpp.connect({
|
|
|
|
jid : node.userid,
|
|
|
|
password : node.password,
|
|
|
|
host : node.host,
|
|
|
|
port : node.port,
|
|
|
|
skipPresence : true,
|
|
|
|
reconnect : false
|
|
|
|
});
|
|
|
|
} catch(e) {
|
|
|
|
node.error("Bad xmpp configuration");
|
2014-05-31 20:08:15 +01:00
|
|
|
node.status({fill:"red",shape:"ring",text:"not connected"});
|
2014-05-29 18:31:36 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
node.on("input", function(msg) {
|
|
|
|
if (msg.presence) {
|
|
|
|
if (['away', 'dnd', 'xa','chat'].indexOf(msg.presence) > -1 ) {
|
|
|
|
xmpp.setPresence(msg.presence, msg.payload);
|
|
|
|
}
|
|
|
|
else { node.warn("Can't set presence - invalid value"); }
|
2013-11-15 21:28:18 +00:00
|
|
|
}
|
|
|
|
else {
|
2014-05-29 18:31:36 +01:00
|
|
|
var to = msg.topic;
|
2014-06-28 23:36:36 +01:00
|
|
|
if (node.to !== "") { to = node.to; }
|
2014-05-29 18:31:36 +01:00
|
|
|
if (node.sendAll) {
|
|
|
|
xmpp.send(to, JSON.stringify(msg), node.join);
|
|
|
|
}
|
|
|
|
else if (msg.payload) {
|
|
|
|
if (typeof(msg.payload) === "object") {
|
|
|
|
xmpp.send(to, JSON.stringify(msg.payload), node.join);
|
|
|
|
} else {
|
|
|
|
xmpp.send(to, msg.payload.toString(), node.join);
|
|
|
|
}
|
|
|
|
}
|
2013-11-15 21:28:18 +00:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2014-05-29 18:31:36 +01:00
|
|
|
node.on("close", function() {
|
2013-11-15 21:28:18 +00:00
|
|
|
xmpp.setPresence('offline');
|
2014-05-29 18:31:36 +01:00
|
|
|
if (xmpp.conn) { xmpp.conn.end(); }
|
|
|
|
xmpp = null;
|
2013-11-15 21:28:18 +00:00
|
|
|
});
|
2013-10-30 20:38:23 +00:00
|
|
|
}
|
2014-05-29 18:31:36 +01:00
|
|
|
RED.nodes.registerType("xmpp out",XmppOutNode);
|
2013-10-30 20:38:23 +00:00
|
|
|
|
2014-05-29 18:31:36 +01:00
|
|
|
}
|