OmnomIRC.js 6.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233
  1. #!node
  2. var fs = require('fs'),
  3. url = require('url'),
  4. path = require('path'),
  5. vm = require('vm'),
  6. toobusy = function(){return false;},//require('toobusy'),
  7. cluster = require('cluster');
  8. if(cluster.isMaster){
  9. for(var i=0;i<require('os').cpus().length;i++){
  10. cluster.fork();
  11. }
  12. cluster.on('exit', function(worker, code, signal) {
  13. console.log('worker ' + worker.process.pid + ' died');
  14. });
  15. }else{
  16. var RedisStore = require('socket.io/lib/stores/redis'),
  17. redis = require('socket.io/node_modules/redis'),
  18. pub = redis.createClient(),
  19. sub = redis.createClient(),
  20. client = redis.createClient(),
  21. mimeTypes = {
  22. 'html': 'text/html',
  23. 'js': 'text/javascript',
  24. 'css': 'text/css',
  25. 'png': 'image/png',
  26. 'jpg': 'image/jpeg'
  27. },
  28. app = require('http').createServer(function(req,res){
  29. if(toobusy()){
  30. res.writeHead(503,{
  31. 'Content-type': 'text/plain'
  32. });
  33. res.write('503 Server busy.\n');
  34. res.end();
  35. return;
  36. }
  37. req.addListener('end',function(){
  38. logger.debug('served static content for '+req.url);
  39. var uri = url.parse(req.url).pathname,
  40. serveFile = function(filename,req,res){
  41. try{
  42. stats = fs.lstatSync(filename);
  43. }catch(e){
  44. res.writeHead(404,{
  45. 'Content-type': 'text/plain'
  46. });
  47. res.write('404 Not Found.\n');
  48. res.end();
  49. return;
  50. }
  51. if(stats.isFile()){
  52. var fileStream,
  53. mimetype = mimeTypes[path.extname(filename).split('.')[1]];
  54. res.writeHead(200,{
  55. 'Content-Type': mimetype
  56. });
  57. fileStream = fs.createReadStream(filename);
  58. fileStream.pipe(res);
  59. }else if(stats.isDirectory()){
  60. if(fs.existsSync(path.join(filename,'index.html'))){
  61. serveFile(path.join(filename,'index.html'),req,res);
  62. }else if(fs.existsSync(path.join(filename,'index.htm'))){
  63. serveFile(path.join(filename,'index.htm'),req,res);
  64. }else if(fs.existsSync(path.join(filename,'index.txt'))){
  65. serveFile(path.join(filename,'index.txt'),req,res);
  66. }else{
  67. res.writeHead(200,{
  68. 'Content-Type': 'text/plain'
  69. });
  70. res.write('Index of '+url+'\n');
  71. res.write('TODO, show index');
  72. res.end();
  73. }
  74. }else{
  75. res.writeHead(500,{
  76. 'Content-Type': 'text/plain'
  77. });
  78. res.write('500 Internal server error\n');
  79. res.end();
  80. }
  81. },
  82. filepath = unescape(uri);
  83. if(filepath.substr(0,5) == '/api/'){
  84. filepath = path.join('./api/',filepath.substr(5));
  85. logger.debug('Attempting to run api script '+filepath);
  86. if(fs.existsSync(filepath)){
  87. fs.readFile(filepath,function(e,data){
  88. if(e){
  89. logger.error(e);
  90. res.end('null;');
  91. }else{
  92. var output = '',
  93. sandbox = {
  94. log: function(text){
  95. output += text;
  96. },
  97. error: function(msg){
  98. logger.error(msg);
  99. },
  100. info: function(msg){
  101. logger.info(msg);
  102. },
  103. debug: function(msg){
  104. logger.debug(msg);
  105. },
  106. head: {
  107. 'Content-Type': 'text/javascript'
  108. },
  109. returnCode: 200,
  110. vm: vm,
  111. fs: fs
  112. };
  113. vm.runInNewContext(data,sandbox,filepath);
  114. res.writeHead(sandbox.returnCode,sandbox.head);
  115. res.end(output);
  116. }
  117. });
  118. }else{
  119. res.writeHead(404,{
  120. 'Content-Type': 'text/javascript'
  121. });
  122. res.end('null;');
  123. }
  124. }else{
  125. serveFile(path.join('./www/',filepath),req,res);
  126. }
  127. }).resume();
  128. }).listen(80),
  129. io = require('socket.io').listen(app)
  130. logger = io.log;
  131. io.set('log level',3);
  132. io.set('store', new RedisStore({
  133. redisPub : pub,
  134. redisSub : sub,
  135. redisClient : client
  136. }));
  137. io.sockets.on('connection',function(socket){
  138. socket.on('join',function(data){
  139. socket.join(data.name);
  140. data.title = data.name;
  141. socket.emit('join',{
  142. name: data.name,
  143. title: data.title
  144. });
  145. sendUserList(data.name);
  146. socket.get('nick',function(e,nick){
  147. logger.debug(nick+' joined '+data.name);
  148. io.sockets.in(data.name).emit('message',{
  149. message: nick+' joined the channel',
  150. room: data.name,
  151. from: 0
  152. });
  153. });
  154. });
  155. socket.on('part',function(data){
  156. socket.leave(data.name);
  157. socket.get('nick',function(e,nick){
  158. logger.debug(nick+' left '+data.name);
  159. sendUserList(data.name);
  160. });
  161. });
  162. socket.on('disconnect',function(data){
  163. var rooms = io.sockets.manager.rooms,
  164. i,
  165. room;
  166. for(i in rooms){
  167. if(rooms[i] != '' && typeof rooms[i] == 'string'){
  168. try{
  169. room = rooms[i].substr(1);
  170. }catch(e){}
  171. sendUserList(names);
  172. }
  173. }
  174. });
  175. socket.on('message',function(data){
  176. logger.debug('message sent to '+data.room);
  177. io.sockets.in(data.room).emit('message',data);
  178. });
  179. socket.on('echo',function(data){
  180. logger.debug('echoing to '+data.room);
  181. socket.emit('message',data);
  182. });
  183. socket.on('names',function(data){
  184. var sockets = io.sockets.clients(data.name),
  185. i;
  186. socket.emit('message',{
  187. message: data.name+' users:',
  188. room: data.name,
  189. from: 0
  190. });
  191. for(i in sockets){
  192. sockets[i].get('nick',function(e,nick){
  193. socket.emit('message',{
  194. message: ' '+nick,
  195. room: data.name,
  196. from: 0
  197. });
  198. });
  199. }
  200. sendUserList(data.name);
  201. });
  202. socket.on('auth',function(data){
  203. logger.info(data.nick+' registered');
  204. // TODO - authorize
  205. socket.set('nick',data.nick.substr(0,12));
  206. socket.emit('authorized',{
  207. nick: data.nick.substr(0,12)
  208. });
  209. });
  210. var usersInRoom = function(room){
  211. var sockets = io.sockets.clients(room),
  212. i,
  213. ret = [];
  214. for(i in sockets){
  215. sockets[i].get('nick',function(e,nick){
  216. ret.push(nick);
  217. });
  218. }
  219. return ret;
  220. },
  221. sendUserList = function(room){
  222. if(typeof room != 'undefined'){
  223. io.sockets.in(room).emit('names',{
  224. room: room,
  225. names: usersInRoom(room)
  226. });
  227. }
  228. };
  229. });
  230. }
  231. process.on('uncaughtException',function(e){
  232. logger.error(e);
  233. });