-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathServerSocket.js
223 lines (202 loc) · 7.87 KB
/
ServerSocket.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
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
const net = require('net');
const utils = require('./utils');
const _ = require('lodash');
const scanner = require('readline').createInterface({
input: process.stdin,
output: process.stdout
});
/**
* Server in which clients connect to communicate with one another.
* This server broadcasts all standard messages it receives to all other
* clients currently connected to the server.
*
* This server also broadcasts "event" messages, such as anyone joining or leaving
* the chat room, going AFK, and changing of one's nickname.
* @class Server
*/
class Server {
/**
* Creates an instance of Server and sets necessary event listeners.
* @param {Number} MAX_NUM_CONNECTIONS The max number of connections allowed to a chat room
* @memberof Server
*/
constructor(MAX_NUM_CONNECTIONS) {
this.sockets = []; // All the clients connected to the server.
this.MAX_NUM_CONNECTIONS = MAX_NUM_CONNECTIONS; // The max number of connections allowed
this.numConnections = 0; // The current number of connections to the server.
// Creates the server and listens for any incoming connections
net.createServer((s) => {
// Room capacity meet
if (this.numConnections >= this.MAX_NUM_CONNECTIONS) {
s.end("Room at capacity");
}
else {
// Sets up a socket object with event listeners and adds it to the current connections
let clientObj = {
socket: s,
id: ++this.numConnections,
nickname: 'PLACEHOLDER',
isAFK: false
};
this.setSocketEvents(clientObj);
this.sockets.push(clientObj);
}
})
.on('error', (error) => {
console.error(error.message);
})
.listen(8080, () => {
console.log('Listening on port 8080');
});
}
/**
* This sets up the standard socket events to listen for, based on the net library.
* @param {ClientObject} clientObj
* @memberof Server
*/
setSocketEvents(clientObj) {
const self = this;
// Listens for any data that it receives
clientObj.socket.on('data', (data) => {
if (data.length > 0) {
// Checks if the message is a slash command
let message = utils.getCommand(data.toString());
if (message) {
self.processCommand(clientObj, message.command, message.argument);
}
else {
message = data.toString();
message = utils.formatMessage(clientObj.nickname, message);
self.broadcast(clientObj.id, message);
console.log(message);
}
}
});
// When a socket disconnects from the server
clientObj.socket.on('end', (socket) => {
// Since this checks for disconnects of any kind
// A special case is in place for automatically
// disconnecting sockets when the room is at capacity.
if (self.numConnections <= self.MAX_NUM_CONNECTIONS) {
let timeStamp = utils.getTimestamp();
let message = `${timeStamp} ${clientObj.nickname} left the room`;
self.removeSocket(clientObj);
console.log(message);
self.broadcast(clientObj.id, message);
}
});
}
/**
* Gets the current users in the room
* @param {ClientObj} client Client that currently is checking the users in the room
* @returns String that contains the current users in the room.
* @memberof Server
*/
getUsersInRoom(client) {
let message;
// Checks if there are any other clients in the room.
if (this.sockets.length > 1) {
message = "Users in the room: ";
this.sockets.forEach(c => {
if (!_.isEqual(client, c)) {
message += `${c.nickname}, `;
}
});
message = message.slice(0, message.length - 2);
}
else {
message = `You're the only user in the room. Invite your friends!`;
}
return message;
}
/**
* Parses commands and does the appropriate action based on the command.
* @param {ClientObject} clientObj The client object sending the command.
* @param {String} command The command being sent.
* @param {String} argument The argument for the command, which is optional.
* @memberof Server
*/
processCommand(clientObj, command, argument) {
let broadcastMessage;
let userMessage;
let timeStamp = utils.getTimestamp();
switch (command) {
case utils.COMMANDS.nick:
// We use PLACEHOLDER as the first name to determine whether or not
// a client is connecting for the first time or not.
if (clientObj.nickname !== 'PLACEHOLDER') {
broadcastMessage = `${timeStamp} ${clientObj.nickname} changed their nickname to ${argument}`;
}
else {
userMessage = this.getUsersInRoom(clientObj);
broadcastMessage = `${timeStamp} ${argument} joined the room`;
}
clientObj.nickname = argument;
break;
case utils.COMMANDS.afk:
clientObj.isAFK = !clientObj.isAFK;
broadcastMessage = clientObj.isAFK ? `${clientObj.nickname} went AFK` :
`${clientObj.nickname} is back!`;
broadcastMessage = utils.formatEventMessage(broadcastMessage);
userMessage = clientObj.isAFK ? `You have gone AFK. Type /afk to receive messages again` :
`You are not AFK anymore. Type /afk to go AFK again`;
break;
case utils.COMMANDS.exit:
this.removeSocket(clientObj);
clientObj.socket.end();
default:
broadcastMessage = utils.formatMessage(clientObj.nickname, `/${command} ${argument}`);
}
if (broadcastMessage)
this.broadcast(clientObj.id, broadcastMessage);
if (userMessage)
clientObj.socket.write(userMessage);
console.log(broadcastMessage);
}
/**
* Broadcasts the message given to all other clients connected.
*
* @param {any} senderId The Id of the user sending the message
* @param {any} message The message being sent to all other clients
* @memberof Server
*/
broadcast(senderId, message) {
if (this.sockets.length === 0) {
console.log(`${utils.getTimestamp()} No users in the chat`);
return;
}
// Doesn't broadcast a message to a client if they are AFK or they are the
// the client in question.
this.sockets.forEach((client) => {
if (client.id === senderId || client.isAFK) return;
client.socket.write(message);
});
}
/**
* Removes the socket from the current connections list.
* @param {any} socket
* @memberof Server
*/
removeSocket(socket) {
this.sockets.splice(this.sockets.indexOf(socket), 1);
this.numConnections--;
}
}
/**
* Starts the server after getting the room capacity for the chat room.
*/
function main() {
process.stdout.write('Please enter the room capacity\n> ');
scanner.on('line', (capacity) => {
let size = Number.parseInt(capacity);
if (isNaN(size)) {
process.stdout.write('Please enter a valid integer\n> ');
}
else {
scanner.close();
process.stdin.destroy();
new Server(size);
}
});
}
main();