2024-01-01 14:58:21 -05:00
|
|
|
// Copyright 2018-2024 the Deno authors. All rights reserved. MIT license.
|
2023-06-06 06:29:55 -04:00
|
|
|
|
2024-08-20 15:14:37 -04:00
|
|
|
// deno-lint-ignore-file no-console
|
|
|
|
|
2023-06-06 06:29:55 -04:00
|
|
|
import * as http2 from "node:http2";
|
2024-05-29 02:04:34 -04:00
|
|
|
import { Buffer } from "node:buffer";
|
|
|
|
import { readFile } from "node:fs/promises";
|
|
|
|
import { join } from "node:path";
|
2023-06-06 06:29:55 -04:00
|
|
|
import * as net from "node:net";
|
2024-07-25 01:30:28 -04:00
|
|
|
import { assert, assertEquals } from "@std/assert";
|
2024-03-07 08:58:46 -05:00
|
|
|
import { curlRequest } from "../unit/test_util.ts";
|
2023-06-06 06:29:55 -04:00
|
|
|
|
2024-09-27 10:07:20 -04:00
|
|
|
for (const url of ["http://localhost:4246", "https://localhost:4247"]) {
|
2023-09-15 15:51:25 -04:00
|
|
|
Deno.test(`[node/http2 client] ${url}`, {
|
|
|
|
ignore: Deno.build.os === "windows",
|
|
|
|
}, async () => {
|
|
|
|
// Create a server to respond to the HTTP2 requests
|
|
|
|
const client = http2.connect(url, {});
|
|
|
|
client.on("error", (err) => console.error(err));
|
2023-06-06 06:29:55 -04:00
|
|
|
|
2023-09-15 15:51:25 -04:00
|
|
|
const req = client.request({ ":method": "POST", ":path": "/" }, {
|
|
|
|
waitForTrailers: true,
|
|
|
|
});
|
2023-06-06 06:29:55 -04:00
|
|
|
|
2023-09-15 15:51:25 -04:00
|
|
|
let receivedTrailers;
|
|
|
|
let receivedHeaders;
|
|
|
|
let receivedData = "";
|
2023-06-06 06:29:55 -04:00
|
|
|
|
2023-09-15 15:51:25 -04:00
|
|
|
req.on("response", (headers, _flags) => {
|
|
|
|
receivedHeaders = headers;
|
|
|
|
});
|
2023-06-06 06:29:55 -04:00
|
|
|
|
2023-09-15 15:51:25 -04:00
|
|
|
req.write("hello");
|
|
|
|
req.setEncoding("utf8");
|
2023-06-06 06:29:55 -04:00
|
|
|
|
2023-09-15 15:51:25 -04:00
|
|
|
req.on("wantTrailers", () => {
|
|
|
|
req.sendTrailers({ foo: "bar" });
|
|
|
|
});
|
2023-06-06 06:29:55 -04:00
|
|
|
|
2023-09-15 15:51:25 -04:00
|
|
|
req.on("trailers", (trailers, _flags) => {
|
|
|
|
receivedTrailers = trailers;
|
|
|
|
});
|
|
|
|
|
|
|
|
req.on("data", (chunk) => {
|
|
|
|
receivedData += chunk;
|
|
|
|
});
|
|
|
|
req.end();
|
|
|
|
|
2023-11-22 06:11:20 -05:00
|
|
|
const { promise, resolve } = Promise.withResolvers<void>();
|
2023-09-15 15:51:25 -04:00
|
|
|
setTimeout(() => {
|
|
|
|
try {
|
|
|
|
client.close();
|
|
|
|
} catch (_) {
|
|
|
|
// pass
|
|
|
|
}
|
2023-11-22 06:11:20 -05:00
|
|
|
resolve();
|
2023-09-15 15:51:25 -04:00
|
|
|
}, 2000);
|
|
|
|
|
2023-11-22 06:11:20 -05:00
|
|
|
await promise;
|
2023-09-15 15:51:25 -04:00
|
|
|
assertEquals(receivedHeaders, { ":status": 200 });
|
|
|
|
assertEquals(receivedData, "hello world\n");
|
|
|
|
|
|
|
|
assertEquals(receivedTrailers, {
|
|
|
|
"abc": "def",
|
|
|
|
"opr": "stv",
|
|
|
|
"foo": "bar",
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
2023-06-06 06:29:55 -04:00
|
|
|
|
2023-10-12 10:03:19 -04:00
|
|
|
Deno.test(`[node/http2 client createConnection]`, {
|
|
|
|
ignore: Deno.build.os === "windows",
|
|
|
|
}, async () => {
|
|
|
|
const url = "http://127.0.0.1:4246";
|
2023-11-22 06:11:20 -05:00
|
|
|
const createConnDeferred = Promise.withResolvers<void>();
|
2023-10-12 10:03:19 -04:00
|
|
|
// Create a server to respond to the HTTP2 requests
|
|
|
|
const client = http2.connect(url, {
|
|
|
|
createConnection() {
|
|
|
|
const socket = net.connect({ host: "127.0.0.1", port: 4246 });
|
|
|
|
|
|
|
|
socket.on("connect", () => {
|
2023-11-22 06:11:20 -05:00
|
|
|
createConnDeferred.resolve();
|
2023-10-12 10:03:19 -04:00
|
|
|
});
|
|
|
|
|
|
|
|
return socket;
|
|
|
|
},
|
|
|
|
});
|
|
|
|
client.on("error", (err) => console.error(err));
|
|
|
|
|
|
|
|
const req = client.request({ ":method": "POST", ":path": "/" });
|
|
|
|
|
|
|
|
let receivedData = "";
|
|
|
|
|
|
|
|
req.write("hello");
|
|
|
|
req.setEncoding("utf8");
|
|
|
|
|
|
|
|
req.on("data", (chunk) => {
|
|
|
|
receivedData += chunk;
|
|
|
|
});
|
|
|
|
req.end();
|
|
|
|
|
2023-11-22 06:11:20 -05:00
|
|
|
const endPromise = Promise.withResolvers<void>();
|
2023-10-12 10:03:19 -04:00
|
|
|
setTimeout(() => {
|
|
|
|
try {
|
|
|
|
client.close();
|
|
|
|
} catch (_) {
|
|
|
|
// pass
|
|
|
|
}
|
|
|
|
endPromise.resolve();
|
|
|
|
}, 2000);
|
|
|
|
|
2023-11-22 06:11:20 -05:00
|
|
|
await createConnDeferred.promise;
|
|
|
|
await endPromise.promise;
|
2023-10-12 10:03:19 -04:00
|
|
|
assertEquals(receivedData, "hello world\n");
|
|
|
|
});
|
|
|
|
|
2024-02-21 07:43:01 -05:00
|
|
|
Deno.test("[node/http2 client GET https://www.example.com]", async () => {
|
|
|
|
const clientSession = http2.connect("https://www.example.com");
|
|
|
|
const req = clientSession.request({
|
|
|
|
":method": "GET",
|
|
|
|
":path": "/",
|
|
|
|
});
|
|
|
|
let headers = {};
|
|
|
|
let status: number | undefined = 0;
|
|
|
|
let chunk = new Uint8Array();
|
|
|
|
const endPromise = Promise.withResolvers<void>();
|
|
|
|
req.on("response", (h) => {
|
|
|
|
status = h[":status"];
|
|
|
|
headers = h;
|
|
|
|
});
|
|
|
|
req.on("data", (c) => {
|
|
|
|
chunk = c;
|
|
|
|
});
|
|
|
|
req.on("end", () => {
|
|
|
|
clientSession.close();
|
|
|
|
req.close();
|
|
|
|
endPromise.resolve();
|
|
|
|
});
|
|
|
|
req.end();
|
|
|
|
await endPromise.promise;
|
|
|
|
assert(Object.keys(headers).length > 0);
|
|
|
|
assertEquals(status, 200);
|
|
|
|
assert(chunk.length > 0);
|
|
|
|
});
|
2024-03-07 08:58:46 -05:00
|
|
|
|
|
|
|
Deno.test("[node/http2.createServer()]", {
|
|
|
|
// TODO(satyarohith): enable the test on windows.
|
|
|
|
ignore: Deno.build.os === "windows",
|
|
|
|
}, async () => {
|
|
|
|
const server = http2.createServer((_req, res) => {
|
|
|
|
res.setHeader("Content-Type", "text/html");
|
|
|
|
res.setHeader("X-Foo", "bar");
|
|
|
|
res.writeHead(200, { "Content-Type": "text/plain; charset=utf-8" });
|
|
|
|
res.write("Hello, World!");
|
|
|
|
res.end();
|
|
|
|
});
|
|
|
|
server.listen(0);
|
2024-09-27 10:07:20 -04:00
|
|
|
const port = (server.address() as net.AddressInfo).port;
|
2024-03-07 08:58:46 -05:00
|
|
|
const endpoint = `http://localhost:${port}`;
|
|
|
|
|
|
|
|
const response = await curlRequest([
|
|
|
|
endpoint,
|
|
|
|
"--http2-prior-knowledge",
|
|
|
|
]);
|
|
|
|
assertEquals(response, "Hello, World!");
|
|
|
|
server.close();
|
|
|
|
// Wait to avoid leaking the timer from here
|
|
|
|
// https://github.com/denoland/deno/blob/749b6e45e58ac87188027f79fe403d130f86bd73/ext/node/polyfills/net.ts#L2396-L2402
|
|
|
|
// Issue: https://github.com/denoland/deno/issues/22764
|
|
|
|
await new Promise<void>((resolve) => server.on("close", resolve));
|
|
|
|
});
|
2024-05-29 02:04:34 -04:00
|
|
|
|
|
|
|
Deno.test("[node/http2 client] write image buffer on request stream works", async () => {
|
|
|
|
const url = "https://localhost:5545";
|
|
|
|
const client = http2.connect(url);
|
|
|
|
client.on("error", (err) => console.error(err));
|
|
|
|
|
|
|
|
const imagePath = join(import.meta.dirname!, "testdata", "green.jpg");
|
|
|
|
const buffer = await readFile(imagePath);
|
|
|
|
const req = client.request({ ":method": "POST", ":path": "/echo_server" });
|
|
|
|
req.write(buffer, (err) => {
|
|
|
|
if (err) throw err;
|
|
|
|
});
|
|
|
|
|
|
|
|
let receivedData: Buffer;
|
|
|
|
req.on("data", (chunk) => {
|
|
|
|
if (!receivedData) {
|
|
|
|
receivedData = chunk;
|
|
|
|
} else {
|
|
|
|
receivedData = Buffer.concat([receivedData, chunk]);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
req.end();
|
|
|
|
|
|
|
|
const endPromise = Promise.withResolvers<void>();
|
|
|
|
setTimeout(() => {
|
|
|
|
try {
|
|
|
|
client.close();
|
|
|
|
} catch (_) {
|
|
|
|
// pass
|
|
|
|
}
|
|
|
|
endPromise.resolve();
|
|
|
|
}, 2000);
|
|
|
|
|
|
|
|
await endPromise.promise;
|
|
|
|
assertEquals(receivedData!, buffer);
|
|
|
|
});
|
2024-07-15 07:40:59 -04:00
|
|
|
|
|
|
|
Deno.test("[node/http2 client] write 512kb buffer on request stream works", async () => {
|
|
|
|
const url = "https://localhost:5545";
|
|
|
|
const client = http2.connect(url);
|
|
|
|
client.on("error", (err) => console.error(err));
|
|
|
|
|
|
|
|
const filePath = join(
|
|
|
|
import.meta.dirname!,
|
|
|
|
"testdata",
|
|
|
|
"lorem_ipsum_512kb.txt",
|
|
|
|
);
|
|
|
|
const buffer = await readFile(filePath);
|
|
|
|
const req = client.request({ ":method": "POST", ":path": "/echo_server" });
|
|
|
|
req.write(buffer, (err) => {
|
|
|
|
if (err) throw err;
|
|
|
|
});
|
|
|
|
|
|
|
|
let receivedData: Buffer;
|
|
|
|
req.on("data", (chunk) => {
|
|
|
|
if (!receivedData) {
|
|
|
|
receivedData = chunk;
|
|
|
|
} else {
|
|
|
|
receivedData = Buffer.concat([receivedData, chunk]);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
req.end();
|
|
|
|
|
|
|
|
const endPromise = Promise.withResolvers<void>();
|
|
|
|
setTimeout(() => {
|
|
|
|
try {
|
|
|
|
client.close();
|
|
|
|
} catch (_) {
|
|
|
|
// pass
|
|
|
|
}
|
|
|
|
endPromise.resolve();
|
|
|
|
}, 2000);
|
|
|
|
|
|
|
|
await endPromise.promise;
|
|
|
|
assertEquals(receivedData!, buffer);
|
|
|
|
});
|
2024-07-29 07:33:55 -04:00
|
|
|
|
|
|
|
// https://github.com/denoland/deno/issues/24678
|
|
|
|
Deno.test("[node/http2 client] deno doesn't panic on uppercase headers", async () => {
|
|
|
|
const url = "http://127.0.0.1:4246";
|
|
|
|
const client = http2.connect(url);
|
|
|
|
client.on("error", (err) => console.error(err));
|
|
|
|
|
|
|
|
// The "User-Agent" header has uppercase characters to test the panic.
|
|
|
|
const req = client.request({
|
|
|
|
":method": "POST",
|
|
|
|
":path": "/",
|
|
|
|
"User-Agent": "http2",
|
|
|
|
});
|
|
|
|
const endPromise = Promise.withResolvers<void>();
|
|
|
|
|
|
|
|
let receivedData = "";
|
|
|
|
|
|
|
|
req.write("hello");
|
|
|
|
req.setEncoding("utf8");
|
|
|
|
|
|
|
|
req.on("data", (chunk) => {
|
|
|
|
receivedData += chunk;
|
|
|
|
});
|
|
|
|
req.on("end", () => {
|
|
|
|
req.close();
|
|
|
|
client.close();
|
|
|
|
endPromise.resolve();
|
|
|
|
});
|
|
|
|
req.end();
|
|
|
|
await endPromise.promise;
|
|
|
|
assertEquals(receivedData, "hello world\n");
|
|
|
|
});
|
2024-08-14 17:59:22 -04:00
|
|
|
|
|
|
|
Deno.test("[node/http2 ClientHttp2Session.socket]", async () => {
|
|
|
|
const url = "http://127.0.0.1:4246";
|
|
|
|
const client = http2.connect(url);
|
|
|
|
client.on("error", (err) => console.error(err));
|
|
|
|
|
|
|
|
const req = client.request({ ":method": "POST", ":path": "/" });
|
|
|
|
const endPromise = Promise.withResolvers<void>();
|
|
|
|
|
|
|
|
// test that we can access session.socket
|
|
|
|
client.socket.setTimeout(10000);
|
|
|
|
// nodejs allows setting arbitrary properties
|
|
|
|
// deno-lint-ignore no-explicit-any
|
|
|
|
(client.socket as any).nonExistant = 9001;
|
|
|
|
// deno-lint-ignore no-explicit-any
|
|
|
|
assertEquals((client.socket as any).nonExistant, 9001);
|
|
|
|
|
|
|
|
// regular request dance to make sure it keeps working
|
|
|
|
let receivedData = "";
|
|
|
|
req.write("hello");
|
|
|
|
req.setEncoding("utf8");
|
|
|
|
|
|
|
|
req.on("data", (chunk) => {
|
|
|
|
receivedData += chunk;
|
|
|
|
});
|
|
|
|
req.on("end", () => {
|
|
|
|
req.close();
|
|
|
|
client.close();
|
|
|
|
endPromise.resolve();
|
|
|
|
});
|
|
|
|
req.end();
|
|
|
|
await endPromise.promise;
|
|
|
|
assertEquals(client.socket.remoteAddress, "127.0.0.1");
|
|
|
|
assertEquals(client.socket.remotePort, 4246);
|
|
|
|
assertEquals(client.socket.remoteFamily, "IPv4");
|
|
|
|
client.socket.setTimeout(0);
|
|
|
|
assertEquals(receivedData, "hello world\n");
|
|
|
|
});
|
2024-08-30 13:46:17 -04:00
|
|
|
|
|
|
|
Deno.test("[node/http2 client] connection states", async () => {
|
|
|
|
const expected = {
|
|
|
|
beforeConnect: { connecting: true, closed: false, destroyed: false },
|
|
|
|
afterConnect: { connecting: false, closed: false, destroyed: false },
|
|
|
|
afterClose: { connecting: false, closed: true, destroyed: false },
|
|
|
|
afterDestroy: { connecting: false, closed: true, destroyed: true },
|
|
|
|
};
|
|
|
|
const actual: Partial<typeof expected> = {};
|
|
|
|
|
|
|
|
const url = "http://127.0.0.1:4246";
|
|
|
|
const connectPromise = Promise.withResolvers<void>();
|
|
|
|
const client = http2.connect(url, {}, () => {
|
|
|
|
connectPromise.resolve();
|
|
|
|
});
|
|
|
|
client.on("error", (err) => console.error(err));
|
|
|
|
|
|
|
|
// close event happens after destory has been called
|
|
|
|
const destroyPromise = Promise.withResolvers<void>();
|
|
|
|
client.on("close", () => {
|
|
|
|
destroyPromise.resolve();
|
|
|
|
});
|
|
|
|
|
|
|
|
actual.beforeConnect = {
|
|
|
|
connecting: client.connecting,
|
|
|
|
closed: client.closed,
|
|
|
|
destroyed: client.destroyed,
|
|
|
|
};
|
|
|
|
|
|
|
|
await connectPromise.promise;
|
|
|
|
actual.afterConnect = {
|
|
|
|
connecting: client.connecting,
|
|
|
|
closed: client.closed,
|
|
|
|
destroyed: client.destroyed,
|
|
|
|
};
|
|
|
|
|
|
|
|
// leave a request open to prevent immediate destroy
|
|
|
|
const req = client.request();
|
|
|
|
req.on("data", () => {});
|
|
|
|
req.on("error", (err) => console.error(err));
|
|
|
|
const reqClosePromise = Promise.withResolvers<void>();
|
|
|
|
req.on("close", () => {
|
|
|
|
reqClosePromise.resolve();
|
|
|
|
});
|
|
|
|
|
|
|
|
client.close();
|
|
|
|
actual.afterClose = {
|
|
|
|
connecting: client.connecting,
|
|
|
|
closed: client.closed,
|
|
|
|
destroyed: client.destroyed,
|
|
|
|
};
|
|
|
|
|
|
|
|
await destroyPromise.promise;
|
|
|
|
actual.afterDestroy = {
|
|
|
|
connecting: client.connecting,
|
|
|
|
closed: client.closed,
|
|
|
|
destroyed: client.destroyed,
|
|
|
|
};
|
|
|
|
|
|
|
|
await reqClosePromise.promise;
|
|
|
|
|
|
|
|
assertEquals(actual, expected);
|
|
|
|
});
|
2024-09-11 22:03:57 -04:00
|
|
|
|
|
|
|
Deno.test("request and response exports", () => {
|
|
|
|
assert(http2.Http2ServerRequest);
|
|
|
|
assert(http2.Http2ServerResponse);
|
|
|
|
});
|