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