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
281 lines
7.3 KiB
JavaScript
281 lines
7.3 KiB
JavaScript
/* global globalThis */
|
|
import { NextRequest, NextResponse } from 'next/server'
|
|
import { getSomeData } from './lib/some-data'
|
|
import magicValue from 'shared-package'
|
|
|
|
export const config = {
|
|
regions: 'auto',
|
|
runtime: 'nodejs',
|
|
}
|
|
|
|
// When there is a top level await, requiring middleware.js will return a Promise
|
|
await new Promise((resolve) => {
|
|
setTimeout(resolve, 100)
|
|
})
|
|
|
|
const PATTERNS = []
|
|
|
|
const params = (url) => {
|
|
const input = url.split('?')[0]
|
|
let result = {}
|
|
|
|
for (const [pattern, handler] of PATTERNS) {
|
|
const patternResult = pattern.exec(input)
|
|
if (patternResult !== null && 'pathname' in patternResult) {
|
|
result = handler(patternResult)
|
|
break
|
|
}
|
|
}
|
|
return result
|
|
}
|
|
|
|
export async function middleware(request) {
|
|
getSomeData()
|
|
|
|
const url = request.nextUrl
|
|
|
|
if (process.env.NEXT_RUNTIME === 'nodejs') {
|
|
if (url.pathname.startsWith('/test-node-fs')) {
|
|
const fs = await import('fs')
|
|
const path = await import('path')
|
|
const pkgPath = path.join(process.cwd(), 'package.json')
|
|
const pkgData = JSON.parse(await fs.promises.readFile(pkgPath, 'utf8'))
|
|
return NextResponse.json(pkgData)
|
|
}
|
|
}
|
|
|
|
if (request.headers.get('x-prerender-revalidate')) {
|
|
return NextResponse.next({
|
|
headers: { 'x-middleware': 'hi' },
|
|
})
|
|
}
|
|
|
|
// this is needed for tests to get the BUILD_ID
|
|
if (url.pathname.startsWith('/_next/static/__BUILD_ID')) {
|
|
return NextResponse.next()
|
|
}
|
|
|
|
if (
|
|
url.pathname.includes('/_next/static/chunks/pages/_app-non-existent.js')
|
|
) {
|
|
return NextResponse.rewrite('https://example.vercel.sh')
|
|
}
|
|
|
|
if (url.pathname === '/api/edge-search-params') {
|
|
const newUrl = url.clone()
|
|
newUrl.searchParams.set('foo', 'bar')
|
|
return NextResponse.rewrite(newUrl)
|
|
}
|
|
|
|
if (url.pathname === '/') {
|
|
url.pathname = '/ssg/first'
|
|
return NextResponse.rewrite(url)
|
|
}
|
|
|
|
if (url.pathname === '/to-ssg') {
|
|
url.pathname = '/ssg/hello'
|
|
url.searchParams.set('from', 'middleware')
|
|
return NextResponse.rewrite(url)
|
|
}
|
|
|
|
if (url.pathname === '/sha') {
|
|
url.pathname = '/shallow'
|
|
return NextResponse.rewrite(url)
|
|
}
|
|
|
|
if (url.pathname.startsWith('/fetch-user-agent-default')) {
|
|
try {
|
|
const apiRoute = new URL(url)
|
|
apiRoute.pathname = '/api/headers'
|
|
const res = await fetch(withLocalIp(apiRoute))
|
|
return serializeData(await res.text())
|
|
} catch (err) {
|
|
return serializeError(err)
|
|
}
|
|
}
|
|
|
|
if (url.pathname === '/rewrite-to-dynamic') {
|
|
url.pathname = '/blog/from-middleware'
|
|
url.searchParams.set('some', 'middleware')
|
|
return NextResponse.rewrite(url)
|
|
}
|
|
|
|
if (url.pathname === '/rewrite-to-config-rewrite') {
|
|
url.pathname = '/rewrite-3'
|
|
url.searchParams.set('some', 'middleware')
|
|
return NextResponse.rewrite(url)
|
|
}
|
|
|
|
if (url.pathname.startsWith('/fetch-user-agent-crypto')) {
|
|
try {
|
|
const apiRoute = new URL(url)
|
|
apiRoute.pathname = '/api/headers'
|
|
const res = await fetch(withLocalIp(apiRoute), {
|
|
headers: {
|
|
'user-agent': 'custom-agent',
|
|
},
|
|
})
|
|
return serializeData(await res.text())
|
|
} catch (err) {
|
|
return serializeError(err)
|
|
}
|
|
}
|
|
|
|
if (url.pathname === '/global') {
|
|
return serializeData(
|
|
JSON.stringify({
|
|
process: {
|
|
env: {
|
|
ANOTHER_MIDDLEWARE_TEST: process.env.ANOTHER_MIDDLEWARE_TEST,
|
|
STRING_ENV_VAR: process.env.STRING_ENV_VAR,
|
|
MIDDLEWARE_TEST: process.env.MIDDLEWARE_TEST,
|
|
NEXT_RUNTIME: process.env.NEXT_RUNTIME,
|
|
},
|
|
},
|
|
})
|
|
)
|
|
}
|
|
|
|
if (url.pathname.endsWith('/globalthis')) {
|
|
return serializeData(JSON.stringify(Object.keys(globalThis)))
|
|
}
|
|
|
|
if (url.pathname.endsWith('/webcrypto')) {
|
|
const response = {}
|
|
try {
|
|
const algorithm = {
|
|
name: 'RSA-PSS',
|
|
hash: 'SHA-256',
|
|
publicExponent: new Uint8Array([0x01, 0x00, 0x01]),
|
|
modulusLength: 2048,
|
|
}
|
|
const keyUsages = ['sign', 'verify']
|
|
await crypto.subtle.generateKey(algorithm, false, keyUsages)
|
|
} catch (err) {
|
|
response.error = true
|
|
} finally {
|
|
return serializeData(JSON.stringify(response))
|
|
}
|
|
}
|
|
|
|
if (url.pathname.endsWith('/fetch-url')) {
|
|
const response = {}
|
|
try {
|
|
await fetch(new URL('http://localhost'))
|
|
} catch (err) {
|
|
response.error = {
|
|
name: err.name,
|
|
message: err.message,
|
|
}
|
|
} finally {
|
|
return serializeData(JSON.stringify(response))
|
|
}
|
|
}
|
|
|
|
if (url.pathname === '/abort-controller') {
|
|
const controller = new AbortController()
|
|
const signal = controller.signal
|
|
|
|
controller.abort()
|
|
const response = {}
|
|
|
|
try {
|
|
await fetch('https://example.vercel.sh', { signal })
|
|
} catch (err) {
|
|
response.error = {
|
|
name: err.name,
|
|
message: err.message,
|
|
}
|
|
} finally {
|
|
return serializeData(JSON.stringify(response))
|
|
}
|
|
}
|
|
|
|
if (url.pathname.endsWith('/root-subrequest')) {
|
|
const res = await fetch(url)
|
|
res.headers.set('x-dynamic-path', 'true')
|
|
return res
|
|
}
|
|
|
|
if (url.pathname === '/about') {
|
|
if (magicValue !== 42) throw new Error('shared-package problem')
|
|
return NextResponse.rewrite(new URL('/about/a', request.url))
|
|
}
|
|
|
|
if (url.pathname === '/redirect-to-somewhere') {
|
|
url.pathname = '/somewhere'
|
|
return NextResponse.redirect(url, {
|
|
headers: {
|
|
'x-redirect-header': 'hi',
|
|
},
|
|
})
|
|
}
|
|
|
|
if (url.pathname.startsWith('/url')) {
|
|
try {
|
|
if (request.nextUrl.pathname === '/url/relative-url') {
|
|
new URL('/relative')
|
|
return Response.next()
|
|
}
|
|
|
|
if (request.nextUrl.pathname === '/url/relative-request') {
|
|
await fetch(new Request('/urls-b'))
|
|
return Response.next()
|
|
}
|
|
|
|
if (request.nextUrl.pathname === '/url/relative-redirect') {
|
|
return Response.redirect('/urls-b')
|
|
}
|
|
|
|
if (request.nextUrl.pathname === '/url/relative-next-redirect') {
|
|
return NextResponse.redirect('/urls-b')
|
|
}
|
|
|
|
if (request.nextUrl.pathname === '/url/relative-next-rewrite') {
|
|
return NextResponse.rewrite('/urls-b')
|
|
}
|
|
|
|
if (request.nextUrl.pathname === '/url/relative-next-request') {
|
|
await fetch(new NextRequest('/urls-b'))
|
|
return NextResponse.next()
|
|
}
|
|
} catch (error) {
|
|
return new NextResponse(null, { headers: { error: error.message } })
|
|
}
|
|
}
|
|
|
|
if (url.pathname === '/ssr-page') {
|
|
url.pathname = '/ssr-page-2'
|
|
return NextResponse.rewrite(url)
|
|
}
|
|
|
|
if (url.pathname === '/error-throw' && request.__isData) {
|
|
throw new Error('test error')
|
|
}
|
|
|
|
const original = new URL(request.url)
|
|
return NextResponse.next({
|
|
headers: {
|
|
'req-url-path': `${original.pathname}${original.search}`,
|
|
'req-url-basepath': request.nextUrl.basePath,
|
|
'req-url-pathname': request.nextUrl.pathname,
|
|
'req-url-query': request.nextUrl.searchParams.get('foo'),
|
|
'req-url-locale': request.nextUrl.locale,
|
|
'req-url-params':
|
|
url.pathname !== '/static' ? JSON.stringify(params(request.url)) : '{}',
|
|
},
|
|
})
|
|
}
|
|
|
|
function serializeData(data) {
|
|
return new NextResponse(null, { headers: { data } })
|
|
}
|
|
|
|
function serializeError(error) {
|
|
return new NextResponse(null, { headers: { error: error.message } })
|
|
}
|
|
|
|
function withLocalIp(url) {
|
|
return String(url).replace('localhost', '127.0.0.1')
|
|
}
|