2016-12-28 23:49:51 +01:00
|
|
|
import * as http from 'http';
|
|
|
|
import * as websocket from 'websocket';
|
2018-10-11 11:09:41 +02:00
|
|
|
import * as redis from 'redis';
|
2018-07-30 00:20:27 +02:00
|
|
|
import Xev from 'xev';
|
2016-12-28 23:49:51 +01:00
|
|
|
|
2018-10-07 04:06:17 +02:00
|
|
|
import MainStreamConnection from './stream';
|
2018-03-29 13:32:18 +02:00
|
|
|
import { ParsedUrlQuery } from 'querystring';
|
2018-04-11 10:40:01 +02:00
|
|
|
import authenticate from './authenticate';
|
2018-10-11 11:09:41 +02:00
|
|
|
import { EventEmitter } from 'events';
|
|
|
|
import config from '../../config';
|
2016-12-28 23:49:51 +01:00
|
|
|
|
|
|
|
module.exports = (server: http.Server) => {
|
2018-10-07 04:06:17 +02:00
|
|
|
// Init websocket server
|
2016-12-28 23:49:51 +01:00
|
|
|
const ws = new websocket.server({
|
|
|
|
httpServer: server
|
|
|
|
});
|
|
|
|
|
|
|
|
ws.on('request', async (request) => {
|
2018-10-07 04:06:17 +02:00
|
|
|
const q = request.resourceURL.query as ParsedUrlQuery;
|
|
|
|
const [user, app] = await authenticate(q.i as string);
|
2017-06-08 18:03:54 +02:00
|
|
|
|
2018-11-11 16:31:09 +01:00
|
|
|
const connection = request.accept();
|
|
|
|
|
2018-10-11 11:09:41 +02:00
|
|
|
let ev: EventEmitter;
|
|
|
|
|
2019-02-07 13:02:33 +01:00
|
|
|
if (config.redis) {
|
2018-10-11 11:09:41 +02:00
|
|
|
// Connect to Redis
|
|
|
|
const subscriber = redis.createClient(
|
2019-02-07 13:02:33 +01:00
|
|
|
config.redis.port, config.redis.host);
|
2018-10-11 11:09:41 +02:00
|
|
|
|
|
|
|
subscriber.subscribe('misskey');
|
|
|
|
|
|
|
|
ev = new EventEmitter();
|
|
|
|
|
|
|
|
subscriber.on('message', async (_, data) => {
|
|
|
|
const obj = JSON.parse(data);
|
|
|
|
|
|
|
|
ev.emit(obj.channel, obj.message);
|
|
|
|
});
|
|
|
|
|
|
|
|
connection.once('close', () => {
|
|
|
|
subscriber.unsubscribe();
|
|
|
|
subscriber.quit();
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
ev = new Xev();
|
|
|
|
}
|
|
|
|
|
2018-10-07 04:06:17 +02:00
|
|
|
const main = new MainStreamConnection(connection, ev, user, app);
|
2016-12-28 23:49:51 +01:00
|
|
|
|
2018-10-07 10:19:52 +02:00
|
|
|
// 後方互換性のため
|
|
|
|
if (request.resourceURL.pathname !== '/streaming') {
|
2018-10-07 17:58:10 +02:00
|
|
|
main.sendMessageToWsOverride = (type: string, payload: any) => {
|
2018-10-07 10:19:52 +02:00
|
|
|
if (type == 'channel') {
|
|
|
|
type = payload.type;
|
|
|
|
payload = payload.body;
|
|
|
|
}
|
2018-10-08 01:58:12 +02:00
|
|
|
if (type.startsWith('api:')) {
|
2018-10-08 02:02:11 +02:00
|
|
|
type = type.replace('api:', 'api-res:');
|
2018-10-08 01:58:12 +02:00
|
|
|
}
|
2018-10-07 10:19:52 +02:00
|
|
|
connection.send(JSON.stringify({
|
|
|
|
type: type,
|
|
|
|
body: payload
|
|
|
|
}));
|
|
|
|
};
|
2018-10-07 17:58:10 +02:00
|
|
|
|
2018-10-13 12:16:47 +02:00
|
|
|
main.connectChannel(Math.random().toString().substr(2, 8), null,
|
2018-10-11 16:01:57 +02:00
|
|
|
request.resourceURL.pathname === '/' ? 'homeTimeline' :
|
|
|
|
request.resourceURL.pathname === '/local-timeline' ? 'localTimeline' :
|
|
|
|
request.resourceURL.pathname === '/hybrid-timeline' ? 'hybridTimeline' :
|
|
|
|
request.resourceURL.pathname === '/global-timeline' ? 'globalTimeline' : null);
|
2018-10-08 20:29:11 +02:00
|
|
|
|
|
|
|
if (request.resourceURL.pathname === '/') {
|
2018-10-13 12:16:47 +02:00
|
|
|
main.connectChannel(Math.random().toString().substr(2, 8), null, 'main');
|
2018-10-08 20:29:11 +02:00
|
|
|
}
|
2018-10-07 10:19:52 +02:00
|
|
|
}
|
|
|
|
|
2018-07-30 00:20:27 +02:00
|
|
|
connection.once('close', () => {
|
|
|
|
ev.removeAllListeners();
|
2018-10-07 04:06:17 +02:00
|
|
|
main.dispose();
|
2016-12-28 23:49:51 +01:00
|
|
|
});
|
|
|
|
|
2018-09-17 02:07:46 +02:00
|
|
|
connection.on('message', async (data) => {
|
|
|
|
if (data.utf8Data == 'ping') {
|
|
|
|
connection.send('pong');
|
|
|
|
}
|
|
|
|
});
|
2016-12-28 23:49:51 +01:00
|
|
|
});
|
|
|
|
};
|