From a02154affd387193f73fb208c86245fc742b1cc7 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Morel=20Se=CC=81bastien?= Date: Wed, 8 Mar 2023 17:24:12 -0800 Subject: [PATCH] Add HTTP/2 Server Push Link header --- contributors.yml | 1 + .../config/defaults/entry.server.node.tsx | 42 +++++++++++++----- templates/remix/app/entry.server.tsx | 44 ++++++++++++++----- 3 files changed, 67 insertions(+), 20 deletions(-) diff --git a/contributors.yml b/contributors.yml index 8b76fad011b..5ec94c4ffa2 100644 --- a/contributors.yml +++ b/contributors.yml @@ -386,6 +386,7 @@ - phishy - plastic041 - plondon +- plopix - pmbanugo - princerajroy - prvnbist diff --git a/packages/remix-dev/config/defaults/entry.server.node.tsx b/packages/remix-dev/config/defaults/entry.server.node.tsx index 98b32eb3867..230ba4dccfd 100644 --- a/packages/remix-dev/config/defaults/entry.server.node.tsx +++ b/packages/remix-dev/config/defaults/entry.server.node.tsx @@ -15,17 +15,17 @@ export default function handleRequest( ) { return isbot(request.headers.get("user-agent")) ? handleBotRequest( - request, - responseStatusCode, - responseHeaders, - remixContext - ) + request, + responseStatusCode, + responseHeaders, + remixContext + ) : handleBrowserRequest( - request, - responseStatusCode, - responseHeaders, - remixContext - ); + request, + responseStatusCode, + responseHeaders, + remixContext + ); } function handleBotRequest( @@ -46,6 +46,13 @@ function handleBotRequest( const body = new PassThrough(); responseHeaders.set("Content-Type", "text/html"); + responseHeaders.set( + "Link", + httpPushLinks(remixContext).map((link: string) => `<${link}>; rel=preload; as=script; crossorigin=anonymous`) + .concat(responseHeaders.get("Link") as string) + .filter(Boolean) + .join(",") + ); resolve( new Response(body, { @@ -88,6 +95,13 @@ function handleBrowserRequest( const body = new PassThrough(); responseHeaders.set("Content-Type", "text/html"); + responseHeaders.set( + "Link", + httpPushLinks(remixContext).map((link: string) => `<${link}>; rel=preload; as=script; crossorigin=anonymous`) + .concat(responseHeaders.get("Link") as string) + .filter(Boolean) + .join(",") + ); resolve( new Response(body, { @@ -111,3 +125,11 @@ function handleBrowserRequest( setTimeout(abort, ABORT_DELAY); }); } + +function httpPushLinks(remixContext: EntryContext) { + return [ + remixContext.manifest.url, + remixContext.manifest.entry.module, + ...remixContext.manifest.entry.imports, + ]; +} diff --git a/templates/remix/app/entry.server.tsx b/templates/remix/app/entry.server.tsx index c78d5047bc9..0f97ff62fa9 100644 --- a/templates/remix/app/entry.server.tsx +++ b/templates/remix/app/entry.server.tsx @@ -15,17 +15,17 @@ export default function handleRequest( ) { return isbot(request.headers.get("user-agent")) ? handleBotRequest( - request, - responseStatusCode, - responseHeaders, - remixContext - ) + request, + responseStatusCode, + responseHeaders, + remixContext + ) : handleBrowserRequest( - request, - responseStatusCode, - responseHeaders, - remixContext - ); + request, + responseStatusCode, + responseHeaders, + remixContext + ); } function handleBotRequest( @@ -46,6 +46,14 @@ function handleBotRequest( const body = new PassThrough(); responseHeaders.set("Content-Type", "text/html"); + responseHeaders.set( + "Link", + httpPushLinks(remixContext) + .map((link: string) => `<${link}>; rel=preload; as=script; crossorigin=anonymous`) + .concat(responseHeaders.get("Link") as string) + .filter(Boolean) + .join(",") + ); resolve( new Response(body, { @@ -88,6 +96,13 @@ function handleBrowserRequest( const body = new PassThrough(); responseHeaders.set("Content-Type", "text/html"); + responseHeaders.set( + "Link", + httpPushLinks(remixContext).map((link: string) => `<${link}>; rel=preload; as=script; crossorigin=anonymous`) + .concat(responseHeaders.get("Link") as string) + .filter(Boolean) + .join(",") + ); resolve( new Response(body, { @@ -111,3 +126,12 @@ function handleBrowserRequest( setTimeout(abort, ABORT_DELAY); }); } + + +function httpPushLinks(remixContext: EntryContext) { + return [ + remixContext.manifest.url, + remixContext.manifest.entry.module, + ...remixContext.manifest.entry.imports, + ]; +}