-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathwebrtc_mesh_server.js
119 lines (110 loc) · 3.18 KB
/
webrtc_mesh_server.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
const path = require('path');
const fs = require('fs');
const https = require('https');
const WebSocket = require('ws');
const sslPort = 8443;
const serverConfig = {
// SSL証明書、環境に合わせてパスを変更する
key: fs.readFileSync('privkey.pem'),
cert: fs.readFileSync('cert.pem')
};
// 接続リスト
let connections = [];
// WebSocket処理
const socketProc = function(ws, req) {
ws._pingTimer = setInterval(() => {
if (ws.readyState === WebSocket.OPEN) {
// 接続確認
ws.send(JSON.stringify({ping: 1}));
}
}, 180000);
ws.on('message', function(message) {
const json = JSON.parse(message);
if (json.join) {
console.log(ws._socket.remoteAddress + ': join room=' + json.join.room + ', id=' + json.join.id);
// 同一IDが存在するときは古い方を削除
connections = connections.filter(data => !(data.room === json.join.room && data.id === json.join.id));
// 接続情報を保存
connections.push({room: json.join.room, id: json.join.id, ws: ws});
connections.forEach(data => {
if (data.room === json.join.room && data.id !== json.join.id && data.ws.readyState === WebSocket.OPEN) {
// 新規参加者を通知
data.ws.send(JSON.stringify({join: json.join.id}));
}
});
return;
}
if (json.pong) {
return;
}
if (json.ping) {
if (ws.readyState === WebSocket.OPEN) {
ws.send(JSON.stringify({pong: 1}));
}
return;
}
// Peerを検索
connections.some(data => {
if (data.room === json.room && data.id === json.dest && data.ws.readyState === WebSocket.OPEN) {
// メッセージの転送
data.ws.send(JSON.stringify(json));
return true;
}
});
});
ws.on('close', function () {
closeConnection(ws);
});
ws.on('error', function(error) {
closeConnection(ws);
console.error(ws._socket.remoteAddress + ': error=' + error);
});
function closeConnection(conn) {
connections = connections.filter(data => {
if (data.ws !== conn) {
return true;
}
console.log(ws._socket.remoteAddress + ': part room=' + data.room + ', id=' + data.id);
connections.forEach(roomData => {
if (data.room === roomData.room && data.id !== roomData.id && roomData.ws.readyState === WebSocket.OPEN) {
// 退出を通知
roomData.ws.send(JSON.stringify({part: 1, src: data.id}));
}
});
data.ws = null;
return false;
});
if (conn._pingTimer) {
clearInterval(conn._pingTimer);
conn._pingTimer = null;
}
}
};
// 静的ファイル処理
const service = function(req, res) {
const url = req.url.replace(/\?.+$/, '');
const file = path.join(process.cwd(), url);
fs.stat(file, (err, stat) => {
if (err) {
res.writeHead(404);
res.end();
return;
}
if (stat.isDirectory()) {
service({url: url.replace(/\/$/, '') + '/index.html'}, res);
} else if (stat.isFile()) {
const stream = fs.createReadStream(file);
stream.pipe(res);
} else {
res.writeHead(404);
res.end();
}
});
};
// HTTPSサーバの開始
const httpsServer = https.createServer(serverConfig, service);
httpsServer.listen(sslPort, '0.0.0.0');
// WebSocketの開始
const wss = new WebSocket.Server({server: httpsServer});
wss.on('connection', socketProc);
console.log('Server running.');