-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathserver.js
More file actions
311 lines (275 loc) · 9.13 KB
/
Copy pathserver.js
File metadata and controls
311 lines (275 loc) · 9.13 KB
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
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
// server.js
const express = require("express");
const http = require("http");
const path = require("path");
const { Server } = require("socket.io");
const winston = require("winston");
const { v4: uuidv4 } = require("uuid");
const os = require("os");
const dotenv = require("dotenv");
dotenv.config({ path: ".env" });
// --- Logging setup (keeps what you had) ---
const logger = winston.createLogger({
level: "info",
format: winston.format.combine(
winston.format.timestamp({ format: "YYYY-MM-DD HH:mm:ss" }),
winston.format.printf(
({ timestamp, level, message }) =>
`${timestamp} - ${level.toUpperCase()} - ${message}`
)
),
transports: [new winston.transports.Console()],
});
// --- App & server ---
const app = express();
const server = http.createServer(app);
const io = new Server(server, { cors: { origin: "*" } });
// Serve static public folder (expects public/index.html)
app.use(express.static(path.join(__dirname, "public")));
app.get("/", (req, res) => {
res.sendFile(path.join(__dirname, "public", "index.html"));
});
app.get("/keep-alive", (req, res) => {
// No body, no auth – just a tiny HTTP response
res.status(200).json({ alive: true, ts: Date.now() });
logger.info(`Keep-alive ping received (IP: ${req.ip})`);
});
// In-memory rooms: Map<roomId, { passkey: string|null, clients: Map<socketId, username> }>
const rooms = new Map();
io.on("connection", (socket) => {
logger.info(`Socket connected: ${socket.id}`);
// Send initial connected state
socket.emit("state", { step: "connected" });
// Host a room request: { private: boolean, passkey?: string }
socket.on("hostRoom", (payload) => {
try {
const passkey =
payload && payload.private ? payload.passkey || null : null;
const roomId = uuidv4();
rooms.set(roomId, { passkey, clients: new Map() });
// Put creator in a transient state until they set a username
socket.data.roomId = roomId;
socket.data.step = "awaiting-username";
// Structured event: room created
socket.emit("roomCreated", {
roomId,
passkeyRequired: !!passkey,
message: "Room created. Please set your username.",
});
logger.info(
`Room ${roomId} created (private=${!!passkey}) by ${socket.id}`
);
} catch (err) {
logger.error(`hostRoom error: ${err.message}`);
socket.emit("error", { message: "Unable to create room" });
}
});
// Join a room request: { roomId, passkey? }
socket.on("joinRoom", (payload) => {
try {
if (!payload || !payload.roomId) {
socket.emit("error", { message: "Missing roomId" });
return;
}
const roomId = payload.roomId;
const room = rooms.get(roomId);
if (!room) {
socket.emit("error", { message: "Room does not exist" });
return;
}
if (room.passkey && !payload.passkey) {
socket.emit("needPasskey", {
roomId,
message: "Passkey required to join this room",
});
return;
}
if (room.passkey && room.passkey !== payload.passkey) {
socket.emit("error", { message: "Invalid passkey" });
return;
}
// store on socket that they are joining and need to provide username
socket.data.roomId = roomId;
socket.data.step = "awaiting-username";
socket.emit("needUsername", {
roomId,
message: "Enter username to join",
});
} catch (err) {
logger.error(`joinRoom error: ${err.message}`);
socket.emit("error", { message: "Unable to join room" });
}
});
// Set username: { username }
socket.on("setUsername", (payload) => {
try {
if (!payload || !payload.username) {
socket.emit("error", { message: "Username cannot be empty" });
return;
}
const username = String(payload.username).trim();
const roomId = socket.data.roomId;
if (!roomId || !rooms.has(roomId)) {
socket.emit("error", { message: "Room not found or expired" });
return;
}
const room = rooms.get(roomId);
// Check for duplicate username in the same room (case-insensitive)
const isDuplicate = Array.from(room.clients.values()).some(
(name) => name.toLowerCase() === username.toLowerCase()
);
if (isDuplicate) {
socket.emit("needUsername", {
roomId,
message: "Username already in use, choose a different one",
});
return;
}
// Save username and join socket room
room.clients.set(socket.id, username);
socket.join(roomId);
socket.data.step = "chat";
// Notify everyone in room
const joinMsg = {
from: "system",
text: `*** ${username} joined the chat ***`,
timestamp: Date.now(),
};
io.to(roomId).emit("chatMessage", joinMsg);
socket.emit("joinedRoom", {
roomId,
username,
message: "Connection successful. Welcome!",
});
logger.info(`Socket ${socket.id} as ${username} joined room ${roomId}`);
} catch (err) {
logger.error(`setUsername error: ${err.message}`);
socket.emit("error", { message: "Unable to set username" });
}
});
// Send chat message: { text, repliedTo? }
socket.on("sendMessage", (payload) => {
try {
if (!payload || !payload.text) return;
const roomId = socket.data.roomId;
if (!roomId || !rooms.has(roomId)) {
socket.emit("error", { message: "Not in a room" });
return;
}
const room = rooms.get(roomId);
const username = room.clients.get(socket.id);
if (!username) {
socket.emit("error", { message: "Username not set" });
return;
}
const msg = {
messageId: uuidv4(), // Server-generated unique ID
from: username,
text: String(payload.text),
timestamp: Date.now(),
repliedTo: payload.repliedTo || null, // Full message object from frontend
};
io.to(roomId).emit("chatMessage", msg);
logger.info(`Message from ${username} in ${roomId}: ${payload.text}`);
} catch (err) {
logger.error(`sendMessage error: ${err.message}`);
socket.emit("error", { message: "Unable to send message" });
}
});
socket.on("ws-heartbeat", () => {
logger.info(
`WebSocket heartbeat from ${socket.id} in room ${
socket.data.roomId || "none"
}`
);
// Optional: Update lastSeen timestamp if you have reconnection logic
if (socket.data.lastSeen) socket.data.lastSeen = Date.now();
});
// Quit event (client leaving intentionally)
socket.on("quit", () => {
try {
const roomId = socket.data.roomId;
if (roomId && rooms.has(roomId)) {
const room = rooms.get(roomId);
const username = room.clients.get(socket.id);
if (username) {
room.clients.delete(socket.id);
io.to(roomId).emit("chatMessage", {
from: "system",
text: `*** ${username} left the chat ***`,
timestamp: Date.now(),
});
logger.info(`${username} quit room ${roomId}`);
if (room.clients.size === 0) {
rooms.delete(roomId);
logger.info(`Room ${roomId} deleted (empty)`);
}
}
}
socket.leave(socket.data.roomId || "");
socket.data.step = "disconnected";
socket.disconnect(true);
} catch (err) {
logger.error(`quit error: ${err.message}`);
}
});
socket.on("disconnect", (reason) => {
// Clean up membership
try {
const roomId = socket.data.roomId;
if (roomId && rooms.has(roomId)) {
const room = rooms.get(roomId);
const username = room.clients.get(socket.id);
if (username) {
room.clients.delete(socket.id);
io.to(roomId).emit("chatMessage", {
from: "system",
text: `*** ${username} left the chat ***`,
timestamp: Date.now(),
});
logger.info(
`${username} disconnected from ${roomId} (reason: ${reason})`
);
if (room.clients.size === 0) {
rooms.delete(roomId);
logger.info(`Room ${roomId} deleted (empty)`);
}
}
}
logger.info(`Socket disconnected: ${socket.id} (reason: ${reason})`);
} catch (err) {
logger.error(`disconnect cleanup error: ${err.message}`);
}
});
});
// Utility to get local IP for logging
function getServerIp() {
const interfaces = os.networkInterfaces();
for (const name of Object.keys(interfaces)) {
for (const iface of interfaces[name]) {
if (iface.family === "IPv4" && !iface.internal) {
return iface.address;
}
}
}
return "localhost";
}
const PORT = process.env.PORT || 3001;
server.listen(PORT, () => {
const ip = getServerIp();
logger.info(`Server listening on http://${ip}:${PORT}`);
});
process.on("SIGINT", () => {
logger.info("Shutting down server...");
for (const roomId of rooms.keys()) {
io.to(roomId).emit("chatMessage", {
from: "system",
text: "*** Server is shutting down ***",
timestamp: Date.now(),
});
}
io.close(() => {
logger.info("Server closed");
process.exit(0);
});
});