2014-09-13 21:29:45 +00:00
|
|
|
var _ = require("lodash");
|
2014-09-16 20:06:13 +00:00
|
|
|
var Chan = require("./models/chan");
|
2014-09-15 21:13:03 +00:00
|
|
|
var crypto = require("crypto");
|
2014-10-11 22:47:24 +00:00
|
|
|
var fs = require("fs");
|
2014-10-11 17:33:28 +00:00
|
|
|
var identd = require("./identd");
|
2014-09-16 19:47:01 +00:00
|
|
|
var log = require("./log");
|
2014-09-13 21:29:45 +00:00
|
|
|
var net = require("net");
|
|
|
|
var Msg = require("./models/msg");
|
|
|
|
var Network = require("./models/network");
|
|
|
|
var slate = require("slate-irc");
|
|
|
|
var tls = require("tls");
|
2014-10-01 15:59:35 +00:00
|
|
|
var Helper = require("./helper");
|
2014-09-13 21:29:45 +00:00
|
|
|
|
|
|
|
module.exports = Client;
|
|
|
|
|
|
|
|
var id = 0;
|
|
|
|
var events = [
|
|
|
|
"ctcp",
|
|
|
|
"error",
|
|
|
|
"join",
|
|
|
|
"kick",
|
|
|
|
"mode",
|
|
|
|
"motd",
|
|
|
|
"message",
|
2014-09-27 19:17:05 +00:00
|
|
|
"link",
|
2014-09-13 21:29:45 +00:00
|
|
|
"names",
|
|
|
|
"nick",
|
|
|
|
"notice",
|
|
|
|
"part",
|
|
|
|
"quit",
|
|
|
|
"topic",
|
|
|
|
"welcome",
|
|
|
|
"whois"
|
|
|
|
];
|
|
|
|
var inputs = [
|
|
|
|
"action",
|
|
|
|
"connect",
|
|
|
|
"invite",
|
|
|
|
"join",
|
|
|
|
"kick",
|
|
|
|
"mode",
|
|
|
|
"msg",
|
|
|
|
"nick",
|
|
|
|
"notice",
|
|
|
|
"part",
|
|
|
|
"quit",
|
|
|
|
"raw",
|
2014-12-11 22:34:22 +00:00
|
|
|
"services",
|
2014-09-13 21:29:45 +00:00
|
|
|
"topic",
|
|
|
|
"whois"
|
|
|
|
];
|
|
|
|
|
2014-09-16 19:47:01 +00:00
|
|
|
function Client(sockets, name, config) {
|
2014-09-13 21:29:45 +00:00
|
|
|
_.merge(this, {
|
2014-09-21 16:46:43 +00:00
|
|
|
activeChannel: -1,
|
2014-09-13 21:29:45 +00:00
|
|
|
config: config,
|
|
|
|
id: id++,
|
2014-09-16 19:47:01 +00:00
|
|
|
name: name,
|
2014-09-13 21:29:45 +00:00
|
|
|
networks: [],
|
|
|
|
sockets: sockets
|
|
|
|
});
|
2014-09-15 21:13:03 +00:00
|
|
|
var client = this;
|
|
|
|
crypto.randomBytes(48, function(err, buf) {
|
|
|
|
client.token = buf.toString("hex");
|
|
|
|
});
|
2014-09-13 21:29:45 +00:00
|
|
|
if (config) {
|
|
|
|
var delay = 0;
|
|
|
|
(config.networks || []).forEach(function(n) {
|
|
|
|
setTimeout(function() {
|
|
|
|
client.connect(n);
|
|
|
|
}, delay);
|
|
|
|
delay += 1000;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Client.prototype.emit = function(event, data) {
|
|
|
|
if (this.sockets !== null) {
|
|
|
|
this.sockets.in(this.id).emit(event, data);
|
|
|
|
}
|
2014-10-14 19:25:04 +00:00
|
|
|
var config = this.config || {};
|
|
|
|
if (config.log === true) {
|
2015-09-30 22:39:57 +00:00
|
|
|
if (event === "msg") {
|
2014-09-16 19:47:01 +00:00
|
|
|
var target = this.find(data.chan);
|
|
|
|
if (target) {
|
2014-09-16 20:06:13 +00:00
|
|
|
var chan = target.chan.name;
|
2015-09-30 22:39:57 +00:00
|
|
|
if (target.chan.type === Chan.Type.LOBBY) {
|
2014-09-16 20:06:13 +00:00
|
|
|
chan = target.network.host;
|
|
|
|
}
|
|
|
|
log.write(
|
|
|
|
this.name,
|
|
|
|
target.network.host,
|
|
|
|
chan,
|
|
|
|
data.msg
|
|
|
|
);
|
2014-09-16 19:47:01 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2014-09-13 17:18:42 +00:00
|
|
|
};
|
2014-09-13 21:29:45 +00:00
|
|
|
|
|
|
|
Client.prototype.find = function(id) {
|
|
|
|
var network = null;
|
|
|
|
var chan = null;
|
|
|
|
for (var i in this.networks) {
|
|
|
|
var n = this.networks[i];
|
|
|
|
chan = _.find(n.channels, {id: id});
|
|
|
|
if (chan) {
|
|
|
|
network = n;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (network && chan) {
|
|
|
|
return {
|
|
|
|
network: network,
|
|
|
|
chan: chan
|
|
|
|
};
|
|
|
|
} else {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
Client.prototype.connect = function(args) {
|
2014-10-11 06:17:41 +00:00
|
|
|
var config = Helper.getConfig();
|
2014-09-13 21:29:45 +00:00
|
|
|
var client = this;
|
|
|
|
var server = {
|
2014-10-11 20:44:56 +00:00
|
|
|
name: args.name || "",
|
2014-09-13 21:29:45 +00:00
|
|
|
host: args.host || "irc.freenode.org",
|
|
|
|
port: args.port || (args.tls ? 6697 : 6667),
|
|
|
|
rejectUnauthorized: false
|
|
|
|
};
|
|
|
|
|
2014-10-11 17:33:28 +00:00
|
|
|
if (config.bind) {
|
2014-10-11 06:17:41 +00:00
|
|
|
server.localAddress = config.bind;
|
2015-09-30 22:39:57 +00:00
|
|
|
if (args.tls) {
|
2014-10-11 06:17:41 +00:00
|
|
|
var socket = net.connect(server);
|
|
|
|
server.socket = socket;
|
|
|
|
}
|
|
|
|
}
|
2014-11-09 16:01:22 +00:00
|
|
|
|
2014-09-13 21:29:45 +00:00
|
|
|
var stream = args.tls ? tls.connect(server) : net.connect(server);
|
2014-11-09 16:01:22 +00:00
|
|
|
|
2014-10-16 21:33:07 +00:00
|
|
|
stream.on("error", function(e) {
|
2014-09-13 21:29:45 +00:00
|
|
|
console.log("Client#connect():\n" + e);
|
|
|
|
stream.end();
|
|
|
|
var msg = new Msg({
|
|
|
|
type: Msg.Type.ERROR,
|
|
|
|
text: "Connection error."
|
|
|
|
});
|
|
|
|
client.emit("msg", {
|
|
|
|
msg: msg
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
var nick = args.nick || "shout-user";
|
2015-09-30 22:39:57 +00:00
|
|
|
var username = args.username || nick.replace(/[^a-zA-Z0-9]/g, "");
|
2014-09-13 21:29:45 +00:00
|
|
|
var realname = args.realname || "Shout User";
|
|
|
|
|
|
|
|
var irc = slate(stream);
|
2014-10-11 17:33:28 +00:00
|
|
|
identd.hook(stream, username);
|
2014-09-13 21:29:45 +00:00
|
|
|
|
|
|
|
if (args.password) {
|
|
|
|
irc.pass(args.password);
|
|
|
|
}
|
|
|
|
|
|
|
|
irc.me = nick;
|
|
|
|
irc.nick(nick);
|
2014-09-24 11:03:58 +00:00
|
|
|
irc.user(username, realname);
|
2014-09-13 21:29:45 +00:00
|
|
|
|
|
|
|
var network = new Network({
|
|
|
|
name: server.name,
|
2014-10-11 20:44:56 +00:00
|
|
|
host: server.host,
|
|
|
|
port: server.port,
|
2014-10-11 22:47:24 +00:00
|
|
|
tls: !!args.tls,
|
2014-10-11 20:44:56 +00:00
|
|
|
password: args.password,
|
|
|
|
username: username,
|
|
|
|
realname: realname,
|
2014-11-09 16:01:22 +00:00
|
|
|
commands: args.commands
|
2014-09-13 21:29:45 +00:00
|
|
|
});
|
|
|
|
|
2014-10-11 20:44:56 +00:00
|
|
|
network.irc = irc;
|
|
|
|
|
2014-09-13 21:29:45 +00:00
|
|
|
client.networks.push(network);
|
|
|
|
client.emit("network", {
|
|
|
|
network: network
|
|
|
|
});
|
|
|
|
|
|
|
|
events.forEach(function(plugin) {
|
|
|
|
var path = "./plugins/irc-events/" + plugin;
|
|
|
|
require(path).apply(client, [
|
|
|
|
irc,
|
|
|
|
network
|
|
|
|
]);
|
|
|
|
});
|
|
|
|
|
|
|
|
irc.once("welcome", function() {
|
|
|
|
var delay = 1000;
|
|
|
|
var commands = args.commands;
|
|
|
|
if (Array.isArray(commands)) {
|
|
|
|
commands.forEach(function(cmd) {
|
|
|
|
setTimeout(function() {
|
|
|
|
client.input({
|
|
|
|
target: network.channels[0].id,
|
|
|
|
text: cmd
|
|
|
|
});
|
|
|
|
}, delay);
|
|
|
|
delay += 1000;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
setTimeout(function() {
|
|
|
|
irc.write("PING " + network.host);
|
|
|
|
}, delay);
|
|
|
|
});
|
|
|
|
|
|
|
|
irc.once("pong", function() {
|
|
|
|
var join = (args.join || "");
|
|
|
|
if (join) {
|
|
|
|
join = join.replace(/\,/g, " ").split(/\s+/g);
|
|
|
|
irc.join(join);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
Client.prototype.input = function(data) {
|
|
|
|
var client = this;
|
2014-09-27 15:56:23 +00:00
|
|
|
var text = data.text.trim();
|
2014-09-13 21:29:45 +00:00
|
|
|
var target = client.find(data.target);
|
|
|
|
if (text.charAt(0) !== "/") {
|
|
|
|
text = "/say " + text;
|
|
|
|
}
|
|
|
|
var args = text.split(" ");
|
|
|
|
var cmd = args.shift().replace("/", "").toLowerCase();
|
|
|
|
_.each(inputs, function(plugin) {
|
|
|
|
try {
|
|
|
|
var path = "./plugins/inputs/" + plugin;
|
|
|
|
var fn = require(path);
|
|
|
|
fn.apply(client, [
|
|
|
|
target.network,
|
|
|
|
target.chan,
|
|
|
|
cmd,
|
|
|
|
args
|
|
|
|
]);
|
|
|
|
} catch (e) {
|
|
|
|
console.log(path + ": " + e);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
Client.prototype.more = function(data) {
|
|
|
|
var client = this;
|
|
|
|
var target = client.find(data.target);
|
|
|
|
if (!target) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
var chan = target.chan;
|
|
|
|
var count = chan.messages.length - (data.count || 0);
|
|
|
|
var messages = chan.messages.slice(Math.max(0, count - 100), count);
|
|
|
|
client.emit("more", {
|
|
|
|
chan: chan.id,
|
|
|
|
messages: messages
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2014-09-21 16:46:43 +00:00
|
|
|
Client.prototype.open = function(data) {
|
|
|
|
var target = this.find(data);
|
|
|
|
if (target) {
|
|
|
|
target.chan.unread = 0;
|
|
|
|
this.activeChannel = target.chan.id;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2014-09-24 19:42:36 +00:00
|
|
|
Client.prototype.sort = function(data) {
|
|
|
|
var self = this;
|
|
|
|
|
|
|
|
var type = data.type;
|
|
|
|
var order = data.order || [];
|
2015-09-30 22:39:57 +00:00
|
|
|
var sorted = [];
|
2014-09-24 19:42:36 +00:00
|
|
|
|
|
|
|
switch (type) {
|
|
|
|
case "networks":
|
|
|
|
_.each(order, function(i) {
|
|
|
|
var find = _.find(self.networks, {id: i});
|
|
|
|
if (find) {
|
|
|
|
sorted.push(find);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
self.networks = sorted;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case "channels":
|
|
|
|
var target = data.target;
|
|
|
|
var network = _.find(self.networks, {id: target});
|
|
|
|
if (!network) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
_.each(order, function(i) {
|
|
|
|
var find = _.find(network.channels, {id: i});
|
|
|
|
if (find) {
|
|
|
|
sorted.push(find);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
network.channels = sorted;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2014-09-13 21:29:45 +00:00
|
|
|
Client.prototype.quit = function() {
|
2014-09-29 15:49:38 +00:00
|
|
|
var sockets = this.sockets.sockets;
|
|
|
|
var room = sockets.adapter.rooms[this.id] || [];
|
|
|
|
for (var user in room) {
|
|
|
|
var socket = sockets.adapter.nsp.connected[user];
|
|
|
|
if (socket) {
|
|
|
|
socket.disconnect();
|
|
|
|
}
|
|
|
|
}
|
2014-09-13 21:29:45 +00:00
|
|
|
this.networks.forEach(function(network) {
|
|
|
|
var irc = network.irc;
|
|
|
|
if (network.connected) {
|
|
|
|
irc.quit();
|
|
|
|
} else {
|
|
|
|
irc.stream.end();
|
|
|
|
}
|
|
|
|
});
|
2014-09-13 17:18:42 +00:00
|
|
|
};
|
2014-10-11 20:44:56 +00:00
|
|
|
|
2014-10-12 00:20:30 +00:00
|
|
|
var timer;
|
|
|
|
Client.prototype.save = function(force) {
|
|
|
|
var client = this;
|
2014-10-12 05:15:03 +00:00
|
|
|
var config = Helper.getConfig();
|
|
|
|
|
2015-09-30 22:39:57 +00:00
|
|
|
if (config.public) {
|
2014-10-12 05:15:03 +00:00
|
|
|
return;
|
|
|
|
}
|
2014-11-09 16:01:22 +00:00
|
|
|
|
2014-10-12 00:20:30 +00:00
|
|
|
if (!force) {
|
|
|
|
clearTimeout(timer);
|
|
|
|
timer = setTimeout(function() {
|
|
|
|
client.save(true);
|
|
|
|
}, 1000);
|
|
|
|
return;
|
|
|
|
}
|
2014-11-09 16:01:22 +00:00
|
|
|
|
2014-10-11 22:47:24 +00:00
|
|
|
var name = this.name;
|
2014-10-14 20:53:26 +00:00
|
|
|
var path = Helper.HOME + "/users/" + name + ".json";
|
2014-11-09 16:01:22 +00:00
|
|
|
|
2014-10-11 22:47:24 +00:00
|
|
|
var networks = _.map(
|
|
|
|
this.networks,
|
|
|
|
function(n) {
|
|
|
|
return n.export();
|
|
|
|
}
|
|
|
|
);
|
2014-11-09 16:01:22 +00:00
|
|
|
|
2014-10-11 22:47:24 +00:00
|
|
|
var json = {};
|
|
|
|
fs.readFile(path, "utf-8", function(err, data) {
|
|
|
|
if (err) {
|
|
|
|
console.log(err);
|
|
|
|
return;
|
|
|
|
}
|
2014-11-09 16:01:22 +00:00
|
|
|
|
2014-10-11 22:47:24 +00:00
|
|
|
try {
|
|
|
|
json = JSON.parse(data);
|
|
|
|
json.networks = networks;
|
2015-09-30 22:39:57 +00:00
|
|
|
} catch (e) {
|
2014-10-11 22:47:24 +00:00
|
|
|
console.log(e);
|
2014-10-12 00:20:30 +00:00
|
|
|
return;
|
2014-10-11 22:47:24 +00:00
|
|
|
}
|
2014-11-09 16:01:22 +00:00
|
|
|
|
2014-10-11 22:47:24 +00:00
|
|
|
fs.writeFile(
|
|
|
|
path,
|
|
|
|
JSON.stringify(json, null, " "),
|
|
|
|
{mode: "0777"},
|
|
|
|
function(err) {
|
|
|
|
if (err) {
|
|
|
|
console.log(err);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
|
|
|
});
|
2014-10-11 20:44:56 +00:00
|
|
|
};
|