2020-04-04 04:08:12 +02:00
|
|
|
import socketIO = require('socket.io');
|
|
|
|
import {Socket} from "socket.io";
|
|
|
|
import * as http from "http";
|
2020-04-04 17:22:02 +02:00
|
|
|
import {MessageUserPosition} from "../Model/Websocket/MessageUserPosition"; //TODO fix import by "_Model/.."
|
|
|
|
import {ExSocketInterface} from "../Model/Websocket/ExSocketInterface"; //TODO fix import by "_Model/.."
|
2020-04-04 14:05:18 +02:00
|
|
|
import Jwt, {JsonWebTokenError} from "jsonwebtoken";
|
2020-04-04 17:22:02 +02:00
|
|
|
import {SECRET_KEY} from "../Enum/EnvironmentVariable"; //TODO fix import by "_Enum/..."
|
2020-04-04 04:08:12 +02:00
|
|
|
|
|
|
|
export class IoSocketController{
|
|
|
|
Io: socketIO.Server;
|
|
|
|
constructor(server : http.Server) {
|
|
|
|
this.Io = socketIO(server);
|
2020-04-04 14:05:18 +02:00
|
|
|
|
2020-04-04 22:35:20 +02:00
|
|
|
// Authentication with token. it will be decoded and stored in the socket.
|
2020-04-04 14:05:18 +02:00
|
|
|
this.Io.use( (socket: Socket, next) => {
|
|
|
|
if (!socket.handshake.query || !socket.handshake.query.token) {
|
|
|
|
return next(new Error('Authentication error'));
|
|
|
|
}
|
|
|
|
Jwt.verify(socket.handshake.query.token, SECRET_KEY, (err: JsonWebTokenError, tokenDecoded: object) => {
|
|
|
|
if (err) {
|
|
|
|
return next(new Error('Authentication error'));
|
|
|
|
}
|
|
|
|
(socket as ExSocketInterface).token = tokenDecoded;
|
|
|
|
next();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2020-04-04 04:08:12 +02:00
|
|
|
this.ioConnection();
|
2020-04-04 19:25:08 +02:00
|
|
|
this.shareUsersPosition();
|
2020-04-04 04:08:12 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
ioConnection() {
|
|
|
|
this.Io.on('connection', (socket: Socket) => {
|
|
|
|
/*join-rom event permit to join one room.
|
|
|
|
message :
|
|
|
|
userId : user identification
|
|
|
|
roomId: room identification
|
2020-04-04 16:25:03 +02:00
|
|
|
position: position of user in map
|
|
|
|
x: user x position on map
|
|
|
|
y: user y position on map
|
2020-04-04 04:08:12 +02:00
|
|
|
*/
|
2020-04-04 16:25:03 +02:00
|
|
|
socket.on('join-room', (message : string) => {
|
2020-04-04 17:22:02 +02:00
|
|
|
let messageUserPosition = this.hydrateMessageReceive(message);
|
2020-04-04 17:56:43 +02:00
|
|
|
if(messageUserPosition instanceof Error){
|
|
|
|
return socket.emit("message-error", JSON.stringify({message: messageUserPosition.message}))
|
2020-04-04 17:22:02 +02:00
|
|
|
}
|
|
|
|
//join user in room
|
2020-04-04 16:25:03 +02:00
|
|
|
socket.join(messageUserPosition.roomId);
|
2020-04-04 12:42:02 +02:00
|
|
|
// sending to all clients in room except sender
|
2020-04-04 19:25:08 +02:00
|
|
|
this.saveUserInformation((socket as ExSocketInterface), messageUserPosition);
|
2020-04-04 16:25:03 +02:00
|
|
|
socket.to(messageUserPosition.roomId).emit('join-room', messageUserPosition.toString());
|
2020-04-04 12:42:02 +02:00
|
|
|
});
|
|
|
|
|
2020-04-04 16:25:03 +02:00
|
|
|
socket.on('user-position', (message : string) => {
|
2020-04-04 17:22:02 +02:00
|
|
|
let messageUserPosition = this.hydrateMessageReceive(message);
|
2020-04-04 17:56:43 +02:00
|
|
|
if(messageUserPosition instanceof Error){
|
|
|
|
return socket.emit("message-error", JSON.stringify({message: messageUserPosition.message}));
|
2020-04-04 17:22:02 +02:00
|
|
|
}
|
2020-04-04 12:42:02 +02:00
|
|
|
// sending to all clients in room except sender
|
2020-04-04 19:25:08 +02:00
|
|
|
this.saveUserInformation((socket as ExSocketInterface), messageUserPosition);
|
2020-04-04 16:25:03 +02:00
|
|
|
socket.to(messageUserPosition.roomId).emit('join-room', messageUserPosition.toString());
|
2020-04-04 04:08:12 +02:00
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
2020-04-04 16:25:03 +02:00
|
|
|
|
|
|
|
//permit to save user position in socket
|
2020-04-04 19:25:08 +02:00
|
|
|
saveUserInformation(socket : ExSocketInterface, message : MessageUserPosition){
|
2020-04-04 16:25:03 +02:00
|
|
|
socket.position = message.position;
|
2020-04-04 19:25:08 +02:00
|
|
|
socket.roomId = message.roomId;
|
|
|
|
socket.userId = message.userId;
|
2020-04-04 16:25:03 +02:00
|
|
|
}
|
2020-04-04 17:22:02 +02:00
|
|
|
|
|
|
|
//Hydrate and manage error
|
2020-04-04 17:56:43 +02:00
|
|
|
hydrateMessageReceive(message : string) : MessageUserPosition | Error{
|
2020-04-04 17:22:02 +02:00
|
|
|
try {
|
|
|
|
return new MessageUserPosition(message);
|
|
|
|
}catch (err) {
|
|
|
|
//TODO log error
|
2020-04-04 17:56:43 +02:00
|
|
|
return new Error(err);
|
2020-04-04 17:22:02 +02:00
|
|
|
}
|
|
|
|
}
|
2020-04-04 19:25:08 +02:00
|
|
|
|
|
|
|
/** permit to share user position
|
|
|
|
** users position will send in event 'user-position'
|
|
|
|
** The data sent is an array with information for each user :
|
|
|
|
[
|
|
|
|
{
|
|
|
|
userId: <string>,
|
|
|
|
roomId: <string>,
|
|
|
|
position: {
|
|
|
|
x : <number>,
|
|
|
|
y : <number>
|
|
|
|
}
|
|
|
|
},
|
|
|
|
...
|
|
|
|
]
|
|
|
|
**/
|
|
|
|
seTimeOutInProgress : any = null;
|
|
|
|
shareUsersPosition(){
|
|
|
|
if(!this.seTimeOutInProgress) {
|
|
|
|
clearTimeout(this.seTimeOutInProgress);
|
|
|
|
}
|
|
|
|
let clients = this.Io.clients();
|
|
|
|
let socketsKey = Object.keys(this.Io.clients().sockets);
|
|
|
|
|
|
|
|
//create mapping with all users in all rooms
|
|
|
|
let mapPositionUserByRoom = new Map();
|
|
|
|
for(let i = 0; i < socketsKey.length; i++){
|
|
|
|
let socket = clients.sockets[socketsKey[i]];
|
|
|
|
if(!(socket as ExSocketInterface).position){
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
let data = {
|
|
|
|
userId : (socket as ExSocketInterface).userId,
|
|
|
|
roomId : (socket as ExSocketInterface).roomId,
|
|
|
|
position : (socket as ExSocketInterface).position,
|
|
|
|
};
|
|
|
|
let dataArray = <any>[];
|
|
|
|
if(mapPositionUserByRoom.get(data.roomId)){
|
|
|
|
dataArray = mapPositionUserByRoom.get(data.roomId);
|
|
|
|
dataArray.push(data);
|
|
|
|
}else{
|
|
|
|
dataArray = [data];
|
|
|
|
}
|
|
|
|
mapPositionUserByRoom.set(data.roomId, dataArray);
|
|
|
|
}
|
|
|
|
|
|
|
|
//send for each room, all data of position user
|
|
|
|
let arrayMap = Array.from(mapPositionUserByRoom);
|
|
|
|
arrayMap.forEach((value) => {
|
|
|
|
let roomId = value[0];
|
|
|
|
let data = value[1];
|
|
|
|
this.Io.in(roomId).emit('user-position', JSON.stringify(data));
|
|
|
|
});
|
|
|
|
this.seTimeOutInProgress = setTimeout(() => {
|
|
|
|
this.shareUsersPosition();
|
|
|
|
}, 10);
|
|
|
|
}
|
2020-04-04 22:35:20 +02:00
|
|
|
}
|