mirror of
https://github.com/socketio/socket.io.git
synced 2026-01-09 15:08:12 -05:00
274 lines
7.7 KiB
JavaScript
274 lines
7.7 KiB
JavaScript
const expect = require("expect.js");
|
|
const { Socket, NodeXHR, NodeWebSocket } = require("../");
|
|
const {
|
|
isIE11,
|
|
isAndroid,
|
|
isEdge,
|
|
isIPad,
|
|
useFetch,
|
|
} = require("./support/env");
|
|
const FakeTimers = require("@sinonjs/fake-timers");
|
|
const { repeat } = require("./util");
|
|
|
|
describe("Socket", function () {
|
|
this.timeout(10000);
|
|
|
|
describe("filterUpgrades", () => {
|
|
it("should return only available transports", () => {
|
|
const socket = new Socket({ transports: ["polling"] });
|
|
expect(socket._filterUpgrades(["polling", "websocket"])).to.eql([
|
|
"polling",
|
|
]);
|
|
socket.close();
|
|
});
|
|
});
|
|
|
|
it("throws an error when no transports are available", (done) => {
|
|
const socket = new Socket({ transports: [] });
|
|
let errorMessage = "";
|
|
socket.on("error", (error) => {
|
|
errorMessage = error;
|
|
});
|
|
setTimeout(() => {
|
|
expect(errorMessage).to.be("No transports available");
|
|
socket.close();
|
|
done();
|
|
});
|
|
});
|
|
|
|
it("should connect with the 2nd transport if tryAllTransports is `true` (polling)", (done) => {
|
|
const socket = new Socket({
|
|
transports: ["websocket", "polling"],
|
|
transportOptions: {
|
|
websocket: {
|
|
query: {
|
|
deny: 1,
|
|
},
|
|
},
|
|
},
|
|
tryAllTransports: true,
|
|
});
|
|
|
|
socket.on("open", () => {
|
|
expect(socket.transport.name).to.eql("polling");
|
|
socket.close();
|
|
done();
|
|
});
|
|
});
|
|
|
|
it("should connect with the 2nd transport if tryAllTransports is `true` (websocket)", (done) => {
|
|
const socket = new Socket({
|
|
transports: ["polling", "websocket"],
|
|
transportOptions: {
|
|
polling: {
|
|
query: {
|
|
deny: 1,
|
|
},
|
|
},
|
|
},
|
|
tryAllTransports: true,
|
|
});
|
|
|
|
socket.on("open", () => {
|
|
expect(socket.transport.name).to.eql("websocket");
|
|
socket.close();
|
|
done();
|
|
});
|
|
});
|
|
|
|
it("should not connect with the 2nd transport if tryAllTransports is `false`", (done) => {
|
|
const socket = new Socket({
|
|
transports: ["polling", "websocket"],
|
|
transportOptions: {
|
|
polling: {
|
|
query: {
|
|
deny: 1,
|
|
},
|
|
},
|
|
},
|
|
});
|
|
|
|
socket.on("error", (err) => {
|
|
expect(err.message).to.eql(
|
|
useFetch ? "fetch read error" : "xhr poll error"
|
|
);
|
|
done();
|
|
});
|
|
});
|
|
|
|
it("should connect with a custom transport implementation (polling)", (done) => {
|
|
const socket = new Socket({
|
|
transports: [NodeXHR],
|
|
});
|
|
|
|
socket.on("open", () => {
|
|
expect(socket.transport.name).to.eql("polling");
|
|
socket.close();
|
|
done();
|
|
});
|
|
});
|
|
|
|
it("should connect with a custom transport implementation (websocket)", (done) => {
|
|
const socket = new Socket({
|
|
transports: [NodeWebSocket],
|
|
});
|
|
|
|
socket.on("open", () => {
|
|
expect(socket.transport.name).to.eql("websocket");
|
|
socket.close();
|
|
done();
|
|
});
|
|
});
|
|
|
|
describe("fake timers", function () {
|
|
before(function () {
|
|
if (isIE11 || isAndroid || isEdge || isIPad) {
|
|
this.skip();
|
|
}
|
|
});
|
|
|
|
it("uses window timeout by default", (done) => {
|
|
const clock = FakeTimers.install();
|
|
const socket = new Socket({ transports: [] });
|
|
let errorMessage = "";
|
|
socket.on("error", (error) => {
|
|
errorMessage = error;
|
|
});
|
|
clock.tick(1); // Should trigger error emit.
|
|
expect(errorMessage).to.be("No transports available");
|
|
clock.uninstall();
|
|
socket.close();
|
|
done();
|
|
});
|
|
|
|
it.skip("uses custom timeout when provided", (done) => {
|
|
const clock = FakeTimers.install();
|
|
const socket = new Socket({
|
|
transports: [],
|
|
useNativeTimers: true,
|
|
});
|
|
|
|
let errorMessage = "";
|
|
socket.on("error", (error) => {
|
|
errorMessage = error;
|
|
});
|
|
socket.open();
|
|
// Socket should not use the mocked clock, so this should have no side
|
|
// effects.
|
|
clock.tick(1);
|
|
expect(errorMessage).to.be("");
|
|
clock.uninstall();
|
|
|
|
setTimeout(() => {
|
|
try {
|
|
expect(errorMessage).to.be("No transports available");
|
|
socket.close();
|
|
done();
|
|
} finally {
|
|
}
|
|
}, 1);
|
|
});
|
|
});
|
|
|
|
describe("close", () => {
|
|
it("provides details when maxHttpBufferSize is reached (polling)", (done) => {
|
|
const socket = new Socket({ transports: ["polling"] });
|
|
socket.on("open", () => {
|
|
socket.send(repeat("a", 101)); // over the maxHttpBufferSize value of the server
|
|
});
|
|
|
|
socket.on("error", (err) => {
|
|
expect(err).to.be.an(Error);
|
|
expect(err.type).to.eql("TransportError");
|
|
expect(err.description).to.eql(413);
|
|
if (useFetch) {
|
|
expect(err.message).to.eql("fetch write error");
|
|
} else {
|
|
expect(err.message).to.eql("xhr post error");
|
|
// err.context is a XMLHttpRequest object
|
|
expect(err.context.readyState).to.eql(4);
|
|
expect(err.context.responseText).to.eql("");
|
|
}
|
|
});
|
|
|
|
socket.on("close", (reason, details) => {
|
|
expect(reason).to.eql("transport error");
|
|
expect(details).to.be.an(Error);
|
|
done();
|
|
});
|
|
});
|
|
|
|
it("provides details when maxHttpBufferSize is reached (websocket)", (done) => {
|
|
const socket = new Socket({ transports: ["websocket"] });
|
|
socket.on("open", () => {
|
|
socket.send(repeat("a", 101)); // over the maxHttpBufferSize value of the server
|
|
});
|
|
|
|
socket.on("close", (reason, details) => {
|
|
if (isIE11) {
|
|
expect(reason).to.eql("transport error");
|
|
expect(details).to.be.an(Error);
|
|
} else {
|
|
expect(reason).to.eql("transport close");
|
|
expect(details.description).to.eql("websocket connection closed");
|
|
// details.context is a CloseEvent object
|
|
expect(details.context.code).to.eql(1009); // "Message Too Big" (see https://developer.mozilla.org/en-US/docs/Web/API/CloseEvent/code)
|
|
expect(details.context.reason).to.eql("");
|
|
// note: details.context.wasClean is false in the browser, but true in Node.js
|
|
}
|
|
done();
|
|
});
|
|
});
|
|
|
|
it("provides details when the session ID is unknown (polling)", (done) => {
|
|
const socket = new Socket({
|
|
transports: ["polling"],
|
|
query: { sid: "abc" },
|
|
});
|
|
|
|
socket.on("error", (err) => {
|
|
expect(err).to.be.an(Error);
|
|
expect(err.type).to.eql("TransportError");
|
|
expect(err.description).to.eql(400);
|
|
if (useFetch) {
|
|
expect(err.message).to.eql("fetch read error");
|
|
} else {
|
|
expect(err.message).to.eql("xhr poll error");
|
|
// err.context is a XMLHttpRequest object
|
|
expect(err.context.readyState).to.eql(4);
|
|
expect(err.context.responseText).to.eql(
|
|
'{"code":1,"message":"Session ID unknown"}'
|
|
);
|
|
}
|
|
});
|
|
|
|
socket.on("close", (reason, details) => {
|
|
expect(reason).to.eql("transport error");
|
|
expect(details).to.be.an(Error);
|
|
done();
|
|
});
|
|
});
|
|
|
|
it("provides details when the session ID is unknown (websocket)", (done) => {
|
|
const socket = new Socket({
|
|
transports: ["websocket"],
|
|
query: { sid: "abc" },
|
|
});
|
|
|
|
socket.on("error", (err) => {
|
|
expect(err).to.be.an(Error);
|
|
expect(err.type).to.eql("TransportError");
|
|
expect(err.message).to.eql("websocket error");
|
|
// err.description is a generic Event
|
|
expect(err.description.type).to.be("error");
|
|
});
|
|
|
|
socket.on("close", (reason, details) => {
|
|
expect(reason).to.eql("transport error");
|
|
expect(details).to.be.an(Error);
|
|
done();
|
|
});
|
|
});
|
|
});
|
|
});
|