first commit
Some checks failed
Test examples / Test Examples (20) (push) Has been cancelled
Test examples / Test Examples (22) (push) Has been cancelled
Lock Threads / action (push) Has been cancelled
Trigger Release / start (push) Has been cancelled
Stale issue handler / stale (push) Has been cancelled
Update Font Data / create-pull-request (push) Has been cancelled
build-and-deploy / deploy-target (push) Has been cancelled
build-and-deploy / build (push) Has been cancelled
build-and-deploy / stable - aarch64-unknown-linux-musl - node@16 (push) Has been cancelled
build-and-deploy / stable - x86_64-unknown-linux-musl - node@16 (push) Has been cancelled
build-and-deploy / stable - aarch64-unknown-linux-gnu - node@16 (push) Has been cancelled
build-and-deploy / stable - x86_64-unknown-linux-gnu - node@16 (push) Has been cancelled
build-and-deploy / stable - aarch64-pc-windows-msvc - node@16 (push) Has been cancelled
build-and-deploy / stable - x86_64-pc-windows-msvc - node@16 (push) Has been cancelled
build-and-deploy / stable - aarch64-apple-darwin - node@16 (push) Has been cancelled
build-and-deploy / stable - x86_64-apple-darwin - node@16 (push) Has been cancelled
build-and-deploy / build-wasm (nodejs) (push) Has been cancelled
build-and-deploy / build-wasm (web) (push) Has been cancelled
build-and-deploy / Deploy preview tarball (push) Has been cancelled
build-and-deploy / Potentially publish release (push) Has been cancelled
build-and-deploy / publish-turbopack-npm-packages (push) Has been cancelled
build-and-deploy / Deploy examples (push) Has been cancelled
build-and-deploy / thank you, build (push) Has been cancelled
build-and-deploy / Upload Turbopack Bytesize metrics to Datadog (push) Has been cancelled
Rspack Next.js development integration tests / Rspack integration tests (push) Has been cancelled
Rspack Next.js production integration tests / Rspack integration tests (push) Has been cancelled
Turbopack Next.js development integration tests / Next.js integration tests (push) Has been cancelled
Turbopack Next.js production integration tests / Next.js integration tests (push) Has been cancelled
Update Rspack test manifest / Update and upload Rspack development test manifest (push) Has been cancelled
Update Rspack test manifest / Update and upload Rspack production test manifest (push) Has been cancelled
Upload bundler test manifests to areweturboyet.com / Upload test results (push) Has been cancelled
Update React / create-pull-request (push) Has been cancelled
test-e2e-project-reset-cron / reset-test-project (push) Has been cancelled
Notify about the top 15 issues/PRs/feature requests (most reacted) in the last 90 days / run (push) Has been cancelled

This commit is contained in:
Arian Tron
2026-03-10 19:37:31 +03:30
commit 61f56f997c
27684 changed files with 2784175 additions and 0 deletions

View File

@@ -0,0 +1,9 @@
import { NextResponse } from 'next/server'
export function middleware(request) {
if (request.nextUrl.pathname.startsWith('/_next/')) return
const res = NextResponse.rewrite(new URL('/', request.url))
res.headers.set('X-From-Middleware', 'true')
return res
}

View File

@@ -0,0 +1,16 @@
export default (props) => (
<div id="from-middleware">{String(props.fromMiddleware)}</div>
)
export async function getServerSideProps({ req, res }) {
return {
props: {
fromMiddleware:
// TODO: this should only use request header once
// start is using the separate renders as well
req.headers['x-from-middleware'] ||
res.getHeader('x-from-middleware') ||
null,
},
}
}

View File

@@ -0,0 +1,131 @@
import {
check,
fetchViaHTTP,
File,
findPort,
killApp,
launchApp,
retry,
} from 'next-test-utils'
import { join } from 'path'
import webdriver from 'next-webdriver'
let app
let appPort
const context = {
appDir: join(__dirname, '../'),
logs: { output: '', stdout: '', stderr: '' },
middleware: new File(join(__dirname, '../middleware.js')),
}
describe('Middleware development errors', () => {
beforeEach(async () => {
context.logs = { output: '', stdout: '', stderr: '' }
appPort = await findPort()
app = await launchApp(context.appDir, appPort, {
onStdout(msg) {
context.logs.output += msg
context.logs.stdout += msg
},
onStderr(msg) {
context.logs.output += msg
context.logs.stderr += msg
},
})
})
afterEach(async () => {
context.middleware.restore()
if (app) {
await killApp(app)
}
})
async function assertMiddlewareFetch(hasMiddleware, path = '/') {
await check(async () => {
const res = await fetchViaHTTP(appPort, path)
expect(res.status).toBe(200)
expect(res.headers.get('x-from-middleware')).toBe(
hasMiddleware ? 'true' : null
)
return 'success'
}, 'success')
}
async function assertMiddlewareRender(hasMiddleware, path = '/') {
const browser = await webdriver(appPort, path)
const fromMiddleware = await browser.elementById('from-middleware').text()
expect(fromMiddleware).toBe(hasMiddleware ? 'true' : 'null')
}
describe('when middleware is removed', () => {
beforeEach(async () => {
await assertMiddlewareFetch(true)
context.middleware.delete()
})
it('sends response correctly', async () => {
await assertMiddlewareFetch(false)
await assertMiddlewareRender(false)
// assert no extra message on stderr
expect(context.logs.stderr).not.toContain('error')
})
})
describe('when middleware is removed and re-added', () => {
beforeEach(async () => {
await assertMiddlewareFetch(true)
context.middleware.delete()
await assertMiddlewareFetch(false)
context.middleware.restore()
})
it('sends response correctly', async () => {
await assertMiddlewareFetch(true)
await assertMiddlewareRender(true)
})
})
describe('when middleware is added', () => {
beforeEach(async () => {
context.middleware.delete()
await assertMiddlewareFetch(false)
context.middleware.restore()
})
it('sends response correctly', async () => {
await retry(() => assertMiddlewareFetch(true))
await assertMiddlewareRender(true)
})
})
describe('when matcher is added', () => {
beforeEach(async () => {
context.middleware.write(
context.middleware.originalContent +
`
export const config = {
matcher: '/',
}
`
)
await assertMiddlewareFetch(true)
context.middleware.write(
context.middleware.originalContent +
`
export const config = {
matcher: '/asdf',
}
`
)
})
it('sends response correctly', async () => {
await retry(() => assertMiddlewareFetch(true, '/asdf'))
await retry(() => assertMiddlewareRender(true, '/asdf'))
})
})
})