Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Undici #5117

Merged
merged 23 commits into from
Jun 28, 2022
Merged

Undici #5117

Show file tree
Hide file tree
Changes from 19 commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 5 additions & 0 deletions .changeset/slimy-ways-stare.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
---
'@sveltejs/kit': patch
---

[breaking] use undici instead of node-fetch
2 changes: 1 addition & 1 deletion packages/kit/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -33,7 +33,6 @@
"locate-character": "^2.0.5",
"marked": "^4.0.16",
"mime": "^3.0.0",
"node-fetch": "^3.2.4",
"port-authority": "^1.2.0",
"rollup": "^2.75.3",
"selfsigned": "^2.0.1",
Expand All @@ -45,6 +44,7 @@
"svelte2tsx": "~0.5.10",
"tiny-glob": "^0.2.9",
"typescript": "^4.7.2",
"undici": "^5.4.0",
"uvu": "^0.5.3"
},
"peerDependencies": {
Expand Down
29 changes: 22 additions & 7 deletions packages/kit/src/node/index.js
Original file line number Diff line number Diff line change
@@ -1,4 +1,3 @@
import { Readable } from 'stream';
import * as set_cookie_parser from 'set-cookie-parser';

/** @param {import('http').IncomingMessage} req */
Expand Down Expand Up @@ -85,13 +84,29 @@ export async function setResponse(res, response) {

res.writeHead(response.status, headers);

if (response.body instanceof Readable) {
response.body.pipe(res);
} else {
if (response.body) {
res.write(new Uint8Array(await response.arrayBuffer()));
}
if (response.body) {
const reader = response.body.getReader();

const next = async () => {
const { done, value } = await reader.read();

if (done) {
res.end();
return;
}

res.write(Buffer.from(value), (error) => {
if (error) {
console.error('Error writing stream', error);
res.end();
} else {
next();
}
});
};

next();
} else {
res.end();
}
}
11 changes: 8 additions & 3 deletions packages/kit/src/node/polyfills.js
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
import fetch, { Response, Request, Headers } from 'node-fetch';
import { fetch, Response, Request, Headers } from 'undici';
import { ReadableStream, TransformStream, WritableStream } from 'stream/web';
import { webcrypto as crypto } from 'crypto';

/** @type {Record<string, any>} */
Expand All @@ -7,13 +8,17 @@ const globals = {
fetch,
Response,
Request,
Headers
Headers,
ReadableStream,
TransformStream,
WritableStream
};

// exported for dev/preview and node environments
export function installPolyfills() {
for (const name in globals) {
// TODO use built-in fetch once https://github.com/nodejs/undici/issues/1262 is resolved
if (name in globalThis) continue;

Object.defineProperty(globalThis, name, {
enumerable: true,
configurable: true,
Expand Down
13 changes: 9 additions & 4 deletions packages/kit/src/runtime/server/endpoint.js
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,8 @@ const text_types = new Set([
'multipart/form-data'
]);

const bodyless_status_codes = new Set([101, 204, 205, 304]);

/**
* Decides how the body should be parsed based on its mime type
*
Expand Down Expand Up @@ -124,8 +126,11 @@ export async function render_endpoint(event, mod) {
}
}

return new Response(method !== 'head' ? normalized_body : undefined, {
status,
headers
});
return new Response(
method !== 'head' && !bodyless_status_codes.has(status) ? normalized_body : undefined,
{
status,
headers
}
);
}
5 changes: 5 additions & 0 deletions packages/kit/src/runtime/server/page/load_node.js
Original file line number Diff line number Diff line change
Expand Up @@ -250,6 +250,11 @@ export async function load_node({
if (cookie) opts.headers.set('cookie', cookie);
}

// we need to delete the connection header, as explained here:
// https://github.com/nodejs/undici/issues/1470#issuecomment-1140798467
// TODO this may be a case for being selective about which headers we let through
opts.headers.delete('connection');

const external_request = new Request(requested, /** @type {RequestInit} */ (opts));
response = await options.hooks.externalFetch.call(null, external_request);
}
Expand Down
10 changes: 7 additions & 3 deletions packages/kit/src/runtime/server/utils.js
Original file line number Diff line number Diff line change
Expand Up @@ -39,9 +39,13 @@ export function is_pojo(body) {
if (body) {
if (body instanceof Uint8Array) return false;

// body could be a node Readable, but we don't want to import
// node built-ins, so we use duck typing
if (body._readableState && typeof body.pipe === 'function') return false;
// if body is a node Readable, throw an error
// TODO remove this for 1.0
if (body._readableState && typeof body.pipe === 'function') {
throw new Error('Node streams are no longer supported — use a ReadableStream instead');
}

if (body instanceof ReadableStream) return false;

// similarly, it could be a web ReadableStream
if (typeof ReadableStream !== 'undefined' && body instanceof ReadableStream) return false;
Expand Down
4 changes: 3 additions & 1 deletion packages/kit/src/utils/http.spec.js
Original file line number Diff line number Diff line change
@@ -1,7 +1,9 @@
import { test } from 'uvu';
import * as assert from 'uvu/assert';
import { to_headers } from './http.js';
import { Headers } from 'node-fetch';
import { Headers } from 'undici';

// @ts-ignore
globalThis.Headers = Headers;

test('handle header string value', () => {
Expand Down
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
<script context="module">
/** @type {import('@sveltejs/kit').Load} */
export async function load({ url, fetch }) {
const res = await fetch(`http://localhost:${url.searchParams.get('port')}/large-response.json`);
export async function load({ fetch }) {
const res = await fetch('/load/large-response/text.txt');
const text = await res.text();

return {
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,23 @@
const chunk_size = 50000;
const chunk_count = 100;

let chunk = '';
for (let i = 0; i < chunk_size; i += 1) {
chunk += String(i % 10);
}

export function get() {
let i = 0;

return {
body: new ReadableStream({
pull: (controller) => {
if (i++ < chunk_count) {
controller.enqueue(chunk);
} else {
controller.close();
}
}
})
};
}
41 changes: 2 additions & 39 deletions packages/kit/test/apps/basics/test/test.js
Original file line number Diff line number Diff line change
Expand Up @@ -1360,45 +1360,8 @@ test.describe.parallel('Load', () => {
test('handles large responses', async ({ page }) => {
await page.goto('/load');

const chunk_size = 50000;
const chunk_count = 100;
const total_size = chunk_size * chunk_count;

let chunk = '';
for (let i = 0; i < chunk_size; i += 1) {
chunk += String(i % 10);
}

let times_responded = 0;

const { port, server } = await start_server(async (req, res) => {
if (req.url === '/large-response.json') {
times_responded += 1;

res.writeHead(200, {
'Access-Control-Allow-Origin': '*'
});

for (let i = 0; i < chunk_count; i += 1) {
if (!res.write(chunk)) {
await new Promise((fulfil) => {
res.once('drain', () => {
fulfil(undefined);
});
});
}
}

res.end();
}
});

await page.goto(`/load/large-response?port=${port}`);
expect(await page.textContent('h1')).toBe(`text.length is ${total_size}`);

expect(times_responded).toBe(1);

server.close();
await page.goto('/load/large-response');
expect(await page.textContent('h1')).toBe('text.length is 5000000');
});

test('handles external api', async ({ page }) => {
Expand Down
9 changes: 7 additions & 2 deletions pnpm-lock.yaml

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.