Skip to content

Commit 9d86397

Browse files
sebamarynissendarrachequesne
authored andcommitted
fix: fix race condition in dynamic namespaces (#4137)
Using an async operation with `io.use()` could lead to the creation of several instances of a same namespace, each of them overriding the previous one. Example: ```js io.use(async (nsp, auth, next) => { await anOperationThatTakesSomeTime(); next(); }); ``` Related: #4136
1 parent 44e20ba commit 9d86397

File tree

3 files changed

+53
-9
lines changed

3 files changed

+53
-9
lines changed

lib/client.ts

-1
Original file line numberDiff line numberDiff line change
@@ -116,7 +116,6 @@ export class Client<
116116
| false
117117
) => {
118118
if (dynamicNspName) {
119-
debug("dynamic namespace %s was created", dynamicNspName);
120119
this.doConnect(name, auth);
121120
} else {
122121
debug("creation of namespace %s was denied", name);

lib/index.ts

+11-8
Original file line numberDiff line numberDiff line change
@@ -202,15 +202,18 @@ export class Server<
202202
}
203203
nextFn.value(name, auth, (err, allow) => {
204204
if (err || !allow) {
205-
run();
206-
} else {
207-
const namespace = this.parentNsps
208-
.get(nextFn.value)!
209-
.createChild(name);
210-
// @ts-ignore
211-
this.sockets.emitReserved("new_namespace", namespace);
212-
fn(namespace);
205+
return run();
213206
}
207+
if (this._nsps.has(name)) {
208+
// the namespace was created in the meantime
209+
debug("dynamic namespace %s already exists", name);
210+
return fn(this._nsps.get(name) as Namespace);
211+
}
212+
const namespace = this.parentNsps.get(nextFn.value)!.createChild(name);
213+
debug("dynamic namespace %s was created", name);
214+
// @ts-ignore
215+
this.sockets.emitReserved("new_namespace", namespace);
216+
fn(namespace);
214217
});
215218
};
216219

test/socket.io.ts

+42
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,8 @@ import { io as ioc, Socket as ClientSocket } from "socket.io-client";
1515
import "./support/util";
1616
import "./utility-methods";
1717

18+
type callback = (err: Error | null, success: boolean) => void;
19+
1820
// Creates a socket.io client for the given server
1921
function client(srv, nsp?: string | object, opts?: object): ClientSocket {
2022
if ("object" == typeof nsp) {
@@ -998,6 +1000,46 @@ describe("socket.io", () => {
9981000
const socket = client(srv, "/dynamic-101");
9991001
});
10001002
});
1003+
1004+
it("should handle race conditions with dynamic namespaces (#4136)", (done) => {
1005+
const srv = createServer();
1006+
const sio = new Server(srv);
1007+
const counters = {
1008+
connected: 0,
1009+
created: 0,
1010+
events: 0,
1011+
};
1012+
const buffer: callback[] = [];
1013+
sio.on("new_namespace", (namespace) => {
1014+
counters.created++;
1015+
});
1016+
srv.listen(() => {
1017+
const handler = () => {
1018+
if (++counters.events === 2) {
1019+
expect(counters.created).to.equal(1);
1020+
done();
1021+
}
1022+
};
1023+
1024+
sio
1025+
.of((name, query, next) => {
1026+
buffer.push(next);
1027+
if (buffer.length === 2) {
1028+
buffer.forEach((next) => next(null, true));
1029+
}
1030+
})
1031+
.on("connection", (socket) => {
1032+
if (++counters.connected === 2) {
1033+
sio.of("/dynamic-101").emit("message");
1034+
}
1035+
});
1036+
1037+
let one = client(srv, "/dynamic-101");
1038+
let two = client(srv, "/dynamic-101");
1039+
one.on("message", handler);
1040+
two.on("message", handler);
1041+
});
1042+
});
10011043
});
10021044
});
10031045

0 commit comments

Comments
 (0)