2016-12-29 00:49:51 +02:00
|
|
|
import * as http from 'http';
|
|
|
|
import * as websocket from 'websocket';
|
|
|
|
|
2021-08-19 15:55:45 +03:00
|
|
|
import MainStreamConnection from './stream/index';
|
2018-03-29 14:32:18 +03:00
|
|
|
import { ParsedUrlQuery } from 'querystring';
|
2021-08-19 15:55:45 +03:00
|
|
|
import authenticate from './authenticate';
|
2018-10-11 12:09:41 +03:00
|
|
|
import { EventEmitter } from 'events';
|
2021-08-19 15:55:45 +03:00
|
|
|
import { subsdcriber as redisClient } from '../../db/redis';
|
|
|
|
import { Users } from '@/models/index';
|
2016-12-29 00:49:51 +02:00
|
|
|
|
|
|
|
module.exports = (server: http.Server) => {
|
2018-10-07 05:06:17 +03:00
|
|
|
// Init websocket server
|
2016-12-29 00:49:51 +02:00
|
|
|
const ws = new websocket.server({
|
2021-12-09 16:58:30 +02:00
|
|
|
httpServer: server,
|
2016-12-29 00:49:51 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
ws.on('request', async (request) => {
|
2018-10-07 05:06:17 +03:00
|
|
|
const q = request.resourceURL.query as ParsedUrlQuery;
|
2020-04-26 05:19:57 +03:00
|
|
|
|
|
|
|
// TODO: トークンが間違ってるなどしてauthenticateに失敗したら
|
|
|
|
// コネクション切断するなりエラーメッセージ返すなりする
|
|
|
|
// (現状はエラーがキャッチされておらずサーバーのログに流れて邪魔なので)
|
2018-10-07 05:06:17 +03:00
|
|
|
const [user, app] = await authenticate(q.i as string);
|
2017-06-08 19:03:54 +03:00
|
|
|
|
2021-07-18 13:57:53 +03:00
|
|
|
if (user?.isSuspended) {
|
|
|
|
request.reject(400);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-11-11 17:31:09 +02:00
|
|
|
const connection = request.accept();
|
|
|
|
|
2021-03-23 04:53:25 +02:00
|
|
|
const ev = new EventEmitter();
|
2018-10-11 12:09:41 +03:00
|
|
|
|
2021-03-23 04:53:25 +02:00
|
|
|
async function onRedisMessage(_: string, data: string) {
|
|
|
|
const parsed = JSON.parse(data);
|
|
|
|
ev.emit(parsed.channel, parsed.message);
|
|
|
|
}
|
2018-10-11 12:09:41 +03:00
|
|
|
|
2021-03-23 04:53:25 +02:00
|
|
|
redisClient.on('message', onRedisMessage);
|
2018-10-11 12:09:41 +03:00
|
|
|
|
2018-10-07 05:06:17 +03:00
|
|
|
const main = new MainStreamConnection(connection, ev, user, app);
|
2016-12-29 00:49:51 +02:00
|
|
|
|
2021-04-18 16:35:47 +03:00
|
|
|
const intervalId = user ? setInterval(() => {
|
|
|
|
Users.update(user.id, {
|
|
|
|
lastActiveDate: new Date(),
|
|
|
|
});
|
|
|
|
}, 1000 * 60 * 5) : null;
|
|
|
|
if (user) {
|
|
|
|
Users.update(user.id, {
|
|
|
|
lastActiveDate: new Date(),
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2018-07-30 01:20:27 +03:00
|
|
|
connection.once('close', () => {
|
|
|
|
ev.removeAllListeners();
|
2018-10-07 05:06:17 +03:00
|
|
|
main.dispose();
|
2021-03-23 04:53:25 +02:00
|
|
|
redisClient.off('message', onRedisMessage);
|
2021-04-18 16:35:47 +03:00
|
|
|
if (intervalId) clearInterval(intervalId);
|
2016-12-29 00:49:51 +02:00
|
|
|
});
|
|
|
|
|
2018-09-17 03:07:46 +03:00
|
|
|
connection.on('message', async (data) => {
|
2020-04-04 02:46:54 +03:00
|
|
|
if (data.utf8Data === 'ping') {
|
2018-09-17 03:07:46 +03:00
|
|
|
connection.send('pong');
|
|
|
|
}
|
|
|
|
});
|
2016-12-29 00:49:51 +02:00
|
|
|
});
|
|
|
|
};
|