mizzkey/src/server/api/streaming.ts

68 lines
1.8 KiB
TypeScript
Raw Normal View History

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