diff options
Diffstat (limited to 'tests/unit_node/net_test.ts')
-rw-r--r-- | tests/unit_node/net_test.ts | 201 |
1 files changed, 201 insertions, 0 deletions
diff --git a/tests/unit_node/net_test.ts b/tests/unit_node/net_test.ts new file mode 100644 index 000000000..60cf9d8fc --- /dev/null +++ b/tests/unit_node/net_test.ts @@ -0,0 +1,201 @@ +// Copyright 2018-2024 the Deno authors. All rights reserved. MIT license. + +import * as net from "node:net"; +import { assert, assertEquals } from "@test_util/std/assert/mod.ts"; +import * as path from "@test_util/std/path/mod.ts"; +import * as http from "node:http"; + +Deno.test("[node/net] close event emits after error event", async () => { + const socket = net.createConnection(27009, "doesnotexist"); + const events: ("error" | "close")[] = []; + const errorEmitted = Promise.withResolvers<void>(); + const closeEmitted = Promise.withResolvers<void>(); + socket.once("error", () => { + events.push("error"); + errorEmitted.resolve(); + }); + socket.once("close", () => { + events.push("close"); + closeEmitted.resolve(); + }); + await Promise.all([errorEmitted.promise, closeEmitted.promise]); + + // `error` happens before `close` + assertEquals(events, ["error", "close"]); +}); + +Deno.test("[node/net] the port is available immediately after close callback", async () => { + const deferred = Promise.withResolvers<void>(); + + // This simulates what get-port@5.1.1 does. + const getAvailablePort = (port: number) => + new Promise((resolve, reject) => { + const server = net.createServer(); + server.on("error", reject); + server.listen({ port }, () => { + // deno-lint-ignore no-explicit-any + const { port } = server.address() as any; + server.close(() => { + resolve(port); + }); + }); + }); + + const port = await getAvailablePort(5555); + + const httpServer = http.createServer(); + httpServer.on("error", (e) => deferred.reject(e)); + httpServer.listen(port, () => { + httpServer.close(() => deferred.resolve()); + }); + await deferred.promise; +}); + +Deno.test("[node/net] net.connect().unref() works", async () => { + const ctl = new AbortController(); + const server = Deno.serve({ + signal: ctl.signal, + handler: () => new Response("hello"), + onListen: async ({ port, hostname }) => { + const { stdout, stderr } = await new Deno.Command(Deno.execPath(), { + args: [ + "eval", + ` + import * as net from "node:net"; + const socket = net.connect(${port}, "${hostname}", () => { + console.log("connected"); + socket.unref(); + socket.on("data", (data) => console.log(data.toString())); + socket.write("GET / HTTP/1.1\\n\\n"); + }); + `, + ], + cwd: path.dirname(path.fromFileUrl(import.meta.url)), + }).output(); + if (stderr.length > 0) { + console.log(new TextDecoder().decode(stderr)); + } + assertEquals(new TextDecoder().decode(stdout), "connected\n"); + ctl.abort(); + }, + }); + await server.finished; +}); + +Deno.test({ + name: "[node/net] throws permission error instead of unknown error", + permissions: "none", + fn: () => { + try { + const s = new net.Server(); + s.listen(3000); + } catch (e) { + assert(e instanceof Deno.errors.PermissionDenied); + } + }, +}); + +Deno.test("[node/net] connection event has socket value", async () => { + const deferred = Promise.withResolvers<void>(); + const deferred2 = Promise.withResolvers<void>(); + + const server = net.createServer(); + server.on("error", deferred.reject); + server.on("connection", (socket) => { + assert(socket !== undefined); + socket.end(); + server.close(() => { + deferred.resolve(); + }); + }); + server.listen(async () => { + // deno-lint-ignore no-explicit-any + const { port } = server.address() as any; + + const conn = await Deno.connect({ + port, + transport: "tcp", + }); + + for await (const _ of conn.readable) { + // + } + + deferred2.resolve(); + }); + + await Promise.all([deferred.promise, deferred2.promise]); +}); + +/// We need to make sure that any shared buffers are never used concurrently by two reads. +// https://github.com/denoland/deno/issues/20188 +Deno.test("[node/net] multiple Sockets should get correct server data", async () => { + const socketCount = 9; + + class TestSocket { + dataReceived: ReturnType<typeof Promise.withResolvers<void>> = Promise + .withResolvers<void>(); + events: string[] = []; + socket: net.Socket | undefined; + } + + const finished = Promise.withResolvers<void>(); + const serverSocketsClosed: ReturnType<typeof Promise.withResolvers<void>>[] = + []; + const server = net.createServer(); + server.on("connection", (socket) => { + assert(socket !== undefined); + const i = serverSocketsClosed.push(Promise.withResolvers<void>()); + socket.on("data", (data) => { + socket.write(new TextDecoder().decode(data)); + }); + socket.on("close", () => { + serverSocketsClosed[i - 1].resolve(); + }); + }); + + const sockets: TestSocket[] = []; + for (let i = 0; i < socketCount; i++) { + sockets[i] = new TestSocket(); + } + + server.listen(async () => { + // deno-lint-ignore no-explicit-any + const { port } = server.address() as any; + + for (let i = 0; i < socketCount; i++) { + const socket = sockets[i].socket = net.createConnection(port); + socket.on("data", (data) => { + const count = sockets[i].events.length; + sockets[i].events.push(new TextDecoder().decode(data)); + if (count === 0) { + // Trigger an immediate second write + sockets[i].socket?.write(`${i}`.repeat(3)); + } else { + sockets[i].dataReceived.resolve(); + } + }); + } + + for (let i = 0; i < socketCount; i++) { + sockets[i].socket?.write(`${i}`.repeat(3)); + } + + await Promise.all(sockets.map((socket) => socket.dataReceived.promise)); + + for (let i = 0; i < socketCount; i++) { + sockets[i].socket?.end(); + } + + server.close(() => { + finished.resolve(); + }); + }); + + await finished.promise; + await Promise.all(serverSocketsClosed.map(({ promise }) => promise)); + + for (let i = 0; i < socketCount; i++) { + assertEquals(sockets[i].events, [`${i}`.repeat(3), `${i}`.repeat(3)]); + } +}); |