4 $max_room_number = 3; //最大ルーム数
\r
5 $spilt_size = 1024 * 512; //分割するサイズ
\r
6 $reset_password_diff = 1000 * 60 * 60;
\r
7 $block_message = "メッセージの送信に失敗しました"; //ブロック時のメッセージ
\r
8 $not_match_password = "パスワードが一致しませんでした"; //パスワードが一致しない場合に表示されるメッセージ
\r
9 $password_setted_message = "パスワードを設定しました"; //パスワードが設定されたときに表示されるメッセージ
\r
10 $password_resetted_message = "パスワードをリセットしました"; //パスワードが再設定されたときに表示されるメッセージ
\r
11 $failed_set_password_message = "パスワードの設定に失敗しました"; //パスワードが再設定されたときに表示されるメッセージ
\r
12 $free_password1 = "最初に入室する人が自由にパスワードを設定できます"; //自由パスワードメッセージ1
\r
13 $free_password2 = "この部屋は使用されています。パスワードを入力してください"; //自由パスワードメッセージ2
\r
14 $fixed_password = "この部屋にはパスワードが設定されています"; //固定パスワードルーム
\r
15 $ip_ban_list_file_name = "ipbanlist.txt"; //アクセスを禁止するIPが記録されているファイル
\r
16 $port = process.env.port || 3000; //ポート
\r
17 $username = "admin"; //管理者用のページにアクセスできるユーザ名
\r
18 $password = "admin"; //管理者用のページにアクセスするのに必要なパスワード
\r
19 $token_length = 32; //トークンの長さ
\r
20 $redisHost = "localhost"; //redisサーバのアドレス
\r
21 $redisPort = 6379; //redisサーバのポート
\r
22 $redisPassword = ""; //redisサーバのパスワード
\r
23 $system_name = "system"; //システム発言を表す名前
\r
24 $log_directory = "log"; //ログファイルを置くフォルダー
\r
25 $log_file_name = "logfile%d.txt"; //ログファイル名(%dはそのままにしておくこと)
\r
26 $splited_log_file_name = "logfile%d_%s.txt" //分割後のファイル名(%dと%sはそのままにしておくこと)
\r
27 $pastlogfile_pattern = "logfile%d(_+.*)?\.txt"; //過去ログと判定する正規表現
\r
28 $logfile_pattern = "logfile[0-9]+(_*.*)?\.txt" //過去ログと判定する正規表現
\r
29 //パスワードを自由に設定できる部屋のリスト
\r
31 // password パスワードを設定する。nullにすると利用者がパスワードの設定をすることができる
\r
33 // new RoomInfomationCollection(
\r
34 // {rno:"1",password:"test"},
\r
35 // {rno:"2",password:null}
\r
37 $rooms = new RoomInfomationCollection(
\r
38 {rno:"1",password:"test"},
\r
39 {rno:"2",password:null}
\r
43 * Module dependencies.
\r
47 var express = require('express');
\r
49 var app = module.exports = express.createServer();
\r
51 var util = require("util");
\r
53 var lazy = require("lazy");
\r
55 var fs = require("fs");
\r
57 var parseCookie = require("connect").utils.parseCookie;
\r
59 var RedisStore = require("connect-redis")(express);
\r
60 var sessionStore = new RedisStore({host:$redisHost,port:$redisPort,pass:$redisPassword});
\r
62 var async = require("async");
\r
64 var path = require("path");
\r
68 app.configure(function(){
\r
69 app.disabled("view cache");
\r
70 app.set("view options", { layout: false })
\r
71 app.set("views", __dirname + "/public");
\r
72 app.set("view engine", "ejs");
\r
73 app.use(express.bodyParser());
\r
74 app.use(express.methodOverride());
\r
75 app.use(express.cookieParser());
\r
76 app.use(express.session({
\r
78 secret: "5514EA2B-C9B2-4D65-8D81-1F33A180A0C2",
\r
79 cookie: { httpOnly: false }
\r
81 app.use(app.router);
\r
82 app.use(express.static(__dirname + "/public"));
\r
85 app.configure('development', function(){
\r
86 app.use(express.errorHandler({ dumpExceptions: true, showStack: true }));
\r
89 app.configure('production', function(){
\r
90 app.use(express.errorHandler());
\r
94 app.get("/chat", function(req, res){
\r
95 var auth_string = getRandomString($token_length);
\r
96 req.session.items = {token:auth_string};
\r
98 var room_number = 0;
\r
99 if(typeof(req.query.rno) != "undefined")
\r
100 room_number = req.query.rno;
\r
102 if($rooms.IsFixedPassword(room_number))
\r
103 msg = $fixed_password;
\r
104 else if($rooms.IsContains(room_number))
\r
105 msg = $free_password2;
\r
106 if($rooms.IsFirstAuth(room_number))
\r
107 msg = $free_password1;
\r
108 res.render("chat",{rno:room_number,token:auth_string,message:msg});
\r
111 app.all("/log/" + $logfile_pattern,express.basicAuth(function (user, pass) {
\r
112 return user === $username && pass === $password;
\r
115 app.get("/log/" + $logfile_pattern, function(req, res){
\r
116 res.sendfile(__dirname + req.url);
\r
119 app.all("/admin",express.basicAuth(function (user, pass) {
\r
120 return user === $username && pass === $password;
\r
123 app.get("/admin", function(req, res){
\r
124 renderAdmin(req,res);
\r
127 app.post("/admin",function(req,res){
\r
128 if(req.session.items.token != req.body.token)
\r
130 res.send($invaild_token_message);
\r
133 if(typeof(req.body.erase) != "undefined")
\r
135 removeLog(req.body.file,function(){
\r
136 renderAdmin(req,res);
\r
139 if(typeof(req.body.registor) != "undefined")
\r
141 updateIpBanList(req.body.newbanlist,function(){
\r
142 renderAdmin(req,res);
\r
145 if(typeof(req.body.updateroom) != "undefined")
\r
147 CreateRoomsFromString(req.body.newroomlist);
\r
148 renderAdmin(req,res);
\r
152 function renderAdmin(req,res)
\r
154 var auth_string = getRandomString($token_length);
\r
155 req.session.items = {token:auth_string};
\r
157 fs.readdir($log_directory,function(err,list){
\r
158 res.render("admin", {
\r
160 log_directory:$log_directory,
\r
161 ipbanlist:getTextFromIpBanlist(ipbanlist),
\r
163 roomlist:$rooms.GetString()
\r
168 function getRandomString(length)
\r
170 var RandomString = "";
\r
171 var BaseString ="ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789!#$%&'()=~|-^\@[;:],./\`{+*}<>?_";
\r
172 for(var i=0; i<length; i++) {
\r
173 RandomString += BaseString.charAt( Math.floor( Math.random() * BaseString.length));
\r
175 return RandomString
\r
178 function removeLog(files,callback)
\r
181 function(item,callback){
\r
182 fs.unlink($log_directory + "/" + item,callback);
\r
184 function(err,results){
\r
185 if(typeof(callback) == "function")
\r
190 function getTextFromIpBanlist(list)
\r
193 for(var key in ipbanlist)
\r
195 if(ipbanlist[key] == "")
\r
196 text += key + "\r\n";
\r
198 text += key + ":" + ipbanlist[key] + "\r\n";
\r
203 function updateIpBanList(text,callfunc)
\r
206 function(callback){
\r
207 fs.open($ip_ban_list_file_name,"w",callback);
\r
209 function(fd,callback){
\r
210 var buf = new Buffer(text);
\r
211 fs.write(fd,buf,0,Buffer.byteLength(text),null,function(){
\r
215 function(fd,callback){
\r
216 fs.close(fd,function(){
\r
217 getIpBanList(callfunc);
\r
224 console.log("Express server listening on port %d in %s mode", app.address().port, app.settings.env);
\r
230 var io = require("socket.io").listen(app);
\r
231 io.configure('production', function(){
\r
232 io.enable('browser client minification'); // minified されたクライアントファイルを送信する
\r
233 io.enable('browser client etag'); // バージョンによって etag によるキャッシングを有効にする
\r
234 io.set('log level', 1); // ログレベルを設定(デフォルトより下げている)
\r
236 var clients = new Array();
\r
238 var ipbanlist = {};
\r
240 createLogDirectory();
\r
244 for(var i = 0; i < $max_room_number; i++)
\r
247 .of(GetNameFromRoomNumber(i))
\r
248 .authorization(ParseAuthorization)
\r
249 .on("connection", function (socket) {
\r
250 console.log("connected from %s",GetClientIPAdress(socket));
\r
251 socket.json.emit("send roomlist",$rooms.GetKeys());
\r
252 socket.on("get pastLogList", function (msg) {
\r
253 ParseGetPastLogList(socket,msg);
\r
255 socket.on("get pastLog", function (msg) {
\r
256 ParseGetPastLog(socket,msg);
\r
258 socket.on("join",function(msg){
\r
259 ParseJoin(socket,msg);
\r
261 socket.on("quit",function(msg){
\r
262 ParseQuit(socket,msg);
\r
264 socket.on("set password",function(msg){
\r
265 ParseSetPassword(socket,msg);
\r
267 socket.on("send msg", function (msg) {
\r
268 ParseSendMsg(socket,msg);
\r
270 socket.on("disconnect", function (msg) {
\r
271 ParseDisconnect(socket,msg);
\r
276 function createLogDirectory()
\r
278 path.exists($log_directory,function(exists){
\r
279 if(exists == false)
\r
280 fs.mkdirSync($log_directory);
\r
284 function getIpBanList(callback)
\r
287 path.exists($ip_ban_list_file_name,function(exists){
\r
288 if(exists == false)
\r
290 if(typeof(callback) == "function")
\r
294 var stream = fs.createReadStream($ip_ban_list_file_name);
\r
297 .forEach(function(line){
\r
298 var token = line.toString().split(":");
\r
299 var ip = token[0].replace(/(\r|\n|\r\n)/gm, "");
\r
300 if(token.length == 1)
\r
301 ipbanlist[ip] = "";
\r
303 ipbanlist[ip] = token[1];
\r
306 if(typeof(callback) == "function")
\r
312 function ParseAuthorization(handshakeData, callback)
\r
314 if(handshakeData.headers.cookie) {
\r
315 var cookie = handshakeData.headers.cookie;
\r
316 var sessionID = parseCookie(cookie)["connect.sid"];
\r
317 sessionStore.get(sessionID, function (err, session) {
\r
319 if (err || ipbanlist[handshakeData.address.address] == "r")
\r
320 result = "failed get from session store";
\r
321 else if(handshakeData.query.token != session.items.token)
\r
322 result = "invaild token";
\r
323 sessionStore.destroy(sessionID);
\r
324 callback(result,result == null && !err);
\r
327 return callback("failed get cookie", false);
\r
331 function ParseDisconnect(socket,msg)
\r
333 console.log("disconnected");
\r
336 function ParseSetPassword(socket,msg)
\r
338 var rno = GetRoomNumberFromName(socket.namespace.name);
\r
343 if($rooms.SetPassword(rno,msg.owner,msg.password))
\r
344 newMeg.message = $password_setted_message;
\r
346 newMeg.message = $failed_set_password_message;
\r
347 ParseSendMsg(socket,newMeg);
\r
350 function ParseJoin(socket,msg)
\r
352 var ip = GetClientIPAdress(socket);
\r
354 if(ip in ipbanlist)
\r
356 socket.emit("error",$block_message);
\r
360 var rno = GetRoomNumberFromName(socket.namespace.name);
\r
361 if($rooms.IsContains(rno))
\r
363 if($rooms.IsTimeout(rno) ||
\r
364 $rooms.IsFirstAuth(rno))
\r
366 $rooms.Reset(rno,msg.name);
\r
367 ParseGetPastLog(socket,util.format($log_file_name,rno));
\r
369 else if($rooms.Auth(rno,msg.name,msg.password))
\r
371 ParseGetPastLog(socket,util.format($log_file_name,rno));
\r
375 socket.emit("error",$not_match_password);
\r
382 message:util.format("/enteredby %s %s %s",msg.name,msg.color,msg.mailto),
\r
384 ParseSendMsg(socket,newMeg);
\r
387 function ParseQuit(socket,msg)
\r
389 var ip = GetClientIPAdress(socket);
\r
391 if(ip in ipbanlist)
\r
393 socket.emit("error",$block_message);
\r
397 var rno = GetRoomNumberFromName(socket.namespace.name);
\r
401 message:$password_resetted_message,
\r
403 if($rooms.IsContains(rno))
\r
405 if($rooms.IsOwner(rno,msg.name))
\r
407 $rooms.Reset(rno,null);
\r
408 ParseSendMsg(socket,newMeg);
\r
410 if(!$rooms.IsFirstAuth(rno) &&
\r
411 !$rooms.IsAuthed(rno,msg.name))
\r
414 $rooms.RemoveAuth(rno,msg.name);
\r
417 newMeg.message = util.format("/quitedby %s",msg.name);
\r
418 ParseSendMsg(socket,newMeg);
\r
423 function ParseSendMsg(socket,msg)
\r
425 var ip = GetClientIPAdress(socket);
\r
427 if(ip in ipbanlist)
\r
429 socket.emit("error",$block_message);
\r
433 var rno = GetRoomNumberFromName(socket.namespace.name);
\r
435 if(msg.name != $system_name &&
\r
436 $rooms.IsContains(rno) &&
\r
437 !$rooms.IsAuthed(rno,msg.name) &&
\r
438 !$rooms.IsOwner(rno,msg.name))
\r
443 var date = new Date();
\r
445 var repacked_msg = CreateMessage(msg.name,date,msg.message);
\r
447 socket.json.emit("req msg", repacked_msg);
\r
449 socket.json.broadcast.emit("req msg", repacked_msg);
\r
451 var path = $log_directory + "/" + util.format($log_file_name,rno);
\r
452 var log = new ChatLog(path);
\r
453 log.Save(repacked_msg,ip,rno);
\r
456 function GetNameFromRoomNumber(number)
\r
458 return "/" + number;
\r
461 function GetRoomNumberFromName(name)
\r
463 if(name.charAt(0) == "/")
\r
464 return parseInt(name.substr(1));
\r
465 throw "GetRoomNumberFromName error";
\r
468 function ParseGetPastLogList(socket,msg)
\r
470 var list = fs.readdir($log_directory,function(err,files){
\r
472 var rno = GetRoomNumberFromName(socket.namespace.name);
\r
473 var pattern = $pastlogfile_pattern.replace("%d",rno);
\r
474 for(var i = 0; i < files.length; i++)
\r
476 var logname = files[i];
\r
477 if(logname.match(pattern))
\r
478 text += files[i] + "\n";
\r
480 socket.emit("req pastloglist",text);
\r
484 function ParseGetPastLog(socket,file)
\r
488 var path = $log_directory + "/" + file;
\r
489 var log = new ChatLog(path);
\r
490 log.ToArray(function(array){
\r
491 socket.json.emit("req pastlog",array);
\r
495 function ChatLog(path)
\r
497 this.ToArray = function(callback)
\r
499 var state = fs.stat(path,function(err,state){
\r
502 var array = new Array();
\r
503 var stream = fs.createReadStream(path);
\r
506 .forEach(function(line){
\r
507 array.push(CreateMessageFromText(line.toString()));
\r
515 this.Save = function(msg,ip,rno){
\r
516 var text = GetTextFromMessage(msg,ip);
\r
518 SplitLog(rno,function(){
\r
519 WritePastLog(path,text);
\r
523 function GetTextFromMessage(msg,ip)
\r
525 var text = msg.name + "<>" +
\r
533 function SplitLog(rno,callback)
\r
535 var state = fs.stat(path,function(err,state){
\r
536 if(err && typeof(callback) == "function")
\r
541 if(state.size > $spilt_size)
\r
543 var date = new Date();
\r
544 var dateString = ""+date.getFullYear()+date.getMonth()+date.getDate()+date.getHours()+date.getMinutes()+date.getSeconds();
\r
546 var newpath = $log_directory + "/" +
\r
547 util.format($splited_log_file_name,rno,dateString);
\r
548 fs.rename(path,newpath,callback);
\r
550 if(typeof(callback) == "function")
\r
556 function WritePastLog(path,text)
\r
559 function(callback){
\r
560 fs.open(path,"a",callback);
\r
562 function(fd,callback){
\r
563 var buf = new Buffer(text);
\r
564 fs.write(fd,buf,0,Buffer.byteLength(text),null,function(){
\r
575 function GetClientIPAdress(socket)
\r
577 return socket.handshake.headers["x-forwarded-for"] || socket.handshake.address.address;
\r
581 function CreateMessage(name,date,message)
\r
583 var result = {name:name,
\r
588 function CreateMessageFromText(text)
\r
590 var data = text.split("<>");
\r
591 var msg = {name:data[0],
\r
597 //RoomInfomationCollecionクラス
\r
598 function RoomInfomationCollection()
\r
600 var collection = {};
\r
601 this.Clear = function(){
\r
604 this.Add = function(rno,pass){
\r
605 collection[rno] = {time : null,
\r
610 collection[rno].owner = $system_name;
\r
612 this.Reset = function(rno,owner){
\r
613 var date = new Date();
\r
614 var time = date.getTime();
\r
615 collection[rno].password = null;
\r
616 collection[rno].authed_list = {};
\r
617 collection[rno].owner = owner;
\r
618 collection[rno].time = time;
\r
619 console.log(util.format("password is reseted in %s",rno));
\r
621 this.IsContains = function(rno){
\r
622 return rno in collection;
\r
624 this.IsFirstAuth = function(rno){
\r
625 if(!this.IsContains(rno) || typeof(collection[rno].owner) == "undefined")
\r
627 return collection[rno].owner == null;
\r
629 this.IsAuthed = function(rno,name){
\r
630 if(!this.IsContains(rno))
\r
632 console.log(util.inspect(collection[rno].authed_list));
\r
633 return name == collection[rno].owner ||
\r
634 name in collection[rno].authed_list;
\r
636 this.IsFixedPassword = function(rno){
\r
637 if(!this.IsContains(rno))
\r
639 return collection[rno].owner == $system_name;
\r
641 this.IsOwner = function(rno,name){
\r
642 return this.IsContains(rno) &&
\r
643 typeof(collection[rno].owner) != "undefined" &&
\r
644 collection[rno].owner == name;
\r
646 this.IsTimeout = function(rno){
\r
647 var date = new Date();
\r
648 var time = date.getTime();
\r
649 return this.IsContains(rno) &&
\r
650 !this.IsFixedPassword(rno) &&
\r
651 (typeof(collection[rno].time) != "undefined" &&
\r
652 time - collection[rno].time >= $reset_password_diff);
\r
654 this.RemoveAuth = function(rno,name)
\r
656 delete collection[rno].authed_list[name];
\r
658 this.Auth = function(rno,name,password){
\r
659 if(typeof(collection[rno].password) != "undefined" &&
\r
660 collection[rno].password != password)
\r
662 var date = new Date();
\r
663 var time = date.getTime();
\r
664 collection[rno].time = time;
\r
665 collection[rno].authed_list[name] = "";
\r
668 this.SetPassword = function(rno,owner,password){
\r
669 if(this.IsContains(rno) &&
\r
670 owner == collection[rno].owner &&
\r
671 !this.IsFixedPassword(rno))
\r
673 var date = new Date();
\r
674 collection[rno].time = date.getTime();
\r
675 collection[rno].password = password;
\r
677 console.log(util.format("password is seted to %s in %s",password,rno));
\r
682 this.GetString = function(){
\r
684 for(var rno in collection)
\r
686 var pass = collection[rno].password;
\r
689 retval += rno + ":" + pass + "\r\n";
\r
693 this.GetKeys = function(){
\r
695 for(var rno in collection)
\r
702 for(var i=0; i<arguments.length; i++)
\r
704 this.Add(arguments[i].rno,arguments[i].password);
\r
708 function CreateRoomsFromString(str)
\r
711 var lines = str.split("\r\n");
\r
712 for(var i in lines)
\r
714 var token = lines[i].split(":");
\r
715 console.log(util.inspect(token));
\r
716 if(token.length == 1)
\r
718 $rooms.Add(token[0],null);
\r
720 else if(token.length == 2)
\r
722 var rno = token[0];
\r
723 var pass = token[1];
\r
726 $rooms.Add(rno, pass);
\r