1 /*jslint regexp: true, confusion: true, undef: false, node: true, sloppy: true, nomen: true, plusplus: true, maxerr: 50, indent: 4 */
3 var tls
= require('tls'),
5 http
= require('http'),
6 https
= require('https'),
9 ws
= require('socket.io'),
10 _
= require('./lib/underscore.min.js'),
11 starttls
= require('./lib/starttls.js');
15 * Find a config file in the following order:
16 * - /etc/kiwi/config.json
19 var config
= null, config_filename
= 'config.json';
20 var config_dirs
= ['/etc/kiwiirc/', __dirname
+ '/'];
21 for(var i
in config_dirs
){
23 if(fs
.lstatSync(config_dirs
[i
] + config_filename
).isDirectory() === false){
24 config
= JSON
.parse(fs
.readFileSync(config_dirs
[i
] + config_filename
, 'ascii'));
25 console
.log('Using config file ' + config_dirs
[i
] + config_filename
);
34 console
.log('Couldn\'t find a config file!');
42 * Some process changes
44 process
.title
= 'kiwiirc';
45 function changeUser(){
46 if(typeof config
.group
!== 'undefined' && config
.group
!== ''){
48 process
.setgid(config
.group
);
51 console
.log('Failed to set gid: ' + err
);
56 if(typeof config
.user
!== 'undefined' && config
.user
!== ''){
58 process
.setuid(config
.user
);
61 console
.log('Failed to set uid: ' + err
);
69 * And now KiwiIRC, the server :)
76 RPL_WHOISSERVER
: '312',
77 RPL_WHOISOPERATOR
: '313',
79 RPL_ENDOFWHOIS
: '318',
80 RPL_WHOISCHANNELS
: '319',
83 RPL_ENDOFNAMES
: '366',
85 RPL_WHOISMODES
: '379',
86 ERR_NOSUCHNICK
: '401',
87 ERR_LINKCHANNEL
: '470',
92 var parseIRCMessage = function (websocket
, ircSocket
, data
) {
93 /*global ircSocketDataHandler */
94 var msg
, regex
, opts
, options
, opt
, i
, j
, matches
, nick
, users
, chan
, params
, prefix
, prefixes
, nicklist
, caps
, rtn
;
95 regex
= /^(?::(?:([a-z0-9\x5B-\x60\x7B-\x7D\.\-]+)|([a-z0-9\x5B-\x60\x7B-\x7D\.\-]+)!([a-z0-9~\.\-_|]+)@([a-z0-9\.\-:]+)) )?([a-z0-9]+)(?:(?: ([^:]+))?(?: :(.+))?)$/i;
96 msg
= regex
.exec(data
);
104 params
: msg
[6] || '',
105 trailing
: (msg
[7]) ? msg
[7].trim() : ''
107 switch (msg
.command
.toUpperCase()) {
109 ircSocket
.write('PONG ' + msg
.trailing
+ '\r\n');
111 case ircNumerics
.RPL_WELCOME
:
112 if (ircSocket
.IRC
.CAP
.negotiating
) {
113 ircSocket
.IRC
.CAP
.negotiating
= false;
114 ircSocket
.IRC
.CAP
.enabled
= [];
115 ircSocket
.IRC
.CAP
.requested
= [];
117 websocket
.emit('message', {event
: 'connect', connected
: true, host
: null});
119 case ircNumerics
.RPL_ISUPPORT
:
120 opts
= msg
.params
.split(" ");
122 for (i
= 0; i
< opts
.length
; i
++) {
123 opt
= opts
[i
].split("=", 2);
124 opt
[0] = opt
[0].toUpperCase();
125 ircSocket
.IRC
.options
[opt
[0]] = opt
[1] || true;
126 if (_
.include(['NETWORK', 'PREFIX', 'CHANTYPES'], opt
[0])) {
127 if (opt
[0] === 'PREFIX') {
128 regex
= /\(([^)]*)\)(.*)/;
129 matches
= regex
.exec(opt
[1]);
130 if ((matches
) && (matches
.length
=== 3)) {
131 ircSocket
.IRC
.options
[opt
[0]] = [];
132 for (j
= 0; j
< matches
[2].length
; j
++) {
133 //ircSocket.IRC.options[opt[0]][matches[2].charAt(j)] = matches[1].charAt(j);
134 ircSocket
.IRC
.options
[opt
[0]].push({symbol
: matches
[2].charAt(j
), mode
: matches
[1].charAt(j
)});
135 //console.log({symbol: matches[2].charAt(j), mode: matches[1].charAt(j)});
137 console
.log(ircSocket
.IRC
.options
);
142 websocket
.emit('message', {event
: 'options', server
: '', "options": ircSocket
.IRC
.options
});
144 case ircNumerics
.RPL_WHOISUSER
:
145 case ircNumerics
.RPL_WHOISSERVER
:
146 case ircNumerics
.RPL_WHOISOPERATOR
:
147 case ircNumerics
.RPL_ENDOFWHOIS
:
148 case ircNumerics
.RPL_WHOISCHANNELS
:
149 case ircNumerics
.RPL_WHOISMODES
:
150 websocket
.emit('message', {event
: 'whois', server
: '', nick
: msg
.params
.split(" ", 3)[1], "msg": msg
.trailing
});
152 case ircNumerics
.RPL_WHOISIDLE
:
153 params
= msg
.params
.split(" ", 4);
154 rtn
= {event
: 'whois', server
: '', nick
: params
[1], idle
: params
[2]};
156 rtn
.logon
= params
[3];
158 websocket
.emit('message', rtn
);
160 case ircNumerics
.RPL_MOTD
:
161 websocket
.emit('message', {event
: 'motd', server
: '', "msg": msg
.trailing
});
163 case ircNumerics
.RPL_NAMEREPLY
:
164 params
= msg
.params
.split(" ");
167 users
= msg
.trailing
.split(" ");
168 prefixes
= _
.values(ircSocket
.IRC
.options
.PREFIX
);
171 _
.each(users
, function (user
) {
172 if (_
.include(prefix
, user
.charAt(0))) {
173 prefix
= user
.charAt(0);
174 user
= user
.substring(1);
175 nicklist
[user
] = prefix
;
180 websocket
.emit('message', {event
: 'userlist', server
: '', "users": nicklist
, channel
: chan
});
186 websocket
.emit('message', {event
: 'userlist', server
: '', "users": nicklist
, channel
: chan
});
191 case ircNumerics
.RPL_ENDOFNAMES
:
192 websocket
.emit('message', {event
: 'userlist_end', server
: '', channel
: msg
.params
.split(" ")[1]});
194 case ircNumerics
.ERR_LINKCHANNEL
:
195 params
= msg
.params
.split(" ");
196 websocket
.emit('message', {event
: 'channel_redirect', from: params
[1], to
: params
[2]});
198 case ircNumerics
.ERR_NOSUCHNICK
:
202 websocket
.emit('message', {event
: 'join', nick
: msg
.nick
, ident
: msg
.ident
, hostname
: msg
.hostname
, channel
: msg
.trailing
});
203 if (msg
.nick
=== ircSocket
.IRC
.nick
) {
204 ircSocket
.write('NAMES ' + msg
.trailing
+ '\r\n');
208 websocket
.emit('message', {event
: 'part', nick
: msg
.nick
, ident
: msg
.ident
, hostname
: msg
.hostname
, channel
: msg
.params
.trim(), message
: msg
.trailing
});
211 params
= msg
.params
.split(" ");
212 websocket
.emit('message', {event
: 'kick', kicked
: params
[1], nick
: msg
.nick
, ident
: msg
.ident
, hostname
: msg
.hostname
, channel
: params
[0].trim(), message
: msg
.trailing
});
215 websocket
.emit('message', {event
: 'quit', nick
: msg
.nick
, ident
: msg
.ident
, hostname
: msg
.hostname
, message
: msg
.trailing
});
218 if ((msg
.trailing
.charAt(0) === '\001') && (msg
.trailing
.charAt(msg
.trailing
.length
- 1) === '\001')) {
219 // It's a CTCP response
220 websocket
.emit('message', {event
: 'ctcp_response', nick
: msg
.nick
, ident
: msg
.ident
, hostname
: msg
.hostname
, channel
: msg
.params
.trim(), msg
: msg
.trailing
.substr(1, msg
.trailing
.length
- 2)});
222 websocket
.emit('message', {event
: 'notice', nick
: msg
.nick
, ident
: msg
.ident
, hostname
: msg
.hostname
, channel
: msg
.params
.trim(), msg
: msg
.trailing
});
226 websocket
.emit('message', {event
: 'nick', nick
: msg
.nick
, ident
: msg
.ident
, hostname
: msg
.hostname
, newnick
: msg
.trailing
});
229 websocket
.emit('message', {event
: 'topic', nick
: msg
.nick
, channel
: msg
.params
, topic
: msg
.trailing
});
231 case ircNumerics
.RPL_TOPIC
:
232 websocket
.emit('message', {event
: 'topic', nick
: '', channel
: msg
.params
.split(" ")[1], topic
: msg
.trailing
});
235 opts
= msg
.params
.split(" ");
236 params
= {event
: 'mode', nick
: msg
.nick
};
237 switch (opts
.length
) {
239 params
.effected_nick
= opts
[0];
240 params
.mode
= msg
.trailing
;
243 params
.channel
= opts
[0];
244 params
.mode
= opts
[1];
247 params
.channel
= opts
[0];
248 params
.mode
= opts
[1];
249 params
.effected_nick
= opts
[2];
252 websocket
.emit('message', params
);
255 if ((msg
.trailing
.charAt(0) === '\001') && (msg
.trailing
.charAt(msg
.trailing
.length
- 1) === '\001')) {
256 // It's a CTCP request
257 if (msg
.trailing
.substr(1, 6) === 'ACTION') {
258 websocket
.emit('message', {event
: 'action', nick
: msg
.nick
, ident
: msg
.ident
, hostname
: msg
.hostname
, channel
: msg
.params
.trim(), msg
: msg
.trailing
.substr(7, msg
.trailing
.length
- 2)});
259 } else if (msg
.trailing
.substr(1, 7) === 'VERSION') {
260 ircSocket
.write('NOTICE ' + msg
.nick
+ ' :\001VERSION KiwiIRC\001\r\n');
262 websocket
.emit('message', {event
: 'ctcp_request', nick
: msg
.nick
, ident
: msg
.ident
, hostname
: msg
.hostname
, channel
: msg
.params
.trim(), msg
: msg
.trailing
.substr(1, msg
.trailing
.length
- 2)});
265 websocket
.emit('message', {event
: 'msg', nick
: msg
.nick
, ident
: msg
.ident
, hostname
: msg
.hostname
, channel
: msg
.params
.trim(), msg
: msg
.trailing
});
269 caps
= config
.cap_options
;
270 options
= msg
.trailing
.split(" ");
271 switch (_
.first(msg
.params
.split(" "))) {
274 _
.each(_
.intersect(caps
, options
), function (cap
) {
279 ircSocket
.IRC
.CAP
.requested
.push(cap
);
281 if (opts
.length
> 0) {
282 ircSocket
.write('CAP REQ :' + opts
+ '\r\n');
284 ircSocket
.write('CAP END\r\n');
287 /*if (_.include(options, 'tls')) {
288 ircSocket.write('STARTTLS\r\n');
289 ircSocket.IRC.CAP.requested.push('tls');
293 _
.each(options
, function (cap
) {
294 ircSocket
.IRC
.CAP
.enabled
.push(cap
);
296 if (_
.last(msg
.params
.split(" ")) !== '*') {
297 ircSocket
.IRC
.CAP
.requested
= [];
298 ircSocket
.IRC
.CAP
.negotiating
= false;
299 ircSocket
.write('CAP END\r\n');
303 ircSocket
.IRC
.CAP
.requested
= [];
304 ircSocket
.IRC
.CAP
.negotiating
= false;
305 ircSocket
.write('CAP END\r\n');
309 /*case ircNumerics.RPL_STARTTLS:
312 listeners = ircSocket.listeners('data');
313 ircSocket.removeAllListeners('data');
314 ssl_socket = starttls(ircSocket, {}, function () {
315 ssl_socket.on("data", function (data) {
316 ircSocketDataHandler(data, websocket, ssl_socket);
318 ircSocket = ssl_socket;
320 _.each(listeners, function (listener) {
321 ircSocket.addListener('data', listener);
324 //console.log(ircSocket);
331 console
.log("Unknown command.\r\n");
335 var ircSocketDataHandler = function (data
, websocket
, ircSocket
) {
337 if ((ircSocket
.holdLast
) && (ircSocket
.held
!== '')) {
338 data
= ircSocket
.held
+ data
;
339 ircSocket
.holdLast
= false;
342 if (data
.substr(-2) === '\r\n') {
343 ircSocket
.holdLast
= true;
345 data
= data
.split("\r\n");
346 for (i
= 0; i
< data
.length
; i
++) {
348 if ((ircSocket
.holdLast
) && (i
=== data
.length
- 1)) {
349 ircSocket
.held
= data
[i
];
352 console
.log("->" + data
[i
]);
353 parseIRCMessage(websocket
, ircSocket
, data
[i
]);
358 if (config
.handle_http
) {
359 var fileServer
= new (require('node-static').Server
)(__dirname
+ config
.public_http
);
360 var jade
= require('jade');
363 var httpHandler = function (request
, response
) {
364 var uri
, subs
, useragent
, agent
, server_set
, server
, nick
, debug
, touchscreen
;
365 if (config
.handle_http
) {
366 uri
= url
.parse(request
.url
);
367 subs
= uri
.pathname
.substr(0, 4);
368 if ((subs
=== '/js/') || (subs
=== '/css') || (subs
=== '/img')) {
369 request
.addListener('end', function () {
370 fileServer
.serve(request
, response
);
372 } else if (uri
.pathname
=== '/') {
373 useragent
= (response
.headers
) ? response
.headers
['user-agent']: '';
374 if (useragent
.indexOf('android') !== -1) {
377 } else if (useragent
.indexOf('iphone') !== -1) {
380 } else if (useragent
.indexOf('ipad') !== -1) {
383 } else if (useragent
.indexOf('ipod') !== -1) {
391 server_set
= (uri
.query
.server
!== '');
392 server
= uri
.query
.server
|| 'irc.anonnet.org';
393 nick
= uri
.query
.nick
|| '';
394 debug
= (uri
.query
.debug
!== '');
396 server
= 'irc.anonnet.org';
399 response
.setHeader('Connection', 'close');
400 response
.setHeader('X-Generated-By', 'KiwiIRC');
401 jade
.renderFile(__dirname
+ '/client/index.html.jade', { locals
: { "touchscreen": touchscreen
, "debug": debug
, "server_set": server_set
, "server": server
, "nick": nick
, "agent": agent
, "config": config
}}, function (err
, html
) {
403 response
.write(html
);
405 response
.statusCode
= 500;
409 } else if (uri
.pathname
.substr(0, 10) === '/socket.io') {
412 response
.statusCode
= 404;
418 //setup websocket listener
419 if (config
.listen_ssl
) {
420 var httpServer
= https
.createServer({key
: fs
.readFileSync(__dirname
+ '/' + config
.ssl_key
), cert
: fs
.readFileSync(__dirname
+ '/' + config
.ssl_cert
)}, httpHandler
);
421 var io
= ws
.listen(httpServer
, {secure
: true});
422 httpServer
.listen(config
.port
, config
.bind_address
);
424 var httpServer
= http
.createServer(httpHandler
);
425 var io
= ws
.listen(httpServer
, {secure
: false});
426 httpServer
.listen(config
.port
, config
.bind_address
);
429 // Now we're listening on the network, set our UID/GIDs if required
432 io
.of('/kiwi').on('connection', function (websocket
) {
433 websocket
.on('irc connect', function (nick
, host
, port
, ssl
, callback
) {
435 //setup IRC connection
437 ircSocket
= net
.createConnection(port
, host
);
439 ircSocket
= tls
.connect(port
, host
);
441 ircSocket
.setEncoding('ascii');
442 ircSocket
.IRC
= {options
: {}, CAP
: {negotiating
: true, requested
: [], enabled
: []}};
443 websocket
.ircSocket
= ircSocket
;
444 ircSocket
.holdLast
= false;
447 ircSocket
.on('data', function (data
) {
448 ircSocketDataHandler(data
, websocket
, ircSocket
);
451 ircSocket
.IRC
.nick
= nick
;
452 // Send the login data
453 ircSocket
.write('CAP LS\r\n');
454 ircSocket
.write('NICK ' + nick
+ '\r\n');
455 ircSocket
.write('USER ' + nick
+ '_kiwi 0 0 :' + nick
+ '\r\n');
457 if ((callback
) && (typeof (callback
) === 'function')) {
461 websocket
.on('message', function (msg
, callback
) {
464 msg
.data
= JSON
.parse(msg
.data
);
465 args
= msg
.data
.args
;
466 switch (msg
.data
.method
) {
468 if ((args
.target
) && (args
.msg
)) {
469 websocket
.ircSocket
.write('PRIVMSG ' + args
.target
+ ' :' + args
.msg
+ '\r\n');
473 if ((args
.target
) && (args
.msg
)) {
474 websocket
.ircSocket
.write('PRIVMSG ' + args
.target
+ ' :\ 1ACTION ' + args
.msg
+ '\ 1\r\n');
478 websocket
.ircSocket
.write(args
.data
+ '\r\n');
482 _
.each(args
.channel
.split(","), function (chan
) {
483 websocket
.ircSocket
.write('JOIN ' + chan
+ '\r\n');
488 websocket
.ircSocket
.end('QUIT :' + args
.message
+ '\r\n');
489 websocket
.sentQUIT
= true;
490 websocket
.ircSocket
.destroySoon();
491 websocket
.disconnect();
494 if ((args
.target
) && (args
.msg
)) {
495 websocket
.ircSocket
.write('NOTICE ' + args
.target
+ ' :' + args
.msg
+ '\r\n');
500 if ((callback
) && (typeof (callback
) === 'function')) {
504 console
.log("Caught error: " + e
);
507 websocket
.on('disconnect', function () {
508 if ((!websocket
.sentQUIT
) && (websocket
.ircSocket
)) {
509 websocket
.ircSocket
.end('QUIT :' + config
.quit_message
+ '\r\n');
510 websocket
.sentQUIT
= true;
511 websocket
.ircSocket
.destroySoon();