FoundKey/src/api/streaming.ts

96 lines
2 KiB
TypeScript
Raw Normal View History

2016-12-28 22:49:51 +00:00
import * as http from 'http';
import * as websocket from 'websocket';
import * as redis from 'redis';
2017-01-17 00:17:52 +00:00
import config from '../conf';
2016-12-28 22:49:51 +00:00
import User from './models/user';
import AccessToken from './models/access-token';
2017-01-06 02:07:42 +00:00
import isNativeToken from './common/is-native-token';
2016-12-28 22:49:51 +00:00
import homeStream from './stream/home';
import messagingStream from './stream/messaging';
2017-06-08 16:03:54 +00:00
import serverStream from './stream/server';
2016-12-28 22:49:51 +00:00
module.exports = (server: http.Server) => {
/**
* Init websocket server
*/
const ws = new websocket.server({
httpServer: server
});
ws.on('request', async (request) => {
const connection = request.accept();
2017-06-08 16:03:54 +00:00
if (request.resourceURL.pathname === '/server') {
serverStream(request, connection);
return;
}
const user = await authenticate(connection, request.resourceURL.query.i);
if (user == null) {
connection.send('authentication-failed');
connection.close();
return;
}
2016-12-28 22:49:51 +00:00
// Connect to Redis
const subscriber = redis.createClient(
config.redis.port, config.redis.host);
connection.on('close', () => {
subscriber.unsubscribe();
subscriber.quit();
});
const channel =
request.resourceURL.pathname === '/' ? homeStream :
request.resourceURL.pathname === '/messaging' ? messagingStream :
null;
if (channel !== null) {
channel(request, connection, subscriber, user);
} else {
connection.close();
}
});
};
function authenticate(connection: websocket.connection, token: string): Promise<any> {
2017-01-20 22:45:49 +00:00
if (token == null) {
return Promise.resolve(null);
}
return new Promise(async (resolve, reject) => {
2017-01-06 02:07:42 +00:00
if (isNativeToken(token)) {
2016-12-28 22:49:51 +00:00
// Fetch user
// SELECT _id
const user = await User
.findOne({
token: token
2016-12-28 22:49:51 +00:00
});
resolve(user);
} else {
const accessToken = await AccessToken.findOne({
2017-01-06 02:50:46 +00:00
hash: token
});
if (accessToken == null) {
2017-01-06 03:30:35 +00:00
return reject('invalid signature');
2016-12-28 22:49:51 +00:00
}
// Fetch user
// SELECT _id
const user = await User
.findOne({ _id: accessToken.user_id }, {
2017-01-20 22:45:06 +00:00
fields: {
_id: true
}
});
2016-12-28 22:49:51 +00:00
resolve(user);
}
2016-12-28 22:49:51 +00:00
});
}