-
Notifications
You must be signed in to change notification settings - Fork 1
Expand file tree
/
Copy pathmiddleware.ts
More file actions
102 lines (85 loc) · 2.67 KB
/
Copy pathmiddleware.ts
File metadata and controls
102 lines (85 loc) · 2.67 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
import type { NextRequest } from 'next/server';
import { NextResponse } from 'next/server';
import { appendMarkdownVaryHeader, shouldServeMarkdown } from '@/lib/markdown-negotiation';
const EXCLUDED_EXACT_PATHS = new Set([
'/.well-known/llms.txt',
'/favicon.ico',
'/llms-full.txt',
'/llms.txt',
'/overview/llms-full.txt',
'/robots.txt',
'/sitemap.xml',
]);
const EXCLUDED_PATH_PREFIXES = ['/_next', '/api', '/llms.mdx', '/og'];
const EXCLUDED_ASSET_EXTENSIONS = new Set([
'.avif',
'.css',
'.gif',
'.ico',
'.jpeg',
'.jpg',
'.js',
'.json',
'.map',
'.otf',
'.pdf',
'.png',
'.svg',
'.ttf',
'.txt',
'.webp',
'.woff',
'.woff2',
'.xml',
'.zip',
]);
function isProgrammaticClient(request: NextRequest): boolean {
// Browsers always send Sec-Fetch-Dest; curl/WebFetch/python-requests do not
return !request.headers.has('sec-fetch-dest');
}
function matchesPathPrefix(pathname: string, prefix: string): boolean {
return pathname === prefix || pathname.startsWith(`${prefix}/`);
}
function hasExcludedAssetExtension(pathname: string): boolean {
const extension = pathname.match(/\.[^./]+$/)?.[0]?.toLowerCase();
return extension ? EXCLUDED_ASSET_EXTENSIONS.has(extension) : false;
}
function isNegotiableDocsPath(pathname: string): boolean {
if (EXCLUDED_EXACT_PATHS.has(pathname)) return false;
if (EXCLUDED_PATH_PREFIXES.some((prefix) => matchesPathPrefix(pathname, prefix))) return false;
return !hasExcludedAssetExtension(pathname);
}
function isNegotiableMethod(method: string): boolean {
return method === 'GET' || method === 'HEAD';
}
function withMarkdownVary(response: NextResponse): NextResponse {
appendMarkdownVaryHeader(response.headers);
return response;
}
export default function middleware(request: NextRequest) {
const { pathname } = request.nextUrl;
if (!isNegotiableMethod(request.method)) {
return NextResponse.next();
}
const wantsMarkdown = shouldServeMarkdown(request.headers);
if (pathname === '/' && (wantsMarkdown || isProgrammaticClient(request))) {
const redirectUrl = request.nextUrl.clone();
redirectUrl.pathname = '/llms.txt';
return withMarkdownVary(NextResponse.redirect(redirectUrl));
}
if (!isNegotiableDocsPath(pathname)) {
return NextResponse.next();
}
if (wantsMarkdown) {
const rewriteUrl = request.nextUrl.clone();
rewriteUrl.pathname = `/llms.mdx${pathname}`;
return withMarkdownVary(NextResponse.rewrite(rewriteUrl));
}
return withMarkdownVary(NextResponse.next());
}
export const config = {
matcher: [
// Match all paths except Next.js internals, API routes, and static files
'/((?!_next|api/).*)', // This excludes /api/ but includes /apis/
],
};