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

fix: adjust middleware json data rewrite to work with recent next@canary #2734

Merged
Show file tree
Hide file tree
Changes from all 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
16 changes: 10 additions & 6 deletions edge-runtime/lib/response.ts
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,8 @@ import {
normalizeLocalePath,
normalizeTrailingSlash,
relativizeURL,
removeBasePath,
rewriteDataPath,
Copy link
Contributor Author

@pieh pieh Jan 3, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Fun fact - we had rewriteDataPath function already but we never used it

} from './util.ts'

export interface FetchEventResult {
Expand Down Expand Up @@ -180,14 +182,16 @@ export const buildResponse = async ({
}

if (isDataReq) {
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

As far as I can tell - this will only be truthy for pages router json requests

// Data requests (i.e. requests for /_next/data ) need special handling
const isDataReq = request.headers.has('x-nextjs-data')

App router RSC requests don't seem to have that header set so this should be safe to do

Copy link
Contributor

@mrstork mrstork Jan 3, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I might be wrong, but it looks like next uses getRequestMeta(params.req, 'isNextDataReq') for isDataRequest? There's a chance we could do the same.
https://github.com/vercel/next.js/blob/canary/packages/next/src/server/request-meta.ts#L183-L205

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This was my initial attempt - f5bb340 but to make it work, something in middleware handling (edge function) would need to signal to origin (lambda) that this meta data should be set - in that commit I just made the previous query param work again (if it was present I was setting that meta data), but this felt wrong given that Next.js explicitly removed support for that. I'm not exactly sure how it works on Vercel - I'm clearly missing something here, but I don't think attempting to set those meta tags directly is the right way to go - maybe figuring out how that meta data is meant to be set would be the best solution here - I'll try to look if I can figure that out

Copy link
Contributor Author

@pieh pieh Jan 3, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Ok, so the part of code that is setting the meta data on requests (~https://github.com/vercel/next.js/blob/canary/packages/next/src/server/lib/router-utils/resolve-routes.ts) ... but it only happens when it is _next/data request to begin with, so because we were using canonical path before we never hit this code path and we just relied on fact that we could "externally" force it with query param.

So current fix (at least in terms of the approach of using data url instead of trying to add the internal meta data on request ourselves) seems like generally correct approach to me, because we now hit the code path that is setting that internal meta data and added e2e test show it working for all Next.js versions that we test

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks for looking into it even more deeply!

// The rewrite target is a data request, but a middleware rewrite target is always for the page route,
// so we need to tell the server this is a data request. Setting the `x-nextjs-data` header is not enough. 🤷
rewriteUrl.searchParams.set('__nextDataReq', '1')
rewriteUrl.pathname = rewriteDataPath({
dataUrl: new URL(request.url).pathname,
newRoute: removeBasePath(rewriteUrl.pathname, nextConfig?.basePath),
basePath: nextConfig?.basePath,
})
} else {
// respect trailing slash rules to prevent 308s
rewriteUrl.pathname = normalizeTrailingSlash(rewriteUrl.pathname, nextConfig?.trailingSlash)
}

// respect trailing slash rules to prevent 308s
rewriteUrl.pathname = normalizeTrailingSlash(rewriteUrl.pathname, nextConfig?.trailingSlash)

const target = normalizeLocalizedTarget({ target: rewriteUrl.toString(), request, nextConfig })
if (target === request.url) {
logger.withFields({ rewrite_url: rewrite }).debug('Rewrite url is same as original url')
Expand Down
17 changes: 17 additions & 0 deletions tests/e2e/edge-middleware.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -52,3 +52,20 @@ test('it should render OpenGraph image meta tag correctly', async ({ page, middl
const size = await getImageSize(Buffer.from(imageBuffer), 'png')
expect([size.width, size.height]).toEqual([1200, 630])
})

test('json data rewrite works', async ({ middlewarePages }) => {
const response = await fetch(`${middlewarePages.url}/_next/data/build-id/sha.json`, {
headers: {
'x-nextjs-data': '1',
},
})

expect(response.ok).toBe(true)
const body = await response.text()

expect(body).toMatch(/^{"pageProps":/)

const data = JSON.parse(body)

expect(data.pageProps.message).toBeDefined()
})
Comment on lines +56 to +71
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Using already existing setup in one of our fixtures

if (url.pathname === '/sha/') {
url.pathname = '/shallow'
return NextResponse.rewrite(url)
}
for middleware rewrite

and

export const getServerSideProps = () => {
return {
props: {
message: `Random: ${++i}${Math.random()}`,
},
}
}
for asserting body

2 changes: 1 addition & 1 deletion tests/fixtures/middleware-pages/next-env.d.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,4 +2,4 @@
/// <reference types="next/image-types/global" />

// NOTE: This file should not be edited
// see https://nextjs.org/docs/basic-features/typescript for more information.
// see https://nextjs.org/docs/pages/api-reference/config/typescript for more information.
1 change: 1 addition & 0 deletions tests/fixtures/middleware-pages/next.config.js
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,7 @@ if (platform === 'win32') {
}
}

/** @type {import('next').NextConfig} */
module.exports = {
trailingSlash: true,
output: 'standalone',
Expand Down
4 changes: 3 additions & 1 deletion tests/fixtures/middleware-pages/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,8 @@
"react-dom": "18.2.0"
},
"devDependencies": {
"@types/react": "18.2.47"
"@types/node": "^20.10.6",
"@types/react": "18.2.47",
"typescript": "^5.3.3"
}
}
3 changes: 2 additions & 1 deletion tests/fixtures/middleware-pages/tsconfig.json
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,8 @@
"moduleResolution": "node",
"resolveJsonModule": true,
"isolatedModules": true,
"jsx": "preserve"
"jsx": "preserve",
"target": "ES2017"
},
"include": ["next-env.d.ts", "**/*.ts", "**/*.tsx"],
"exclude": ["node_modules"]
Expand Down
11 changes: 4 additions & 7 deletions tests/integration/edge-handler.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -387,8 +387,7 @@ describe('page router', () => {
})
const res = await response.json()
const url = new URL(res.url, 'http://n/')
expect(url.pathname).toBe('/ssr-page-2/')
expect(url.searchParams.get('__nextDataReq')).toBe('1')
expect(url.pathname).toBe('/_next/data/build-id/ssr-page-2.json')
expect(res.headers['x-nextjs-data']).toBe('1')
expect(response.headers.get('x-nextjs-rewrite')).toBe('/ssr-page-2/')
expect(response.status).toBe(200)
Expand Down Expand Up @@ -420,7 +419,7 @@ describe('page router', () => {
expect(response.status).toBe(200)
})

test<FixtureTestContext>('should rewrite un-rewritten data requests to page route', async (ctx) => {
test<FixtureTestContext>('should NOT rewrite un-rewritten data requests to page route', async (ctx) => {
await createFixture('middleware-pages', ctx)
await runPlugin(ctx)
const origin = await LocalServer.run(async (req, res) => {
Expand All @@ -443,8 +442,7 @@ describe('page router', () => {
})
const res = await response.json()
const url = new URL(res.url, 'http://n/')
expect(url.pathname).toBe('/ssg/hello/')
expect(url.searchParams.get('__nextDataReq')).toBe('1')
expect(url.pathname).toBe('/_next/data/build-id/ssg/hello.json')
expect(res.headers['x-nextjs-data']).toBe('1')
expect(response.status).toBe(200)
})
Expand Down Expand Up @@ -472,8 +470,7 @@ describe('page router', () => {
})
const res = await response.json()
const url = new URL(res.url, 'http://n/')
expect(url.pathname).toBe('/blog/first/')
expect(url.searchParams.get('__nextDataReq')).toBe('1')
expect(url.pathname).toBe('/_next/data/build-id/blog/first.json')
expect(url.searchParams.get('slug')).toBe('first')
expect(res.headers['x-nextjs-data']).toBe('1')
expect(response.status).toBe(200)
Expand Down
16 changes: 0 additions & 16 deletions tests/integration/page-router.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -95,22 +95,6 @@ test<FixtureTestContext>('Should revalidate path with On-demand Revalidation', a
expect(dateCacheInitial).not.toBe(dateCacheRevalidated)
})

test<FixtureTestContext>('Should return JSON for data req to page route', async (ctx) => {
await createFixture('page-router', ctx)
await runPlugin(ctx)

const response = await invokeFunction(ctx, {
url: '/static/revalidate-manual?__nextDataReq=1',
headers: { 'x-nextjs-data': '1' },
})

expect(response.body).toMatch(/^{"pageProps":/)

const data = JSON.parse(response.body)

expect(data.pageProps.show).toBeDefined()
})

test.skipIf(platform === 'win32')<FixtureTestContext>(
'Should set permanent "netlify-cdn-cache-control" header on fully static pages"',
async (ctx) => {
Expand Down
1 change: 1 addition & 0 deletions tests/utils/create-e2e-fixture.ts
Original file line number Diff line number Diff line change
Expand Up @@ -325,6 +325,7 @@ export const fixtureFactories = {
bun: () => createE2EFixture('simple', { packageManger: 'bun' }),
middleware: () => createE2EFixture('middleware'),
middlewareOg: () => createE2EFixture('middleware-og'),
middlewarePages: () => createE2EFixture('middleware-pages'),
pageRouter: () => createE2EFixture('page-router'),
pageRouterBasePathI18n: () => createE2EFixture('page-router-base-path-i18n'),
turborepo: () =>
Expand Down
Loading