forked from hcrlab/stretch_web_interface
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathsignaling_sockets.js
200 lines (166 loc) · 8.46 KB
/
signaling_sockets.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
// Some of this code may have been derived from code featured in the following article:
// https://www.html5rocks.com/en/tutorials/webrtc/infrastructure/
function createSignalingSocket(io) {
var numClients = 0;
var connected_robots = new Set();
var available_robots = new Set();
var namespace = '/';
function sendAvailableRobotsUpdate(socket) {
// let operators know about available robots
console.log('letting operators know about available robots');
console.log('available_robots =');
console.log(available_robots);
var robots = Array.from(available_robots.values());
socket.to('operators').emit('available robots', robots);
}
io.on('connection', function(socket){
////////////////
// see
// https://www.codementor.io/tips/0217388244/sharing-passport-js-sessions-with-both-express-and-socket-io
// for more information about socket.io using passport middleware
console.log('new socket.io connection');
console.log('socket.handshake = ');
console.log(socket.handshake);
var user = socket.request.user;
var role = user.role;
var robot_operator_room = 'none';
if(role === 'robot') {
var robot_name = user.username;
var room = robot_name; // use the robot's name as the robot's room identifier
console.log('A ROBOT HAS CONNECTED');
console.log('intended room name = ' + room);
// If the robot's room already exists, disconnect all
// sockets in the room. For example, this will disconnect
// operators that were connected to the robot in a
// previous session.
io.of(namespace).in(room).disconnectSockets(true);
// Add this robot's socket to the following two rooms:
// 1) a new room named after the robot used to pair with an operator
// 2) a room named "robots" to which all robots are added
socket.join([room, 'robots']);
robot_operator_room = room;
console.log('adding robot to the "robots" room');
console.log('creating room for the robot and having it join the room');
connected_robots.add(robot_name);
available_robots.add(robot_name);
// let operators know about the new robot
sendAvailableRobotsUpdate(socket);
// io.to(room).emit('a room for the robot has been created, and the robot is in it'); // broadcast to everyone in the room
console.log('connected robots = ' + Array.from(connected_robots).join(' '))
console.log('available robots = ' + Array.from(available_robots).join(' '))
} else {
if(role === 'operator') {
console.log('AN OPERATOR HAS CONNECTED');
// create the robot operator pairing room and add to the robots room
socket.join('operators', () => {
console.log('adding operator to the "operators" room');
});
// let operator know about available robots
var robots = Array.from(connected_robots.values());
console.log('available_robots =');
console.log(available_robots);
var robots = Array.from(available_robots.values());
socket.emit('available robots', robots);
}
}
// https://github.com/socketio/socket.io/blob/master/docs/API.md
// "A client always connects to / (the main namespace), then
// potentially connect to other namespaces (while using the
// same underlying connection)."
// convenience function to log server messages on the client
socket.on('what robots are available', function() {
if (role === 'operator') {
console.log('operator has requested the available robots');
console.log('available_robots =', available_robots);
var robots = Array.from(available_robots.values());
socket.emit('available robots', robots);
} else {
console.log('NO REPLY SENT: non-operator requested the available robots');
}
});
socket.on('bye', message => {
if (role === 'operator' && robot_operator_room !== 'none') {
socket.to(robot_operator_room).emit('bye');
console.log('Attempting to have the operator leave the robot room.');
console.log('');
socket.leave(robot_operator_room);
available_robots.add(robot_operator_room);
robot_operator_room = 'none';
sendAvailableRobotsUpdate(socket);
} else {
console.error("WRONG")
}
})
socket.on('signalling', function (message) {
//console.log('Client sent WebRTC message: ', message);
if (robot_operator_room !== 'none') {
//console.log('sending WebRTC message to any other clients in the room named "' +
// robot_operator_room + '".');
socket.to(robot_operator_room).emit('signalling', message);
} else {
console.log('robot_operator_room is none, so there is nobody to send the WebRTC message to');
}
});
socket.on('join', function (room) {
console.log('Received request to join room ' + room);
if (!io.sockets.adapter.rooms.get(room)) {
console.warn("Someone tried to join the nonresistant room " + room)
return;
}
numClients = io.sockets.adapter.rooms.get(room).size
console.log('Requested room ' + room + ' currently has ' + numClients + ' client(s)');
if (numClients < 1) {
//socket.join(room);
console.log('*********************************************');
console.log('RECEIVED REQUEST TO JOIN A NON-EXISTENT ROOM');
console.log('THIS IS UNUSUAL AND SHOULD BE AVOIDED');
console.log('Client ID ' + socket.id + ' created room ' + room);
console.log('DOING NOTHING...');
console.log('Apparently, no robot exists with the requested name');
console.log('Since no room exists with the requested name');
console.log('*********************************************');
//socket.emit('created', room, socket.id);
} else if (numClients < 2) {
console.log('Client ID ' + socket.id + ' joined room ' + room);
io.sockets.in(room).emit('join', room);
socket.join(room);
available_robots.delete(room);
robot_operator_room = room;
sendAvailableRobotsUpdate(socket);
socket.emit('joined', room, socket.id);
io.sockets.in(room).emit('ready');
} else { // max two clients
socket.emit('full', room);
}
});
socket.on('disconnect', function(){
console.log('socket disconnected');
if(user.role === 'robot') {
var robot_name = user.username;
console.log('ROBOT "' + robot_name + '" DISCONNECTED');
console.log('attempting to delete it from the set');
robot_operator_room = 'none';
// could this result in deleting an element of the set
// that should be there because of asynchronous
// execution?
connected_robots.delete(robot_name);
available_robots.delete(robot_name);
sendAvailableRobotsUpdate(socket); // might be good to include this in an object that tracks the robots
} else {
if (robot_operator_room !== "none") {
socket.to(robot_operator_room).emit('bye');
available_robots.add(robot_operator_room);
robot_operator_room = 'none';
sendAvailableRobotsUpdate(socket);
}
}
console.log('user disconnected');
});
});
// "Disconnects this client. If value of close is true, closes the underlying connection."
// - https://socket.io/docs/server-api/
// console.log('disconnecting...');
// socket.disconnect(true)
};
////////////////////////////////////////////////////////
module.exports = createSignalingSocket;