mirror of
https://github.com/oven-sh/bun
synced 2026-02-10 02:48:50 +00:00
495 lines
13 KiB
TypeScript
495 lines
13 KiB
TypeScript
import { ServerWebSocket, TCPSocket, Socket as _BunSocket, TCPSocketListener } from "bun";
|
|
import { afterAll, afterEach, beforeAll, beforeEach, describe, expect, it } from "bun:test";
|
|
import { connect, isIP, isIPv4, isIPv6, Socket, createConnection, Server } from "net";
|
|
import { realpathSync, mkdtempSync } from "fs";
|
|
import { tmpdir } from "os";
|
|
import { join } from "path";
|
|
import { bunEnv, bunExe } from "harness";
|
|
|
|
const socket_domain = mkdtempSync(join(realpathSync(tmpdir()), "node-net"));
|
|
|
|
it("should support net.isIP()", () => {
|
|
expect(isIP("::1")).toBe(6);
|
|
expect(isIP("foobar")).toBe(0);
|
|
expect(isIP("127.0.0.1")).toBe(4);
|
|
expect(isIP("127.0.0.1/24")).toBe(0);
|
|
expect(isIP("127.000.000.001")).toBe(0);
|
|
});
|
|
|
|
it("should support net.isIPv4()", () => {
|
|
expect(isIPv4("::1")).toBe(false);
|
|
expect(isIPv4("foobar")).toBe(false);
|
|
expect(isIPv4("127.0.0.1")).toBe(true);
|
|
expect(isIPv4("127.0.0.1/24")).toBe(false);
|
|
expect(isIPv4("127.000.000.001")).toBe(false);
|
|
});
|
|
|
|
it("should support net.isIPv6()", () => {
|
|
expect(isIPv6("::1")).toBe(true);
|
|
expect(isIPv6("foobar")).toBe(false);
|
|
expect(isIPv6("127.0.0.1")).toBe(false);
|
|
expect(isIPv6("127.0.0.1/24")).toBe(false);
|
|
expect(isIPv6("127.000.000.001")).toBe(false);
|
|
});
|
|
|
|
describe("net.Socket read", () => {
|
|
var unix_servers = 0;
|
|
for (let [message, label] of [
|
|
["Hello World!".repeat(1024), "long message"],
|
|
["Hello!", "short message"],
|
|
]) {
|
|
describe(label, () => {
|
|
function runWithServer(cb: (..._: any[]) => void, unix_domain_path?: any) {
|
|
return (done: (_: any) => void) => {
|
|
function drain(socket: _BunSocket<{ message: string }>) {
|
|
const message = socket.data.message;
|
|
const written = socket.write(message);
|
|
if (written < message.length) {
|
|
socket.data.message = message.slice(written);
|
|
} else {
|
|
socket.end();
|
|
}
|
|
}
|
|
|
|
var server = unix_domain_path
|
|
? Bun.listen({
|
|
unix: join(unix_domain_path, `${unix_servers++}.sock`),
|
|
socket: {
|
|
open(socket) {
|
|
socket.data.message = message;
|
|
drain(socket);
|
|
},
|
|
drain,
|
|
error(socket, err) {
|
|
done(err);
|
|
},
|
|
},
|
|
data: {
|
|
message: "",
|
|
},
|
|
})
|
|
: Bun.listen({
|
|
hostname: "localhost",
|
|
port: 0,
|
|
socket: {
|
|
open(socket) {
|
|
socket.data.message = message;
|
|
drain(socket);
|
|
},
|
|
drain,
|
|
error(socket, err) {
|
|
done(err);
|
|
},
|
|
},
|
|
data: {
|
|
message: "",
|
|
},
|
|
});
|
|
|
|
function onDone(err: any) {
|
|
server.stop();
|
|
done(err);
|
|
}
|
|
|
|
try {
|
|
cb(server, drain, onDone);
|
|
} catch (e) {
|
|
onDone(e);
|
|
}
|
|
};
|
|
}
|
|
|
|
it(
|
|
"should work with .connect(port)",
|
|
runWithServer((server, drain, done) => {
|
|
var data = "";
|
|
const socket = new Socket()
|
|
.connect(server.port)
|
|
.on("connect", () => {
|
|
expect(socket).toBeDefined();
|
|
expect(socket.connecting).toBe(false);
|
|
})
|
|
.setEncoding("utf8")
|
|
.on("data", chunk => {
|
|
data += chunk;
|
|
})
|
|
.on("end", () => {
|
|
try {
|
|
expect(data).toBe(message);
|
|
done();
|
|
} catch (e) {
|
|
server.stop();
|
|
done(e);
|
|
}
|
|
})
|
|
.on("error", done);
|
|
}),
|
|
);
|
|
|
|
it(
|
|
"should work with .connect(port, listener)",
|
|
runWithServer((server, drain, done) => {
|
|
var data = "";
|
|
const socket = new Socket()
|
|
.connect(server.port, () => {
|
|
expect(socket).toBeDefined();
|
|
expect(socket.connecting).toBe(false);
|
|
})
|
|
.setEncoding("utf8")
|
|
.on("data", chunk => {
|
|
data += chunk;
|
|
})
|
|
.on("end", () => {
|
|
try {
|
|
expect(data).toBe(message);
|
|
done();
|
|
} catch (e) {
|
|
server.stop();
|
|
done(e);
|
|
}
|
|
})
|
|
.on("error", done);
|
|
}),
|
|
);
|
|
|
|
it(
|
|
"should work with .connect(port, host, listener)",
|
|
runWithServer((server, drain, done) => {
|
|
var data = "";
|
|
const socket = new Socket()
|
|
.connect(server.port, "localhost", () => {
|
|
expect(socket).toBeDefined();
|
|
expect(socket.connecting).toBe(false);
|
|
})
|
|
.setEncoding("utf8")
|
|
.on("data", chunk => {
|
|
data += chunk;
|
|
})
|
|
.on("end", () => {
|
|
try {
|
|
expect(data).toBe(message);
|
|
done();
|
|
} catch (e) {
|
|
done(e);
|
|
}
|
|
})
|
|
.on("error", done);
|
|
}),
|
|
);
|
|
|
|
it(
|
|
"should work with .createConnection(path)",
|
|
runWithServer((server, drain, done) => {
|
|
var data = "";
|
|
const socket = createConnection(server.unix)
|
|
.on("connect", () => {
|
|
expect(socket).toBeDefined();
|
|
expect(socket.connecting).toBe(false);
|
|
})
|
|
.setEncoding("utf8")
|
|
.on("data", chunk => {
|
|
data += chunk;
|
|
})
|
|
.on("end", () => {
|
|
try {
|
|
expect(data).toBe(message);
|
|
done();
|
|
} catch (e) {
|
|
server.stop();
|
|
done(e);
|
|
}
|
|
})
|
|
.on("error", done);
|
|
}, socket_domain),
|
|
);
|
|
it(
|
|
"should work with .connect(path)",
|
|
runWithServer((server, drain, done) => {
|
|
var data = "";
|
|
const socket = new Socket()
|
|
.connect(server.unix)
|
|
.on("connect", () => {
|
|
expect(socket).toBeDefined();
|
|
expect(socket.connecting).toBe(false);
|
|
})
|
|
.setEncoding("utf8")
|
|
.on("data", chunk => {
|
|
data += chunk;
|
|
})
|
|
.on("end", () => {
|
|
try {
|
|
expect(data).toBe(message);
|
|
done();
|
|
} catch (e) {
|
|
server.stop();
|
|
done(e);
|
|
}
|
|
})
|
|
.on("error", done);
|
|
}, socket_domain),
|
|
);
|
|
|
|
it(
|
|
"should work with .connect(path, listener)",
|
|
runWithServer((server, drain, done) => {
|
|
var data = "";
|
|
const socket = new Socket()
|
|
.connect(server.unix, () => {
|
|
expect(socket).toBeDefined();
|
|
expect(socket.connecting).toBe(false);
|
|
})
|
|
.setEncoding("utf8")
|
|
.on("data", chunk => {
|
|
data += chunk;
|
|
})
|
|
.on("end", () => {
|
|
try {
|
|
expect(data).toBe(message);
|
|
done();
|
|
} catch (e) {
|
|
done(e);
|
|
}
|
|
})
|
|
.on("error", done);
|
|
}, socket_domain),
|
|
);
|
|
|
|
it(
|
|
"should support onread callback",
|
|
runWithServer((server, drain, done) => {
|
|
var data = "";
|
|
const options = {
|
|
host: server.hostname,
|
|
port: server.port,
|
|
onread: {
|
|
buffer: Buffer.alloc(4096),
|
|
callback: (size, buf) => {
|
|
data += buf.slice(0, size).toString("utf8");
|
|
},
|
|
},
|
|
};
|
|
const socket = createConnection(options, () => {
|
|
expect(socket).toBeDefined();
|
|
expect(socket.connecting).toBe(false);
|
|
})
|
|
.on("end", () => {
|
|
try {
|
|
expect(data).toBe(message);
|
|
done();
|
|
} catch (e) {
|
|
done(e);
|
|
}
|
|
})
|
|
.on("error", done);
|
|
}),
|
|
);
|
|
});
|
|
}
|
|
});
|
|
|
|
describe("net.Socket write", () => {
|
|
const message = "Hello World!".repeat(1024);
|
|
|
|
function runWithServer(cb: (..._: any[]) => void) {
|
|
return (done: (_?: any) => void) => {
|
|
let server: TCPSocketListener<unknown>;
|
|
|
|
function close(socket: _BunSocket<Buffer[]>) {
|
|
expect(Buffer.concat(socket.data).toString("utf8")).toBe(message);
|
|
server.stop();
|
|
done();
|
|
}
|
|
|
|
var leaky;
|
|
server = Bun.listen({
|
|
hostname: "0.0.0.0",
|
|
port: 0,
|
|
socket: {
|
|
close,
|
|
data(socket, buffer) {
|
|
leaky = socket;
|
|
if (!Buffer.isBuffer(buffer)) {
|
|
done(new Error("buffer is not a Buffer"));
|
|
}
|
|
|
|
socket.data.push(buffer);
|
|
},
|
|
end: close,
|
|
error(socket, err) {
|
|
leaky = socket;
|
|
done(err);
|
|
},
|
|
open(socket) {
|
|
leaky = socket;
|
|
socket.data = [];
|
|
},
|
|
},
|
|
data: [] as Buffer[],
|
|
});
|
|
|
|
function onDone(err: any) {
|
|
server.stop();
|
|
done(err);
|
|
}
|
|
|
|
try {
|
|
cb(server, onDone);
|
|
} catch (e) {
|
|
onDone(e);
|
|
}
|
|
};
|
|
}
|
|
|
|
it(
|
|
"should work with .end(data)",
|
|
runWithServer((server, done) => {
|
|
const socket = new Socket()
|
|
.connect(server.port, server.hostname)
|
|
.on("ready", () => {
|
|
expect(socket).toBeDefined();
|
|
expect(socket.connecting).toBe(false);
|
|
})
|
|
.on("error", done)
|
|
.end(message);
|
|
}),
|
|
);
|
|
|
|
it(
|
|
"should work with .write(data).end()",
|
|
runWithServer((server, done) => {
|
|
const socket = new Socket()
|
|
.connect(server.port, server.hostname, () => {
|
|
expect(socket).toBeDefined();
|
|
expect(socket.connecting).toBe(false);
|
|
})
|
|
.on("error", done);
|
|
socket.write(message);
|
|
socket.end();
|
|
}),
|
|
);
|
|
|
|
it(
|
|
"should work with multiple .write()s",
|
|
runWithServer((server, done) => {
|
|
const socket = new Socket()
|
|
.connect(server.port, server.hostname, () => {
|
|
expect(socket).toBeDefined();
|
|
expect(socket.connecting).toBe(false);
|
|
})
|
|
.on("error", done);
|
|
const size = 10;
|
|
for (let i = 0; i < message.length; i += size) {
|
|
socket.write(message.slice(i, i + size));
|
|
}
|
|
socket.end();
|
|
}),
|
|
);
|
|
|
|
it("should allow reconnecting after end()", async () => {
|
|
const server = new Server(socket => socket.end());
|
|
const port = await new Promise(resolve => {
|
|
server.once("listening", () => resolve(server.address().port));
|
|
server.listen();
|
|
});
|
|
|
|
const socket = new Socket();
|
|
socket.on("data", data => console.log(data.toString()));
|
|
socket.on("error", err => console.error(err));
|
|
|
|
async function run() {
|
|
return new Promise((resolve, reject) => {
|
|
socket.once("connect", (...args) => {
|
|
socket.write("script\n", err => {
|
|
if (err) return reject(err);
|
|
socket.end(() => setTimeout(resolve, 3));
|
|
});
|
|
});
|
|
socket.connect(port, "127.0.0.1");
|
|
});
|
|
}
|
|
|
|
for (let i = 0; i < 10; i++) {
|
|
await run();
|
|
}
|
|
});
|
|
});
|
|
|
|
it("should handle connection error", done => {
|
|
let errored = false;
|
|
|
|
// @ts-ignore
|
|
const socket = connect(55555, () => {
|
|
done(new Error("Should not have connected"));
|
|
});
|
|
|
|
socket.on("error", error => {
|
|
if (errored) {
|
|
return done(new Error("Should not have errored twice"));
|
|
}
|
|
errored = true;
|
|
expect(error).toBeDefined();
|
|
expect(error.message).toBe("Failed to connect");
|
|
expect((error as any).code).toBe("ECONNREFUSED");
|
|
});
|
|
|
|
socket.on("connect", () => {
|
|
done(new Error("Should not have connected"));
|
|
});
|
|
|
|
socket.on("close", () => {
|
|
expect(errored).toBe(true);
|
|
done();
|
|
});
|
|
});
|
|
|
|
it("should handle connection error (unix)", done => {
|
|
let errored = false;
|
|
|
|
// @ts-ignore
|
|
const socket = connect("loser", () => {
|
|
done(new Error("Should not have connected"));
|
|
});
|
|
|
|
socket.on("error", error => {
|
|
if (errored) {
|
|
return done(new Error("Should not have errored twice"));
|
|
}
|
|
errored = true;
|
|
expect(error).toBeDefined();
|
|
expect(error.message).toBe("Failed to connect");
|
|
expect((error as any).code).toBe("ENOENT");
|
|
});
|
|
|
|
socket.on("connect", () => {
|
|
done(new Error("Should not have connected"));
|
|
});
|
|
|
|
socket.on("close", () => {
|
|
expect(errored).toBe(true);
|
|
done();
|
|
});
|
|
});
|
|
|
|
it("Socket has a prototype", () => {
|
|
function Connection() {}
|
|
function Connection2() {}
|
|
require("util").inherits(Connection, Socket);
|
|
require("util").inherits(Connection2, require("tls").TLSSocket);
|
|
});
|
|
|
|
it("unref should exit when no more work pending", async () => {
|
|
const process = Bun.spawn({
|
|
cmd: [bunExe(), join(import.meta.dir, "node-unref-fixture.js")],
|
|
env: bunEnv,
|
|
});
|
|
expect(await process.exited).toBe(0);
|
|
});
|
|
|
|
it("socket should keep process alive if unref is not called", async () => {
|
|
const process = Bun.spawn({
|
|
cmd: [bunExe(), join(import.meta.dir, "node-ref-default-fixture.js")],
|
|
env: bunEnv,
|
|
});
|
|
expect(await process.exited).toBe(1);
|
|
});
|