mirror of
https://github.com/denoland/deno.git
synced 2025-01-12 17:09:00 -05:00
d59599fc18
partially unblocks #25470 This PR aligns the resolution of `localhost` hostname to Node.js behavior. In Node.js `dns.lookup("localhost", (_, addr) => console.log(addr))` prints ipv6 address `::1`, but it prints ipv4 address `127.0.0.1` in Deno. That difference causes some errors in the work of enabling `createConnection` option in `http.request` (#25470). This PR fixes the issue by aligning `dns.lookup` behavior to Node.js. This PR also changes the following behaviors (resolving TODOs): - `http.createServer` now listens on ipv6 address `[::]` by default on linux/mac - `net.createServer` now listens on ipv6 address `[::]` by default on linux/mac These changes are also alignments to Node.js behaviors.
237 lines
6.5 KiB
TypeScript
237 lines
6.5 KiB
TypeScript
// Copyright 2018-2024 the Deno authors. All rights reserved. MIT license.
|
|
|
|
import {
|
|
assert,
|
|
assertEquals,
|
|
assertInstanceOf,
|
|
assertStringIncludes,
|
|
assertThrows,
|
|
} from "@std/assert";
|
|
import { delay } from "@std/async/delay";
|
|
import { fromFileUrl, join } from "@std/path";
|
|
import * as tls from "node:tls";
|
|
import * as net from "node:net";
|
|
import * as stream from "node:stream";
|
|
import { execCode } from "../unit/test_util.ts";
|
|
|
|
const tlsTestdataDir = fromFileUrl(
|
|
new URL("../testdata/tls", import.meta.url),
|
|
);
|
|
const key = Deno.readTextFileSync(join(tlsTestdataDir, "localhost.key"));
|
|
const cert = Deno.readTextFileSync(join(tlsTestdataDir, "localhost.crt"));
|
|
const rootCaCert = Deno.readTextFileSync(join(tlsTestdataDir, "RootCA.pem"));
|
|
|
|
for (
|
|
const [alpnServer, alpnClient, expected] of [
|
|
[["a", "b"], ["a"], ["a"]],
|
|
[["a", "b"], ["b"], ["b"]],
|
|
[["a", "b"], ["a", "b"], ["a"]],
|
|
[["a", "b"], [], []],
|
|
[[], ["a", "b"], []],
|
|
]
|
|
) {
|
|
Deno.test(`tls.connect sends correct ALPN: '${alpnServer}' + '${alpnClient}' = '${expected}'`, async () => {
|
|
const listener = Deno.listenTls({
|
|
hostname: "localhost",
|
|
port: 0,
|
|
key,
|
|
cert,
|
|
alpnProtocols: alpnServer,
|
|
});
|
|
const outgoing = tls.connect({
|
|
host: "::1",
|
|
servername: "localhost",
|
|
port: listener.addr.port,
|
|
ALPNProtocols: alpnClient,
|
|
secureContext: {
|
|
ca: rootCaCert,
|
|
// deno-lint-ignore no-explicit-any
|
|
} as any,
|
|
});
|
|
|
|
const conn = await listener.accept();
|
|
const handshake = await conn.handshake();
|
|
assertEquals(handshake.alpnProtocol, expected[0] || null);
|
|
conn.close();
|
|
outgoing.destroy();
|
|
listener.close();
|
|
await new Promise((resolve) => outgoing.on("close", resolve));
|
|
});
|
|
}
|
|
|
|
Deno.test("tls.connect makes tls connection", async () => {
|
|
const ctl = new AbortController();
|
|
let port;
|
|
const serve = Deno.serve({
|
|
hostname: "localhost",
|
|
port: 0,
|
|
key,
|
|
cert,
|
|
signal: ctl.signal,
|
|
onListen: (listen) => port = listen.port,
|
|
}, () => new Response("hello"));
|
|
|
|
await delay(200);
|
|
|
|
const conn = tls.connect({
|
|
host: "::1",
|
|
servername: "localhost",
|
|
port,
|
|
secureContext: {
|
|
ca: rootCaCert,
|
|
// deno-lint-ignore no-explicit-any
|
|
} as any,
|
|
});
|
|
conn.write(`GET / HTTP/1.1
|
|
Host: localhost
|
|
Connection: close
|
|
|
|
`);
|
|
const chunk = Promise.withResolvers<Uint8Array>();
|
|
conn.on("data", (received) => {
|
|
conn.destroy();
|
|
ctl.abort();
|
|
chunk.resolve(received);
|
|
});
|
|
|
|
await serve.finished;
|
|
|
|
const text = new TextDecoder().decode(await chunk.promise);
|
|
const bodyText = text.split("\r\n\r\n").at(-1)?.trim();
|
|
assertEquals(bodyText, "hello");
|
|
});
|
|
|
|
// https://github.com/denoland/deno/pull/20120
|
|
Deno.test("tls.connect mid-read tcp->tls upgrade", async () => {
|
|
const { promise, resolve } = Promise.withResolvers<void>();
|
|
const ctl = new AbortController();
|
|
const serve = Deno.serve({
|
|
hostname: "localhost",
|
|
port: 8443,
|
|
key,
|
|
cert,
|
|
signal: ctl.signal,
|
|
}, () => new Response("hello"));
|
|
|
|
await delay(200);
|
|
|
|
const conn = tls.connect({
|
|
host: "::1",
|
|
servername: "localhost",
|
|
port: 8443,
|
|
secureContext: {
|
|
ca: rootCaCert,
|
|
// deno-lint-ignore no-explicit-any
|
|
} as any,
|
|
});
|
|
|
|
conn.setEncoding("utf8");
|
|
conn.write(`GET / HTTP/1.1\nHost: www.google.com\n\n`);
|
|
|
|
conn.on("data", (_) => {
|
|
conn.destroy();
|
|
ctl.abort();
|
|
});
|
|
conn.on("close", resolve);
|
|
|
|
await serve.finished;
|
|
await promise;
|
|
});
|
|
|
|
Deno.test("tls.createServer creates a TLS server", async () => {
|
|
const deferred = Promise.withResolvers<void>();
|
|
const server = tls.createServer(
|
|
// deno-lint-ignore no-explicit-any
|
|
{ host: "0.0.0.0", key, cert } as any,
|
|
(socket: net.Socket) => {
|
|
socket.write("welcome!\n");
|
|
socket.setEncoding("utf8");
|
|
socket.pipe(socket).on("data", (data) => {
|
|
if (data.toString().trim() === "goodbye") {
|
|
socket.destroy();
|
|
}
|
|
});
|
|
socket.on("close", () => deferred.resolve());
|
|
},
|
|
);
|
|
server.listen(0, async () => {
|
|
const tcpConn = await Deno.connect({
|
|
// deno-lint-ignore no-explicit-any
|
|
port: (server.address() as any).port,
|
|
});
|
|
const conn = await Deno.startTls(tcpConn, {
|
|
hostname: "localhost",
|
|
caCerts: [rootCaCert],
|
|
});
|
|
|
|
const buf = new Uint8Array(100);
|
|
await conn.read(buf);
|
|
let text: string;
|
|
text = new TextDecoder().decode(buf);
|
|
assertEquals(text.replaceAll("\0", ""), "welcome!\n");
|
|
buf.fill(0);
|
|
|
|
await conn.write(new TextEncoder().encode("hey\n"));
|
|
await conn.read(buf);
|
|
text = new TextDecoder().decode(buf);
|
|
assertEquals(text.replaceAll("\0", ""), "hey\n");
|
|
buf.fill(0);
|
|
|
|
await conn.write(new TextEncoder().encode("goodbye\n"));
|
|
await conn.read(buf);
|
|
text = new TextDecoder().decode(buf);
|
|
assertEquals(text.replaceAll("\0", ""), "goodbye\n");
|
|
|
|
conn.close();
|
|
server.close();
|
|
});
|
|
await deferred.promise;
|
|
});
|
|
|
|
Deno.test("TLSSocket can construct without options", () => {
|
|
// deno-lint-ignore no-explicit-any
|
|
new tls.TLSSocket(new stream.PassThrough() as any);
|
|
});
|
|
|
|
Deno.test("tlssocket._handle._parentWrap is set", () => {
|
|
// Note: This feature is used in popular 'http2-wrapper' module
|
|
// https://github.com/szmarczak/http2-wrapper/blob/51eeaf59ff9344fb192b092241bfda8506983620/source/utils/js-stream-socket.js#L6
|
|
const parentWrap =
|
|
// deno-lint-ignore no-explicit-any
|
|
((new tls.TLSSocket(new stream.PassThrough() as any, {}) as any)
|
|
// deno-lint-ignore no-explicit-any
|
|
._handle as any)!
|
|
._parentWrap;
|
|
assertInstanceOf(parentWrap, stream.PassThrough);
|
|
});
|
|
|
|
Deno.test("tls.connect() throws InvalidData when there's error in certificate", async () => {
|
|
// Uses execCode to avoid `--unsafely-ignore-certificate-errors` option applied
|
|
const [status, output] = await execCode(`
|
|
import tls from "node:tls";
|
|
const conn = tls.connect({
|
|
host: "localhost",
|
|
port: 4557,
|
|
});
|
|
|
|
conn.on("error", (err) => {
|
|
console.log(err);
|
|
});
|
|
`);
|
|
|
|
assertEquals(status, 0);
|
|
assertStringIncludes(
|
|
output,
|
|
"InvalidData: invalid peer certificate: UnknownIssuer",
|
|
);
|
|
});
|
|
|
|
Deno.test("tls.rootCertificates is not empty", () => {
|
|
assert(tls.rootCertificates.length > 0);
|
|
assert(Object.isFrozen(tls.rootCertificates));
|
|
assert(tls.rootCertificates instanceof Array);
|
|
assert(tls.rootCertificates.every((cert) => typeof cert === "string"));
|
|
assertThrows(() => {
|
|
(tls.rootCertificates as string[]).push("new cert");
|
|
}, TypeError);
|
|
});
|