Merge pull request #155 from thecodingmachine/fixing_long_disconnect
Fixing disconnection taking ~15 seconds
This commit is contained in:
commit
d630106ff0
@ -388,6 +388,17 @@ export class IoSocketController {
|
|||||||
userId: userId
|
userId: userId
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// Most of the time, sending a disconnect event to one of the players is enough (the player will close the connection
|
||||||
|
// which will be shut for the other player).
|
||||||
|
// However! In the rare case where the WebRTC connection is not yet established, if we close the connection on one of the player,
|
||||||
|
// the other player will try connecting until a timeout happens (during this time, the connection icon will be displayed for nothing).
|
||||||
|
// So we also send the disconnect event to the other player.
|
||||||
|
for (let user of group.getUsers()) {
|
||||||
|
Client.emit(SockerIoEvent.WEBRTC_DISCONNECT, {
|
||||||
|
userId: user.id
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
//disconnect webrtc room
|
//disconnect webrtc room
|
||||||
if(!Client.webRtcRoomId){
|
if(!Client.webRtcRoomId){
|
||||||
return;
|
return;
|
||||||
|
@ -7,6 +7,7 @@ import {SetPlayerDetailsMessage} from "./Messages/SetPlayerDetailsMessage";
|
|||||||
const SocketIo = require('socket.io-client');
|
const SocketIo = require('socket.io-client');
|
||||||
import Socket = SocketIOClient.Socket;
|
import Socket = SocketIOClient.Socket;
|
||||||
import {PlayerAnimationNames} from "./Phaser/Player/Animation";
|
import {PlayerAnimationNames} from "./Phaser/Player/Animation";
|
||||||
|
import {UserSimplePeer} from "./WebRtc/SimplePeer";
|
||||||
|
|
||||||
|
|
||||||
enum EventMessage{
|
enum EventMessage{
|
||||||
@ -97,6 +98,15 @@ export interface GroupCreatedUpdatedMessageInterface {
|
|||||||
groupId: string
|
groupId: string
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export interface WebRtcStartMessageInterface {
|
||||||
|
roomId: string,
|
||||||
|
clients: UserSimplePeer[]
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface WebRtcDisconnectMessageInterface {
|
||||||
|
userId: string
|
||||||
|
}
|
||||||
|
|
||||||
export interface ConnectionInterface {
|
export interface ConnectionInterface {
|
||||||
socket: Socket|null;
|
socket: Socket|null;
|
||||||
token: string|null;
|
token: string|null;
|
||||||
@ -116,9 +126,9 @@ export interface ConnectionInterface {
|
|||||||
|
|
||||||
receiveWebrtcSignal(callBack: Function): void;
|
receiveWebrtcSignal(callBack: Function): void;
|
||||||
|
|
||||||
receiveWebrtcStart(callBack: Function): void;
|
receiveWebrtcStart(callBack: (message: WebRtcStartMessageInterface) => void): void;
|
||||||
|
|
||||||
disconnectMessage(callBack: Function): void;
|
disconnectMessage(callBack: (message: WebRtcDisconnectMessageInterface) => void): void;
|
||||||
}
|
}
|
||||||
|
|
||||||
export class Connection implements ConnectionInterface {
|
export class Connection implements ConnectionInterface {
|
||||||
@ -277,7 +287,7 @@ export class Connection implements ConnectionInterface {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
receiveWebrtcStart(callback: Function) {
|
receiveWebrtcStart(callback: (message: WebRtcStartMessageInterface) => void) {
|
||||||
this.getSocket().on(EventMessage.WEBRTC_START, callback);
|
this.getSocket().on(EventMessage.WEBRTC_START, callback);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -305,7 +315,7 @@ export class Connection implements ConnectionInterface {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
disconnectMessage(callback: Function): void {
|
disconnectMessage(callback: (message: WebRtcDisconnectMessageInterface) => void): void {
|
||||||
this.getSocket().on(EventMessage.WEBRTC_DISCONNECT, callback);
|
this.getSocket().on(EventMessage.WEBRTC_DISCONNECT, callback);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,13 +1,17 @@
|
|||||||
import {ConnectionInterface} from "../Connection";
|
import {ConnectionInterface, WebRtcDisconnectMessageInterface, WebRtcStartMessageInterface} from "../Connection";
|
||||||
import {MediaManager} from "./MediaManager";
|
import {MediaManager} from "./MediaManager";
|
||||||
import * as SimplePeerNamespace from "simple-peer";
|
import * as SimplePeerNamespace from "simple-peer";
|
||||||
let Peer: SimplePeerNamespace.SimplePeer = require('simple-peer');
|
let Peer: SimplePeerNamespace.SimplePeer = require('simple-peer');
|
||||||
|
|
||||||
class UserSimplePeer{
|
export interface UserSimplePeer{
|
||||||
userId: string;
|
userId: string;
|
||||||
name?: string;
|
name?: string;
|
||||||
initiator?: boolean;
|
initiator?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This class manages connections to all the peers in the same group as me.
|
||||||
|
*/
|
||||||
export class SimplePeer {
|
export class SimplePeer {
|
||||||
private Connection: ConnectionInterface;
|
private Connection: ConnectionInterface;
|
||||||
private WebRtcRoomId: string;
|
private WebRtcRoomId: string;
|
||||||
@ -40,7 +44,7 @@ export class SimplePeer {
|
|||||||
this.MediaManager.getCamera().then(() => {
|
this.MediaManager.getCamera().then(() => {
|
||||||
|
|
||||||
//receive message start
|
//receive message start
|
||||||
this.Connection.receiveWebrtcStart((message: any) => {
|
this.Connection.receiveWebrtcStart((message: WebRtcStartMessageInterface) => {
|
||||||
this.receiveWebrtcStart(message);
|
this.receiveWebrtcStart(message);
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -48,22 +52,26 @@ export class SimplePeer {
|
|||||||
console.error("err", err);
|
console.error("err", err);
|
||||||
});
|
});
|
||||||
|
|
||||||
//receive signal by gemer
|
this.Connection.disconnectMessage((data: WebRtcDisconnectMessageInterface): void => {
|
||||||
this.Connection.disconnectMessage((data: any) => {
|
|
||||||
this.closeConnection(data.userId);
|
this.closeConnection(data.userId);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
private receiveWebrtcStart(data: any) {
|
private receiveWebrtcStart(data: WebRtcStartMessageInterface) {
|
||||||
this.WebRtcRoomId = data.roomId;
|
this.WebRtcRoomId = data.roomId;
|
||||||
this.Users = data.clients;
|
this.Users = data.clients;
|
||||||
|
// Note: the clients array contain the list of all clients (event the ones we are already connected to in case a user joints a group)
|
||||||
|
// So we can receive a request we already had before. (which will abort at the first line of createPeerConnection)
|
||||||
|
// TODO: refactor this to only send a message to connect to one user (rather than several users).
|
||||||
|
// This would be symmetrical to the way we handle disconnection.
|
||||||
|
//console.log('Start message', data);
|
||||||
|
|
||||||
//start connection
|
//start connection
|
||||||
this.startWebRtc();
|
this.startWebRtc();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* server has two person connected, start the meet
|
* server has two people connected, start the meet
|
||||||
*/
|
*/
|
||||||
private startWebRtc() {
|
private startWebRtc() {
|
||||||
this.Users.forEach((user: UserSimplePeer) => {
|
this.Users.forEach((user: UserSimplePeer) => {
|
||||||
@ -83,6 +91,8 @@ export class SimplePeer {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//console.log("Creating connection with peer "+user.userId);
|
||||||
|
|
||||||
let name = user.name;
|
let name = user.name;
|
||||||
if(!name){
|
if(!name){
|
||||||
let userSearch = this.Users.find((userSearch: UserSimplePeer) => userSearch.userId === user.userId);
|
let userSearch = this.Users.find((userSearch: UserSimplePeer) => userSearch.userId === user.userId);
|
||||||
@ -93,7 +103,7 @@ export class SimplePeer {
|
|||||||
this.MediaManager.removeActiveVideo(user.userId);
|
this.MediaManager.removeActiveVideo(user.userId);
|
||||||
this.MediaManager.addActiveVideo(user.userId, name);
|
this.MediaManager.addActiveVideo(user.userId, name);
|
||||||
|
|
||||||
let peer : any = new Peer({
|
let peer : SimplePeerNamespace.Instance = new Peer({
|
||||||
initiator: user.initiator ? user.initiator : false,
|
initiator: user.initiator ? user.initiator : false,
|
||||||
reconnectTimer: 10000,
|
reconnectTimer: 10000,
|
||||||
config: {
|
config: {
|
||||||
@ -167,15 +177,25 @@ export class SimplePeer {
|
|||||||
this.addMedia(user.userId);
|
this.addMedia(user.userId);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This is triggered twice. Once by the server, and once by a remote client disconnecting
|
||||||
|
*
|
||||||
|
* @param userId
|
||||||
|
*/
|
||||||
private closeConnection(userId : string) {
|
private closeConnection(userId : string) {
|
||||||
try {
|
try {
|
||||||
this.MediaManager.removeActiveVideo(userId);
|
this.MediaManager.removeActiveVideo(userId);
|
||||||
if (!this.PeerConnectionArray.get(userId)) {
|
let peer = this.PeerConnectionArray.get(userId);
|
||||||
|
if (peer === undefined) {
|
||||||
|
console.warn("Tried to close connection for user "+userId+" but could not find user")
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
// @ts-ignore
|
// FIXME: I don't understand why "Closing connection with" message is displayed TWICE before "Nb users in peerConnectionArray"
|
||||||
this.PeerConnectionArray.get(userId)?.destroy();
|
// I do understand the method closeConnection is called twice, but I don't understand how they manage to run in parallel.
|
||||||
|
//console.log('Closing connection with '+userId);
|
||||||
|
peer.destroy();
|
||||||
this.PeerConnectionArray.delete(userId)
|
this.PeerConnectionArray.delete(userId)
|
||||||
|
//console.log('Nb users in peerConnectionArray '+this.PeerConnectionArray.size);
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
console.error("closeConnection", err)
|
console.error("closeConnection", err)
|
||||||
}
|
}
|
||||||
@ -216,7 +236,7 @@ export class SimplePeer {
|
|||||||
* @param userId
|
* @param userId
|
||||||
* @param stream
|
* @param stream
|
||||||
*/
|
*/
|
||||||
private stream(userId : any, stream: MediaStream) {
|
private stream(userId : string, stream: MediaStream) {
|
||||||
if(!stream){
|
if(!stream){
|
||||||
this.MediaManager.disabledVideoByUserId(userId);
|
this.MediaManager.disabledVideoByUserId(userId);
|
||||||
this.MediaManager.disabledMicrophoneByUserId(userId);
|
this.MediaManager.disabledMicrophoneByUserId(userId);
|
||||||
@ -231,7 +251,6 @@ export class SimplePeer {
|
|||||||
*/
|
*/
|
||||||
private addMedia (userId : any = null) {
|
private addMedia (userId : any = null) {
|
||||||
try {
|
try {
|
||||||
let transceiver : any = null;
|
|
||||||
let localStream: MediaStream|null = this.MediaManager.localStream;
|
let localStream: MediaStream|null = this.MediaManager.localStream;
|
||||||
let peer = this.PeerConnectionArray.get(userId);
|
let peer = this.PeerConnectionArray.get(userId);
|
||||||
if(localStream === null) {
|
if(localStream === null) {
|
||||||
|
Loading…
Reference in New Issue
Block a user