vinext 0.0.37 → 0.0.39

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (262) hide show
  1. package/README.md +33 -20
  2. package/dist/build/nitro-route-rules.d.ts +50 -0
  3. package/dist/build/nitro-route-rules.js +81 -0
  4. package/dist/build/nitro-route-rules.js.map +1 -0
  5. package/dist/build/precompress.d.ts +17 -0
  6. package/dist/build/precompress.js +102 -0
  7. package/dist/build/precompress.js.map +1 -0
  8. package/dist/build/prerender.d.ts +27 -22
  9. package/dist/build/prerender.js +17 -17
  10. package/dist/build/prerender.js.map +1 -1
  11. package/dist/build/report.d.ts +3 -4
  12. package/dist/build/report.js.map +1 -1
  13. package/dist/build/run-prerender.d.ts +3 -4
  14. package/dist/build/run-prerender.js.map +1 -1
  15. package/dist/build/standalone.d.ts +32 -0
  16. package/dist/build/standalone.js +199 -0
  17. package/dist/build/standalone.js.map +1 -0
  18. package/dist/build/static-export.d.ts +17 -29
  19. package/dist/build/static-export.js.map +1 -1
  20. package/dist/cache.d.ts +2 -0
  21. package/dist/cache.js +2 -0
  22. package/dist/check.d.ts +4 -4
  23. package/dist/check.js +1 -1
  24. package/dist/check.js.map +1 -1
  25. package/dist/cli.js +37 -26
  26. package/dist/cli.js.map +1 -1
  27. package/dist/client/empty-module.d.ts +1 -0
  28. package/dist/client/empty-module.js +1 -0
  29. package/dist/client/entry.js +2 -0
  30. package/dist/client/entry.js.map +1 -1
  31. package/dist/client/instrumentation-client-state.d.ts +10 -0
  32. package/dist/client/instrumentation-client-state.js +19 -0
  33. package/dist/client/instrumentation-client-state.js.map +1 -0
  34. package/dist/client/instrumentation-client.d.ts +8 -0
  35. package/dist/client/instrumentation-client.js +8 -0
  36. package/dist/client/instrumentation-client.js.map +1 -0
  37. package/dist/client/vinext-next-data.d.ts +5 -8
  38. package/dist/cloudflare/index.js +1 -1
  39. package/dist/cloudflare/kv-cache-handler.d.ts +5 -3
  40. package/dist/cloudflare/kv-cache-handler.js +1 -1
  41. package/dist/cloudflare/kv-cache-handler.js.map +1 -1
  42. package/dist/cloudflare/tpr.d.ts +35 -27
  43. package/dist/cloudflare/tpr.js +37 -15
  44. package/dist/cloudflare/tpr.js.map +1 -1
  45. package/dist/config/config-matchers.d.ts +2 -2
  46. package/dist/config/config-matchers.js +1 -1
  47. package/dist/config/config-matchers.js.map +1 -1
  48. package/dist/config/dotenv.d.ts +4 -4
  49. package/dist/config/dotenv.js.map +1 -1
  50. package/dist/config/next-config.d.ts +40 -61
  51. package/dist/config/next-config.js +5 -4
  52. package/dist/config/next-config.js.map +1 -1
  53. package/dist/deploy.d.ts +25 -41
  54. package/dist/deploy.js +10 -4
  55. package/dist/deploy.js.map +1 -1
  56. package/dist/entries/app-rsc-entry.d.ts +6 -10
  57. package/dist/entries/app-rsc-entry.js +31 -28
  58. package/dist/entries/app-rsc-entry.js.map +1 -1
  59. package/dist/entries/pages-client-entry.js +2 -0
  60. package/dist/entries/pages-client-entry.js.map +1 -1
  61. package/dist/entries/pages-server-entry.js +42 -263
  62. package/dist/entries/pages-server-entry.js.map +1 -1
  63. package/dist/entries/runtime-entry-module.d.ts +13 -1
  64. package/dist/entries/runtime-entry-module.js +18 -4
  65. package/dist/entries/runtime-entry-module.js.map +1 -1
  66. package/dist/index.d.ts +33 -41
  67. package/dist/index.js +263 -777
  68. package/dist/index.js.map +1 -1
  69. package/dist/init.d.ts +14 -26
  70. package/dist/init.js +8 -2
  71. package/dist/init.js.map +1 -1
  72. package/dist/plugins/client-reference-dedup.js.map +1 -1
  73. package/dist/plugins/fix-use-server-closure-collision.d.ts +29 -0
  74. package/dist/plugins/fix-use-server-closure-collision.js +204 -0
  75. package/dist/plugins/fix-use-server-closure-collision.js.map +1 -0
  76. package/dist/plugins/fonts.d.ts +56 -0
  77. package/dist/plugins/fonts.js +531 -0
  78. package/dist/plugins/fonts.js.map +1 -0
  79. package/dist/plugins/instrumentation-client.d.ts +7 -0
  80. package/dist/plugins/instrumentation-client.js +29 -0
  81. package/dist/plugins/instrumentation-client.js.map +1 -0
  82. package/dist/plugins/og-assets.d.ts +26 -0
  83. package/dist/plugins/og-assets.js +118 -0
  84. package/dist/plugins/og-assets.js.map +1 -0
  85. package/dist/plugins/optimize-imports.d.ts +2 -2
  86. package/dist/plugins/optimize-imports.js +4 -4
  87. package/dist/plugins/optimize-imports.js.map +1 -1
  88. package/dist/plugins/server-externals-manifest.d.ts +27 -0
  89. package/dist/plugins/server-externals-manifest.js +76 -0
  90. package/dist/plugins/server-externals-manifest.js.map +1 -0
  91. package/dist/routing/app-router.d.ts +29 -55
  92. package/dist/routing/app-router.js.map +1 -1
  93. package/dist/routing/file-matcher.d.ts +2 -2
  94. package/dist/routing/file-matcher.js.map +1 -1
  95. package/dist/routing/pages-router.d.ts +6 -11
  96. package/dist/routing/pages-router.js.map +1 -1
  97. package/dist/routing/route-trie.d.ts +2 -2
  98. package/dist/routing/route-trie.js.map +1 -1
  99. package/dist/server/api-handler.js +6 -23
  100. package/dist/server/api-handler.js.map +1 -1
  101. package/dist/server/app-browser-entry.js +274 -39
  102. package/dist/server/app-browser-entry.js.map +1 -1
  103. package/dist/server/app-browser-stream.d.ts +6 -6
  104. package/dist/server/app-browser-stream.js.map +1 -1
  105. package/dist/server/app-page-boundary-render.d.ts +8 -8
  106. package/dist/server/app-page-boundary-render.js +2 -2
  107. package/dist/server/app-page-boundary-render.js.map +1 -1
  108. package/dist/server/app-page-boundary.d.ts +13 -11
  109. package/dist/server/app-page-boundary.js +1 -1
  110. package/dist/server/app-page-boundary.js.map +1 -1
  111. package/dist/server/app-page-cache.d.ts +10 -10
  112. package/dist/server/app-page-cache.js.map +1 -1
  113. package/dist/server/app-page-execution.d.ts +10 -10
  114. package/dist/server/app-page-execution.js.map +1 -1
  115. package/dist/server/app-page-probe.d.ts +2 -2
  116. package/dist/server/app-page-probe.js.map +1 -1
  117. package/dist/server/app-page-render.d.ts +4 -4
  118. package/dist/server/app-page-render.js.map +1 -1
  119. package/dist/server/app-page-request.d.ts +12 -12
  120. package/dist/server/app-page-request.js.map +1 -1
  121. package/dist/server/app-page-response.d.ts +18 -18
  122. package/dist/server/app-page-response.js.map +1 -1
  123. package/dist/server/app-page-stream.d.ts +18 -18
  124. package/dist/server/app-page-stream.js.map +1 -1
  125. package/dist/server/app-route-handler-cache.d.ts +2 -2
  126. package/dist/server/app-route-handler-cache.js.map +1 -1
  127. package/dist/server/app-route-handler-execution.d.ts +6 -6
  128. package/dist/server/app-route-handler-execution.js.map +1 -1
  129. package/dist/server/app-route-handler-policy.d.ts +8 -8
  130. package/dist/server/app-route-handler-policy.js.map +1 -1
  131. package/dist/server/app-route-handler-response.d.ts +6 -6
  132. package/dist/server/app-route-handler-response.js.map +1 -1
  133. package/dist/server/app-route-handler-runtime.d.ts +4 -4
  134. package/dist/server/app-route-handler-runtime.js.map +1 -1
  135. package/dist/server/app-ssr-entry.d.ts +4 -4
  136. package/dist/server/app-ssr-entry.js.map +1 -1
  137. package/dist/server/app-ssr-stream.d.ts +2 -2
  138. package/dist/server/app-ssr-stream.js +1 -3
  139. package/dist/server/app-ssr-stream.js.map +1 -1
  140. package/dist/server/dev-module-runner.d.ts +2 -2
  141. package/dist/server/dev-module-runner.js.map +1 -1
  142. package/dist/server/dev-server.js +8 -8
  143. package/dist/server/dev-server.js.map +1 -1
  144. package/dist/server/image-optimization.d.ts +7 -12
  145. package/dist/server/image-optimization.js.map +1 -1
  146. package/dist/server/instrumentation.d.ts +13 -13
  147. package/dist/server/instrumentation.js +16 -7
  148. package/dist/server/instrumentation.js.map +1 -1
  149. package/dist/server/isr-cache.d.ts +2 -2
  150. package/dist/server/isr-cache.js.map +1 -1
  151. package/dist/server/metadata-routes.d.ts +14 -19
  152. package/dist/server/metadata-routes.js.map +1 -1
  153. package/dist/server/middleware.d.ts +10 -16
  154. package/dist/server/middleware.js +15 -8
  155. package/dist/server/middleware.js.map +1 -1
  156. package/dist/server/pages-api-route.d.ts +23 -0
  157. package/dist/server/pages-api-route.js +40 -0
  158. package/dist/server/pages-api-route.js.map +1 -0
  159. package/dist/server/pages-i18n.d.ts +4 -4
  160. package/dist/server/pages-i18n.js.map +1 -1
  161. package/dist/server/pages-media-type.d.ts +16 -0
  162. package/dist/server/pages-media-type.js +25 -0
  163. package/dist/server/pages-media-type.js.map +1 -0
  164. package/dist/server/pages-node-compat.d.ts +45 -0
  165. package/dist/server/pages-node-compat.js +148 -0
  166. package/dist/server/pages-node-compat.js.map +1 -0
  167. package/dist/server/pages-page-data.d.ts +22 -22
  168. package/dist/server/pages-page-data.js.map +1 -1
  169. package/dist/server/pages-page-response.d.ts +8 -8
  170. package/dist/server/pages-page-response.js.map +1 -1
  171. package/dist/server/prod-server.d.ts +20 -15
  172. package/dist/server/prod-server.js +171 -53
  173. package/dist/server/prod-server.js.map +1 -1
  174. package/dist/server/seed-cache.js.map +1 -1
  175. package/dist/server/static-file-cache.d.ts +57 -0
  176. package/dist/server/static-file-cache.js +219 -0
  177. package/dist/server/static-file-cache.js.map +1 -0
  178. package/dist/shims/app.d.ts +2 -2
  179. package/dist/shims/cache-for-request.d.ts +58 -0
  180. package/dist/shims/cache-for-request.js +74 -0
  181. package/dist/shims/cache-for-request.js.map +1 -0
  182. package/dist/shims/cache-runtime.d.ts +6 -9
  183. package/dist/shims/cache-runtime.js.map +1 -1
  184. package/dist/shims/cache.d.ts +28 -31
  185. package/dist/shims/cache.js.map +1 -1
  186. package/dist/shims/config.d.ts +2 -2
  187. package/dist/shims/config.js.map +1 -1
  188. package/dist/shims/dynamic.d.ts +2 -2
  189. package/dist/shims/dynamic.js +30 -17
  190. package/dist/shims/dynamic.js.map +1 -1
  191. package/dist/shims/error-boundary.d.ts +7 -7
  192. package/dist/shims/error-boundary.js.map +1 -1
  193. package/dist/shims/error.d.ts +2 -2
  194. package/dist/shims/error.js.map +1 -1
  195. package/dist/shims/fetch-cache.d.ts +4 -4
  196. package/dist/shims/fetch-cache.js.map +1 -1
  197. package/dist/shims/font-google-base.d.ts +4 -4
  198. package/dist/shims/font-google-base.js.map +1 -1
  199. package/dist/shims/font-local.d.ts +6 -6
  200. package/dist/shims/font-local.js.map +1 -1
  201. package/dist/shims/form.d.ts +4 -8
  202. package/dist/shims/form.js +4 -6
  203. package/dist/shims/form.js.map +1 -1
  204. package/dist/shims/head-state.d.ts +2 -2
  205. package/dist/shims/head-state.js.map +1 -1
  206. package/dist/shims/head.d.ts +2 -2
  207. package/dist/shims/head.js +18 -20
  208. package/dist/shims/head.js.map +1 -1
  209. package/dist/shims/headers.d.ts +4 -4
  210. package/dist/shims/headers.js +1 -1
  211. package/dist/shims/headers.js.map +1 -1
  212. package/dist/shims/i18n-context.d.ts +2 -2
  213. package/dist/shims/i18n-context.js.map +1 -1
  214. package/dist/shims/i18n-state.d.ts +2 -2
  215. package/dist/shims/i18n-state.js.map +1 -1
  216. package/dist/shims/image-config.d.ts +2 -2
  217. package/dist/shims/image-config.js.map +1 -1
  218. package/dist/shims/image.d.ts +5 -6
  219. package/dist/shims/image.js +24 -8
  220. package/dist/shims/image.js.map +1 -1
  221. package/dist/shims/internal/app-router-context.d.ts +6 -6
  222. package/dist/shims/internal/app-router-context.js.map +1 -1
  223. package/dist/shims/internal/utils.d.ts +2 -2
  224. package/dist/shims/internal/utils.js.map +1 -1
  225. package/dist/shims/layout-segment-context.d.ts +12 -5
  226. package/dist/shims/layout-segment-context.js +18 -7
  227. package/dist/shims/layout-segment-context.js.map +1 -1
  228. package/dist/shims/legacy-image.d.ts +5 -8
  229. package/dist/shims/legacy-image.js.map +1 -1
  230. package/dist/shims/link.d.ts +21 -31
  231. package/dist/shims/link.js +4 -56
  232. package/dist/shims/link.js.map +1 -1
  233. package/dist/shims/metadata.d.ts +23 -31
  234. package/dist/shims/metadata.js.map +1 -1
  235. package/dist/shims/navigation-state.d.ts +2 -2
  236. package/dist/shims/navigation-state.js.map +1 -1
  237. package/dist/shims/navigation.d.ts +102 -17
  238. package/dist/shims/navigation.js +361 -113
  239. package/dist/shims/navigation.js.map +1 -1
  240. package/dist/shims/request-context.d.ts +2 -2
  241. package/dist/shims/request-context.js.map +1 -1
  242. package/dist/shims/router-state.d.ts +4 -4
  243. package/dist/shims/router-state.js.map +1 -1
  244. package/dist/shims/router.d.ts +28 -47
  245. package/dist/shims/router.js.map +1 -1
  246. package/dist/shims/script.d.ts +16 -31
  247. package/dist/shims/script.js.map +1 -1
  248. package/dist/shims/server.d.ts +11 -10
  249. package/dist/shims/server.js +3 -0
  250. package/dist/shims/server.js.map +1 -1
  251. package/dist/shims/unified-request-context.d.ts +4 -4
  252. package/dist/shims/unified-request-context.js +1 -0
  253. package/dist/shims/unified-request-context.js.map +1 -1
  254. package/dist/shims/web-vitals.d.ts +2 -2
  255. package/dist/shims/web-vitals.js.map +1 -1
  256. package/dist/utils/lazy-chunks.d.ts +34 -0
  257. package/dist/utils/lazy-chunks.js +50 -0
  258. package/dist/utils/lazy-chunks.js.map +1 -0
  259. package/dist/utils/vinext-root.d.ts +24 -0
  260. package/dist/utils/vinext-root.js +31 -0
  261. package/dist/utils/vinext-root.js.map +1 -0
  262. package/package.json +8 -4
@@ -24,6 +24,7 @@ async function generateClientEntry(pagesDir, nextConfig, fileMatcher) {
24
24
  });
25
25
  const appFileBase = appFilePath?.replace(/\\/g, "/");
26
26
  return `
27
+ import "vinext/instrumentation-client";
27
28
  import React from "react";
28
29
  import { hydrateRoot } from "react-dom/client";
29
30
  // Eagerly import the router shim so its module-level popstate listener is
@@ -82,6 +83,7 @@ async function hydrate() {
82
83
 
83
84
  const root = hydrateRoot(container, element);
84
85
  window.__VINEXT_ROOT__ = root;
86
+ window.__VINEXT_HYDRATED_AT = performance.now();
85
87
  }
86
88
 
87
89
  hydrate();
@@ -1 +1 @@
1
- {"version":3,"file":"pages-client-entry.js","names":["pagesPatternToNextFormat"],"sources":["../../src/entries/pages-client-entry.ts"],"sourcesContent":["/**\n * Pages Router client hydration entry generator.\n *\n * Generates the virtual client entry module (`virtual:vinext-client-entry`).\n * This is the entry point for `vite build` (client bundle). It maps route\n * patterns to dynamic imports of page modules so Vite code-splits each page\n * into its own chunk. At runtime it reads __NEXT_DATA__ to determine which\n * page to hydrate.\n *\n * Extracted from index.ts.\n */\nimport {\n pagesRouter,\n patternToNextFormat as pagesPatternToNextFormat,\n type Route,\n} from \"../routing/pages-router.js\";\nimport { createValidFileMatcher } from \"../routing/file-matcher.js\";\nimport { type ResolvedNextConfig } from \"../config/next-config.js\";\nimport { findFileWithExts } from \"./pages-entry-helpers.js\";\n\nexport async function generateClientEntry(\n pagesDir: string,\n nextConfig: ResolvedNextConfig,\n fileMatcher: ReturnType<typeof createValidFileMatcher>,\n): Promise<string> {\n const pageRoutes = await pagesRouter(pagesDir, nextConfig?.pageExtensions, fileMatcher);\n\n const appFilePath = findFileWithExts(pagesDir, \"_app\", fileMatcher);\n const hasApp = appFilePath !== null;\n\n // Build a map of route pattern -> dynamic import.\n // Keys must use Next.js bracket format (e.g. \"/user/[id]\") to match\n // __NEXT_DATA__.page which is set via patternToNextFormat() during SSR.\n const loaderEntries = pageRoutes.map((r: Route) => {\n const absPath = r.filePath.replace(/\\\\/g, \"/\");\n const nextFormatPattern = pagesPatternToNextFormat(r.pattern);\n // JSON.stringify safely escapes quotes, backslashes, and special chars in\n // both the route pattern and the absolute file path.\n // lgtm[js/bad-code-sanitization]\n return ` ${JSON.stringify(nextFormatPattern)}: () => import(${JSON.stringify(absPath)})`;\n });\n\n const appFileBase = appFilePath?.replace(/\\\\/g, \"/\");\n\n return `\nimport React from \"react\";\nimport { hydrateRoot } from \"react-dom/client\";\n// Eagerly import the router shim so its module-level popstate listener is\n// registered. Without this, browser back/forward buttons do nothing because\n// navigateClient() is never invoked on history changes.\nimport \"next/router\";\n\nconst pageLoaders = {\n${loaderEntries.join(\",\\n\")}\n};\n\nasync function hydrate() {\n const nextData = window.__NEXT_DATA__;\n if (!nextData) {\n console.error(\"[vinext] No __NEXT_DATA__ found\");\n return;\n }\n\n const { pageProps } = nextData.props;\n const loader = pageLoaders[nextData.page];\n if (!loader) {\n console.error(\"[vinext] No page loader for route:\", nextData.page);\n return;\n }\n\n const pageModule = await loader();\n const PageComponent = pageModule.default;\n if (!PageComponent) {\n console.error(\"[vinext] Page module has no default export\");\n return;\n }\n\n let element;\n ${\n hasApp\n ? `\n try {\n const appModule = await import(${JSON.stringify(appFileBase!)});\n const AppComponent = appModule.default;\n window.__VINEXT_APP__ = AppComponent;\n element = React.createElement(AppComponent, { Component: PageComponent, pageProps });\n } catch {\n element = React.createElement(PageComponent, pageProps);\n }\n `\n : `\n element = React.createElement(PageComponent, pageProps);\n `\n }\n\n // Wrap with RouterContext.Provider so next/compat/router works during hydration\n const { wrapWithRouterContext } = await import(\"next/router\");\n element = wrapWithRouterContext(element);\n\n const container = document.getElementById(\"__next\");\n if (!container) {\n console.error(\"[vinext] No #__next element found\");\n return;\n }\n\n const root = hydrateRoot(container, element);\n window.__VINEXT_ROOT__ = root;\n}\n\nhydrate();\n`;\n}\n"],"mappings":";;;;;;;;;;;;;;;AAoBA,eAAsB,oBACpB,UACA,YACA,aACiB;CACjB,MAAM,aAAa,MAAM,YAAY,UAAU,YAAY,gBAAgB,YAAY;CAEvF,MAAM,cAAc,iBAAiB,UAAU,QAAQ,YAAY;CACnE,MAAM,SAAS,gBAAgB;CAK/B,MAAM,gBAAgB,WAAW,KAAK,MAAa;EACjD,MAAM,UAAU,EAAE,SAAS,QAAQ,OAAO,IAAI;EAC9C,MAAM,oBAAoBA,oBAAyB,EAAE,QAAQ;AAI7D,SAAO,KAAK,KAAK,UAAU,kBAAkB,CAAC,iBAAiB,KAAK,UAAU,QAAQ,CAAC;GACvF;CAEF,MAAM,cAAc,aAAa,QAAQ,OAAO,IAAI;AAEpD,QAAO;;;;;;;;;EASP,cAAc,KAAK,MAAM,CAAC;;;;;;;;;;;;;;;;;;;;;;;;;IA0BxB,SACI;;qCAE6B,KAAK,UAAU,YAAa,CAAC;;;;;;;MAQ1D;;IAGL"}
1
+ {"version":3,"file":"pages-client-entry.js","names":["pagesPatternToNextFormat"],"sources":["../../src/entries/pages-client-entry.ts"],"sourcesContent":["/**\n * Pages Router client hydration entry generator.\n *\n * Generates the virtual client entry module (`virtual:vinext-client-entry`).\n * This is the entry point for `vite build` (client bundle). It maps route\n * patterns to dynamic imports of page modules so Vite code-splits each page\n * into its own chunk. At runtime it reads __NEXT_DATA__ to determine which\n * page to hydrate.\n *\n * Extracted from index.ts.\n */\nimport {\n pagesRouter,\n patternToNextFormat as pagesPatternToNextFormat,\n type Route,\n} from \"../routing/pages-router.js\";\nimport { createValidFileMatcher } from \"../routing/file-matcher.js\";\nimport { type ResolvedNextConfig } from \"../config/next-config.js\";\nimport { findFileWithExts } from \"./pages-entry-helpers.js\";\n\nexport async function generateClientEntry(\n pagesDir: string,\n nextConfig: ResolvedNextConfig,\n fileMatcher: ReturnType<typeof createValidFileMatcher>,\n): Promise<string> {\n const pageRoutes = await pagesRouter(pagesDir, nextConfig?.pageExtensions, fileMatcher);\n\n const appFilePath = findFileWithExts(pagesDir, \"_app\", fileMatcher);\n const hasApp = appFilePath !== null;\n\n // Build a map of route pattern -> dynamic import.\n // Keys must use Next.js bracket format (e.g. \"/user/[id]\") to match\n // __NEXT_DATA__.page which is set via patternToNextFormat() during SSR.\n const loaderEntries = pageRoutes.map((r: Route) => {\n const absPath = r.filePath.replace(/\\\\/g, \"/\");\n const nextFormatPattern = pagesPatternToNextFormat(r.pattern);\n // JSON.stringify safely escapes quotes, backslashes, and special chars in\n // both the route pattern and the absolute file path.\n // lgtm[js/bad-code-sanitization]\n return ` ${JSON.stringify(nextFormatPattern)}: () => import(${JSON.stringify(absPath)})`;\n });\n\n const appFileBase = appFilePath?.replace(/\\\\/g, \"/\");\n\n return `\nimport \"vinext/instrumentation-client\";\nimport React from \"react\";\nimport { hydrateRoot } from \"react-dom/client\";\n// Eagerly import the router shim so its module-level popstate listener is\n// registered. Without this, browser back/forward buttons do nothing because\n// navigateClient() is never invoked on history changes.\nimport \"next/router\";\n\nconst pageLoaders = {\n${loaderEntries.join(\",\\n\")}\n};\n\nasync function hydrate() {\n const nextData = window.__NEXT_DATA__;\n if (!nextData) {\n console.error(\"[vinext] No __NEXT_DATA__ found\");\n return;\n }\n\n const { pageProps } = nextData.props;\n const loader = pageLoaders[nextData.page];\n if (!loader) {\n console.error(\"[vinext] No page loader for route:\", nextData.page);\n return;\n }\n\n const pageModule = await loader();\n const PageComponent = pageModule.default;\n if (!PageComponent) {\n console.error(\"[vinext] Page module has no default export\");\n return;\n }\n\n let element;\n ${\n hasApp\n ? `\n try {\n const appModule = await import(${JSON.stringify(appFileBase!)});\n const AppComponent = appModule.default;\n window.__VINEXT_APP__ = AppComponent;\n element = React.createElement(AppComponent, { Component: PageComponent, pageProps });\n } catch {\n element = React.createElement(PageComponent, pageProps);\n }\n `\n : `\n element = React.createElement(PageComponent, pageProps);\n `\n }\n\n // Wrap with RouterContext.Provider so next/compat/router works during hydration\n const { wrapWithRouterContext } = await import(\"next/router\");\n element = wrapWithRouterContext(element);\n\n const container = document.getElementById(\"__next\");\n if (!container) {\n console.error(\"[vinext] No #__next element found\");\n return;\n }\n\n const root = hydrateRoot(container, element);\n window.__VINEXT_ROOT__ = root;\n window.__VINEXT_HYDRATED_AT = performance.now();\n}\n\nhydrate();\n`;\n}\n"],"mappings":";;;;;;;;;;;;;;;AAoBA,eAAsB,oBACpB,UACA,YACA,aACiB;CACjB,MAAM,aAAa,MAAM,YAAY,UAAU,YAAY,gBAAgB,YAAY;CAEvF,MAAM,cAAc,iBAAiB,UAAU,QAAQ,YAAY;CACnE,MAAM,SAAS,gBAAgB;CAK/B,MAAM,gBAAgB,WAAW,KAAK,MAAa;EACjD,MAAM,UAAU,EAAE,SAAS,QAAQ,OAAO,IAAI;EAC9C,MAAM,oBAAoBA,oBAAyB,EAAE,QAAQ;AAI7D,SAAO,KAAK,KAAK,UAAU,kBAAkB,CAAC,iBAAiB,KAAK,UAAU,QAAQ,CAAC;GACvF;CAEF,MAAM,cAAc,aAAa,QAAQ,OAAO,IAAI;AAEpD,QAAO;;;;;;;;;;EAUP,cAAc,KAAK,MAAM,CAAC;;;;;;;;;;;;;;;;;;;;;;;;;IA0BxB,SACI;;qCAE6B,KAAK,UAAU,YAAa,CAAC;;;;;;;MAQ1D;;IAGL"}
@@ -1,9 +1,8 @@
1
1
  import { apiRouter, pagesRouter } from "../routing/pages-router.js";
2
+ import { resolveEntryPath } from "./runtime-entry-module.js";
2
3
  import { isProxyFile } from "../server/middleware.js";
3
4
  import { generateMiddlewareMatcherCode, generateNormalizePathCode, generateRouteMatchNormalizationCode, generateSafeRegExpCode } from "../server/middleware-codegen.js";
4
5
  import { findFileWithExts } from "./pages-entry-helpers.js";
5
- import path from "node:path";
6
- import { fileURLToPath } from "node:url";
7
6
  //#region src/entries/pages-server-entry.ts
8
7
  /**
9
8
  * Pages Router server entry generator.
@@ -14,12 +13,14 @@ import { fileURLToPath } from "node:url";
14
13
  *
15
14
  * Extracted from index.ts.
16
15
  */
17
- const __dirname = path.dirname(fileURLToPath(import.meta.url));
18
- const _requestContextShimPath = fileURLToPath(new URL("../shims/request-context.js", import.meta.url)).replace(/\\/g, "/");
19
- const _routeTriePath = fileURLToPath(new URL("../routing/route-trie.js", import.meta.url)).replace(/\\/g, "/");
20
- const _pagesI18nPath = fileURLToPath(new URL("../server/pages-i18n.js", import.meta.url)).replace(/\\/g, "/");
21
- const _pagesPageResponsePath = fileURLToPath(new URL("../server/pages-page-response.js", import.meta.url)).replace(/\\/g, "/");
22
- const _pagesPageDataPath = fileURLToPath(new URL("../server/pages-page-data.js", import.meta.url)).replace(/\\/g, "/");
16
+ const _requestContextShimPath = resolveEntryPath("../shims/request-context.js", import.meta.url);
17
+ const _routeTriePath = resolveEntryPath("../routing/route-trie.js", import.meta.url);
18
+ const _pagesI18nPath = resolveEntryPath("../server/pages-i18n.js", import.meta.url);
19
+ const _pagesPageResponsePath = resolveEntryPath("../server/pages-page-response.js", import.meta.url);
20
+ const _pagesPageDataPath = resolveEntryPath("../server/pages-page-data.js", import.meta.url);
21
+ const _pagesNodeCompatPath = resolveEntryPath("../server/pages-node-compat.js", import.meta.url);
22
+ const _pagesApiRoutePath = resolveEntryPath("../server/pages-api-route.js", import.meta.url);
23
+ const _isrCachePath = resolveEntryPath("../server/isr-cache.js", import.meta.url);
23
24
  /**
24
25
  * Generate the virtual SSR server entry module.
25
26
  * This is the entry point for `vite build --ssr`.
@@ -39,9 +40,7 @@ async function generateServerEntry(pagesDir, nextConfig, fileMatcher, middleware
39
40
  const absPath = r.filePath.replace(/\\/g, "/");
40
41
  return ` { pattern: ${JSON.stringify(r.pattern)}, patternParts: ${JSON.stringify(r.patternParts)}, isDynamic: ${r.isDynamic}, params: ${JSON.stringify(r.params)}, module: page_${i}, filePath: ${JSON.stringify(absPath)} }`;
41
42
  });
42
- const apiRouteEntries = apiRoutes.map((r, i) => {
43
- return ` { pattern: ${JSON.stringify(r.pattern)}, patternParts: ${JSON.stringify(r.patternParts)}, isDynamic: ${r.isDynamic}, params: ${JSON.stringify(r.params)}, module: api_${i} }`;
44
- });
43
+ const apiRouteEntries = apiRoutes.map((r, i) => ` { pattern: ${JSON.stringify(r.pattern)}, patternParts: ${JSON.stringify(r.patternParts)}, isDynamic: ${r.isDynamic}, params: ${JSON.stringify(r.params)}, module: api_${i} }`);
45
44
  const appFilePath = findFileWithExts(pagesDir, "_app", fileMatcher);
46
45
  const docFilePath = findFileWithExts(pagesDir, "_document", fileMatcher);
47
46
  const appImportCode = appFilePath !== null ? `import { default as AppComponent } from ${JSON.stringify(appFilePath.replace(/\\/g, "/"))};` : `const AppComponent = null;`;
@@ -138,6 +137,8 @@ async function _runMiddleware(request) {
138
137
  try { response = await middlewareFn(nextRequest, fetchEvent); }
139
138
  catch (e) {
140
139
  console.error("[vinext] Middleware error:", e);
140
+ var _mwCtxErr = _getRequestExecutionContext();
141
+ if (_mwCtxErr && typeof _mwCtxErr.waitUntil === "function") { _mwCtxErr.waitUntil(fetchEvent.drainWaitUntil()); } else { fetchEvent.drainWaitUntil(); }
141
142
  return { continue: false, response: new Response("Internal Server Error", { status: 500 }) };
142
143
  }
143
144
  var _mwCtx = _getRequestExecutionContext();
@@ -194,7 +195,7 @@ import { renderToReadableStream } from "react-dom/server.edge";
194
195
  import { resetSSRHead, getSSRHeadHTML } from "next/head";
195
196
  import { flushPreloads } from "next/dynamic";
196
197
  import { setSSRContext, wrapWithRouterContext } from "next/router";
197
- import { getCacheHandler, _runWithCacheState } from "next/cache";
198
+ import { _runWithCacheState } from "next/cache";
198
199
  import { runWithPrivateCache } from "vinext/cache-runtime";
199
200
  import { ensureFetchPatch, runWithFetchCache } from "vinext/fetch-cache";
200
201
  import { runWithRequestContext as _runWithUnifiedCtx, createRequestContext as _createUnifiedCtx } from "vinext/unified-request-context";
@@ -204,14 +205,21 @@ import { runWithHeadState } from "vinext/head-state";
204
205
  import "vinext/i18n-state";
205
206
  import { setI18nContext } from "vinext/i18n-context";
206
207
  import { safeJsonStringify } from "vinext/html";
207
- import { decode as decodeQueryString } from "node:querystring";
208
208
  import { getSSRFontLinks as _getSSRFontLinks, getSSRFontStyles as _getSSRFontStylesGoogle, getSSRFontPreloads as _getSSRFontPreloadsGoogle } from "next/font/google";
209
209
  import { getSSRFontStyles as _getSSRFontStylesLocal, getSSRFontPreloads as _getSSRFontPreloadsLocal } from "next/font/local";
210
- import { parseCookies, sanitizeDestination as sanitizeDestinationLocal } from ${JSON.stringify(path.resolve(__dirname, "../config/config-matchers.js").replace(/\\/g, "/"))};
210
+ import { sanitizeDestination as sanitizeDestinationLocal } from ${JSON.stringify(resolveEntryPath("../config/config-matchers.js", import.meta.url))};
211
211
  import { runWithExecutionContext as _runWithExecutionContext, getRequestExecutionContext as _getRequestExecutionContext } from ${JSON.stringify(_requestContextShimPath)};
212
212
  import { buildRouteTrie as _buildRouteTrie, trieMatch as _trieMatch } from ${JSON.stringify(_routeTriePath)};
213
213
  import { reportRequestError as _reportRequestError } from "vinext/instrumentation";
214
214
  import { resolvePagesI18nRequest } from ${JSON.stringify(_pagesI18nPath)};
215
+ import { createPagesReqRes as __createPagesReqRes } from ${JSON.stringify(_pagesNodeCompatPath)};
216
+ import { handlePagesApiRoute as __handlePagesApiRoute } from ${JSON.stringify(_pagesApiRoutePath)};
217
+ import {
218
+ isrGet as __sharedIsrGet,
219
+ isrSet as __sharedIsrSet,
220
+ isrCacheKey as __sharedIsrCacheKey,
221
+ triggerBackgroundRegeneration as __sharedTriggerBackgroundRegeneration,
222
+ } from ${JSON.stringify(_isrCachePath)};
215
223
  import { resolvePagesPageData as __resolvePagesPageData } from ${JSON.stringify(_pagesPageDataPath)};
216
224
  import { renderPagesPageResponse as __renderPagesPageResponse } from ${JSON.stringify(_pagesPageResponsePath)};
217
225
  ${instrumentationImportCode}
@@ -228,69 +236,17 @@ const buildId = ${buildIdJson};
228
236
  // Full resolved config for production server (embedded at build time)
229
237
  export const vinextConfig = ${vinextConfigJson};
230
238
 
231
- class ApiBodyParseError extends Error {
232
- constructor(message, statusCode) {
233
- super(message);
234
- this.statusCode = statusCode;
235
- this.name = "ApiBodyParseError";
236
- }
237
- }
238
-
239
- // ISR cache helpers (inlined for the server entry)
240
- async function isrGet(key) {
241
- const handler = getCacheHandler();
242
- const result = await handler.get(key);
243
- if (!result || !result.value) return null;
244
- return { value: result, isStale: result.cacheState === "stale" };
239
+ function isrGet(key) {
240
+ return __sharedIsrGet(key);
245
241
  }
246
- async function isrSet(key, data, revalidateSeconds, tags) {
247
- const handler = getCacheHandler();
248
- await handler.set(key, data, { revalidate: revalidateSeconds, tags: tags || [] });
242
+ function isrSet(key, data, revalidateSeconds, tags) {
243
+ return __sharedIsrSet(key, data, revalidateSeconds, tags);
249
244
  }
250
- const pendingRegenerations = new Map();
251
245
  function triggerBackgroundRegeneration(key, renderFn) {
252
- if (pendingRegenerations.has(key)) return;
253
- const promise = renderFn()
254
- .catch((err) => console.error("[vinext] ISR regen failed for " + key + ":", err))
255
- .finally(() => pendingRegenerations.delete(key));
256
- pendingRegenerations.set(key, promise);
257
- // Register with the Workers ExecutionContext so the isolate is kept alive
258
- // until the regeneration finishes, even after the Response has been sent.
259
- const ctx = _getRequestExecutionContext();
260
- if (ctx && typeof ctx.waitUntil === "function") ctx.waitUntil(promise);
261
- }
262
-
263
- function fnv1a64(input) {
264
- let h1 = 0x811c9dc5;
265
- for (let i = 0; i < input.length; i++) {
266
- h1 ^= input.charCodeAt(i);
267
- h1 = (h1 * 0x01000193) >>> 0;
268
- }
269
- let h2 = 0x050c5d1f;
270
- for (let i = 0; i < input.length; i++) {
271
- h2 ^= input.charCodeAt(i);
272
- h2 = (h2 * 0x01000193) >>> 0;
273
- }
274
- return h1.toString(36) + h2.toString(36);
246
+ return __sharedTriggerBackgroundRegeneration(key, renderFn);
275
247
  }
276
- // Keep prefix construction and hashing logic in sync with isrCacheKey() in server/isr-cache.ts.
277
- // buildId is a top-level const in the generated entry (see "const buildId = ..." above).
278
248
  function isrCacheKey(router, pathname) {
279
- const normalized = pathname === "/" ? "/" : pathname.replace(/\\/$/, "");
280
- const prefix = buildId ? router + ":" + buildId : router;
281
- const key = prefix + ":" + normalized;
282
- if (key.length <= 200) return key;
283
- return prefix + ":__hash:" + fnv1a64(normalized);
284
- }
285
-
286
- function getMediaType(contentType) {
287
- var type = (contentType || "text/plain").split(";")[0];
288
- type = type && type.trim().toLowerCase();
289
- return type || "text/plain";
290
- }
291
-
292
- function isJsonMediaType(mediaType) {
293
- return mediaType === "application/json" || mediaType === "application/ld+json";
249
+ return __sharedIsrCacheKey(router, pathname, buildId || undefined);
294
250
  }
295
251
 
296
252
  async function renderToStringAsync(element) {
@@ -517,126 +473,6 @@ function parseCookieLocaleFromHeader(cookieHeader) {
517
473
  return null;
518
474
  }
519
475
 
520
- // Lightweight req/res facade for getServerSideProps and API routes.
521
- // Next.js pages expect ctx.req/ctx.res with Node-like shapes.
522
- function createReqRes(request, url, query, body) {
523
- const headersObj = {};
524
- for (const [k, v] of request.headers) headersObj[k.toLowerCase()] = v;
525
-
526
- const req = {
527
- method: request.method,
528
- url: url,
529
- headers: headersObj,
530
- query: query,
531
- body: body,
532
- cookies: parseCookies(request.headers.get("cookie")),
533
- };
534
-
535
- let resStatusCode = 200;
536
- const resHeaders = {};
537
- // set-cookie needs array support (multiple Set-Cookie headers are common)
538
- const setCookieHeaders = [];
539
- let resBody = null;
540
- let ended = false;
541
- let resolveResponse;
542
- const responsePromise = new Promise(function(r) { resolveResponse = r; });
543
-
544
- const res = {
545
- get statusCode() { return resStatusCode; },
546
- set statusCode(code) { resStatusCode = code; },
547
- writeHead: function(code, headers) {
548
- resStatusCode = code;
549
- if (headers) {
550
- for (const [k, v] of Object.entries(headers)) {
551
- if (k.toLowerCase() === "set-cookie") {
552
- if (Array.isArray(v)) { for (const c of v) setCookieHeaders.push(c); }
553
- else { setCookieHeaders.push(v); }
554
- } else {
555
- resHeaders[k] = v;
556
- }
557
- }
558
- }
559
- return res;
560
- },
561
- setHeader: function(name, value) {
562
- if (name.toLowerCase() === "set-cookie") {
563
- if (Array.isArray(value)) { for (const c of value) setCookieHeaders.push(c); }
564
- else { setCookieHeaders.push(value); }
565
- } else {
566
- resHeaders[name.toLowerCase()] = value;
567
- }
568
- return res;
569
- },
570
- getHeader: function(name) {
571
- if (name.toLowerCase() === "set-cookie") return setCookieHeaders.length > 0 ? setCookieHeaders : undefined;
572
- return resHeaders[name.toLowerCase()];
573
- },
574
- end: function(data) {
575
- if (ended) return;
576
- ended = true;
577
- if (data !== undefined && data !== null) resBody = data;
578
- const h = new Headers(resHeaders);
579
- for (const c of setCookieHeaders) h.append("set-cookie", c);
580
- resolveResponse(new Response(resBody, { status: resStatusCode, headers: h }));
581
- },
582
- status: function(code) { resStatusCode = code; return res; },
583
- json: function(data) {
584
- resHeaders["content-type"] = "application/json";
585
- res.end(JSON.stringify(data));
586
- },
587
- send: function(data) {
588
- if (Buffer.isBuffer(data)) {
589
- if (!resHeaders["content-type"]) resHeaders["content-type"] = "application/octet-stream";
590
- resHeaders["content-length"] = String(data.length);
591
- res.end(data);
592
- } else if (typeof data === "object" && data !== null) {
593
- res.json(data);
594
- } else {
595
- if (!resHeaders["content-type"]) resHeaders["content-type"] = "text/plain";
596
- res.end(String(data));
597
- }
598
- },
599
- redirect: function(statusOrUrl, url2) {
600
- if (typeof statusOrUrl === "string") { res.writeHead(307, { Location: statusOrUrl }); }
601
- else { res.writeHead(statusOrUrl, { Location: url2 }); }
602
- res.end();
603
- },
604
- getHeaders: function() {
605
- var h = Object.assign({}, resHeaders);
606
- if (setCookieHeaders.length > 0) h["set-cookie"] = setCookieHeaders;
607
- return h;
608
- },
609
- get headersSent() { return ended; },
610
- };
611
-
612
- return { req, res, responsePromise };
613
- }
614
-
615
- /**
616
- * Read request body as text with a size limit.
617
- * Throws if the body exceeds maxBytes. This prevents DoS via chunked
618
- * transfer encoding where Content-Length is absent or spoofed.
619
- */
620
- async function readBodyWithLimit(request, maxBytes) {
621
- if (!request.body) return "";
622
- var reader = request.body.getReader();
623
- var decoder = new TextDecoder();
624
- var chunks = [];
625
- var totalSize = 0;
626
- for (;;) {
627
- var result = await reader.read();
628
- if (result.done) break;
629
- totalSize += result.value.byteLength;
630
- if (totalSize > maxBytes) {
631
- reader.cancel();
632
- throw new Error("Request body too large");
633
- }
634
- chunks.push(decoder.decode(result.value, { stream: true }));
635
- }
636
- chunks.push(decoder.decode());
637
- return chunks.join("");
638
- }
639
-
640
476
  export async function renderPage(request, url, manifest, ctx) {
641
477
  if (ctx) return _runWithExecutionContext(ctx, () => _renderPage(request, url, manifest));
642
478
  return _renderPage(request, url, manifest);
@@ -743,7 +579,7 @@ async function _renderPage(request, url, manifest) {
743
579
  },
744
580
  buildId,
745
581
  createGsspReqRes() {
746
- return createReqRes(request, routeUrl, query, undefined);
582
+ return __createPagesReqRes({ body: undefined, query, request, url: routeUrl });
747
583
  },
748
584
  createPageElement(currentPageProps) {
749
585
  var currentElement = AppComponent
@@ -868,76 +704,19 @@ async function _renderPage(request, url, manifest) {
868
704
 
869
705
  export async function handleApiRoute(request, url) {
870
706
  const match = matchRoute(url, apiRoutes);
871
- if (!match) {
872
- return new Response("404 - API route not found", { status: 404 });
873
- }
874
-
875
- const { route, params } = match;
876
- const handler = route.module.default;
877
- if (typeof handler !== "function") {
878
- return new Response("API route does not export a default function", { status: 500 });
879
- }
880
-
881
- const query = { ...params };
882
- const qs = url.split("?")[1];
883
- if (qs) {
884
- for (const [k, v] of new URLSearchParams(qs)) {
885
- if (k in query) {
886
- // Multi-value: promote to array (Next.js returns string[] for duplicate keys)
887
- query[k] = Array.isArray(query[k]) ? query[k].concat(v) : [query[k], v];
888
- } else {
889
- query[k] = v;
890
- }
891
- }
892
- }
893
-
894
- // Parse request body (enforce 1MB limit to prevent memory exhaustion,
895
- // matching Next.js default bodyParser sizeLimit).
896
- // Check Content-Length first as a fast path, then enforce on the actual
897
- // stream to prevent bypasses via chunked transfer encoding.
898
- const contentLength = parseInt(request.headers.get("content-length") || "0", 10);
899
- if (contentLength > 1 * 1024 * 1024) {
900
- return new Response("Request body too large", { status: 413 });
901
- }
902
- try {
903
- let body;
904
- const mediaType = getMediaType(request.headers.get("content-type"));
905
- let rawBody;
906
- try { rawBody = await readBodyWithLimit(request, 1 * 1024 * 1024); }
907
- catch { return new Response("Request body too large", { status: 413 }); }
908
- if (!rawBody) {
909
- body = isJsonMediaType(mediaType)
910
- ? {}
911
- : mediaType === "application/x-www-form-urlencoded"
912
- ? decodeQueryString(rawBody)
913
- : undefined;
914
- } else if (isJsonMediaType(mediaType)) {
915
- try { body = JSON.parse(rawBody); }
916
- catch { throw new ApiBodyParseError("Invalid JSON", 400); }
917
- } else if (mediaType === "application/x-www-form-urlencoded") {
918
- body = decodeQueryString(rawBody);
919
- } else {
920
- body = rawBody;
921
- }
922
-
923
- const { req, res, responsePromise } = createReqRes(request, url, query, body);
924
- await handler(req, res);
925
- // If handler didn't call res.end(), end it now.
926
- // The end() method is idempotent — safe to call twice.
927
- res.end();
928
- return await responsePromise;
929
- } catch (e) {
930
- if (e instanceof ApiBodyParseError) {
931
- return new Response(e.message, { status: e.statusCode, statusText: e.message });
932
- }
933
- console.error("[vinext] API error:", e);
934
- _reportRequestError(
935
- e instanceof Error ? e : new Error(String(e)),
936
- { path: url, method: request.method, headers: Object.fromEntries(request.headers.entries()) },
937
- { routerKind: "Pages Router", routePath: route.pattern, routeType: "route" },
938
- );
939
- return new Response("Internal Server Error", { status: 500 });
940
- }
707
+ return __handlePagesApiRoute({
708
+ match,
709
+ request,
710
+ url,
711
+ reportRequestError(error, routePattern) {
712
+ console.error("[vinext] API error:", error);
713
+ void _reportRequestError(
714
+ error,
715
+ { path: url, method: request.method, headers: Object.fromEntries(request.headers.entries()) },
716
+ { routerKind: "Pages Router", routePath: routePattern, routeType: "route" },
717
+ );
718
+ },
719
+ });
941
720
  }
942
721
 
943
722
  ${middlewareExportCode}
@@ -1 +1 @@
1
- {"version":3,"file":"pages-server-entry.js","names":[],"sources":["../../src/entries/pages-server-entry.ts"],"sourcesContent":["/**\n * Pages Router server entry generator.\n *\n * Generates the virtual SSR server entry module (`virtual:vinext-server-entry`).\n * This is the entry point for `vite build --ssr`. It handles SSR, API routes,\n * middleware, ISR, and i18n for the Pages Router.\n *\n * Extracted from index.ts.\n */\nimport path from \"node:path\";\nimport { fileURLToPath } from \"node:url\";\nimport { pagesRouter, apiRouter, type Route } from \"../routing/pages-router.js\";\nimport { createValidFileMatcher } from \"../routing/file-matcher.js\";\nimport { type ResolvedNextConfig } from \"../config/next-config.js\";\nimport { isProxyFile } from \"../server/middleware.js\";\nimport {\n generateSafeRegExpCode,\n generateMiddlewareMatcherCode,\n generateNormalizePathCode,\n generateRouteMatchNormalizationCode,\n} from \"../server/middleware-codegen.js\";\nimport { findFileWithExts } from \"./pages-entry-helpers.js\";\n\nconst __dirname = path.dirname(fileURLToPath(import.meta.url));\nconst _requestContextShimPath = fileURLToPath(\n new URL(\"../shims/request-context.js\", import.meta.url),\n).replace(/\\\\/g, \"/\");\nconst _routeTriePath = fileURLToPath(new URL(\"../routing/route-trie.js\", import.meta.url)).replace(\n /\\\\/g,\n \"/\",\n);\nconst _pagesI18nPath = fileURLToPath(new URL(\"../server/pages-i18n.js\", import.meta.url)).replace(\n /\\\\/g,\n \"/\",\n);\nconst _pagesPageResponsePath = fileURLToPath(\n new URL(\"../server/pages-page-response.js\", import.meta.url),\n).replace(/\\\\/g, \"/\");\nconst _pagesPageDataPath = fileURLToPath(\n new URL(\"../server/pages-page-data.js\", import.meta.url),\n).replace(/\\\\/g, \"/\");\n\n/**\n * Generate the virtual SSR server entry module.\n * This is the entry point for `vite build --ssr`.\n */\nexport async function generateServerEntry(\n pagesDir: string,\n nextConfig: ResolvedNextConfig,\n fileMatcher: ReturnType<typeof createValidFileMatcher>,\n middlewarePath: string | null,\n instrumentationPath: string | null,\n): Promise<string> {\n const pageRoutes = await pagesRouter(pagesDir, nextConfig?.pageExtensions, fileMatcher);\n const apiRoutes = await apiRouter(pagesDir, nextConfig?.pageExtensions, fileMatcher);\n\n // Generate import statements using absolute paths since virtual\n // modules don't have a real file location for relative resolution.\n const pageImports = pageRoutes.map((r: Route, i: number) => {\n const absPath = r.filePath.replace(/\\\\/g, \"/\");\n return `import * as page_${i} from ${JSON.stringify(absPath)};`;\n });\n\n const apiImports = apiRoutes.map((r: Route, i: number) => {\n const absPath = r.filePath.replace(/\\\\/g, \"/\");\n return `import * as api_${i} from ${JSON.stringify(absPath)};`;\n });\n\n // Build the route table — include filePath for SSR manifest lookup\n const pageRouteEntries = pageRoutes.map((r: Route, i: number) => {\n const absPath = r.filePath.replace(/\\\\/g, \"/\");\n return ` { pattern: ${JSON.stringify(r.pattern)}, patternParts: ${JSON.stringify(r.patternParts)}, isDynamic: ${r.isDynamic}, params: ${JSON.stringify(r.params)}, module: page_${i}, filePath: ${JSON.stringify(absPath)} }`;\n });\n\n const apiRouteEntries = apiRoutes.map((r: Route, i: number) => {\n return ` { pattern: ${JSON.stringify(r.pattern)}, patternParts: ${JSON.stringify(r.patternParts)}, isDynamic: ${r.isDynamic}, params: ${JSON.stringify(r.params)}, module: api_${i} }`;\n });\n\n // Check for _app and _document\n const appFilePath = findFileWithExts(pagesDir, \"_app\", fileMatcher);\n const docFilePath = findFileWithExts(pagesDir, \"_document\", fileMatcher);\n const appImportCode =\n appFilePath !== null\n ? `import { default as AppComponent } from ${JSON.stringify(appFilePath.replace(/\\\\/g, \"/\"))};`\n : `const AppComponent = null;`;\n\n const docImportCode =\n docFilePath !== null\n ? `import { default as DocumentComponent } from ${JSON.stringify(docFilePath.replace(/\\\\/g, \"/\"))};`\n : `const DocumentComponent = null;`;\n\n // Serialize i18n config for embedding in the server entry\n const i18nConfigJson = nextConfig?.i18n\n ? JSON.stringify({\n locales: nextConfig.i18n.locales,\n defaultLocale: nextConfig.i18n.defaultLocale,\n localeDetection: nextConfig.i18n.localeDetection,\n domains: nextConfig.i18n.domains,\n })\n : \"null\";\n\n // Embed the resolved build ID at build time\n const buildIdJson = JSON.stringify(nextConfig?.buildId ?? null);\n\n // Serialize the full resolved config for the production server.\n // This embeds redirects, rewrites, headers, basePath, trailingSlash\n // so prod-server.ts can apply them without loading next.config.js at runtime.\n const vinextConfigJson = JSON.stringify({\n basePath: nextConfig?.basePath ?? \"\",\n trailingSlash: nextConfig?.trailingSlash ?? false,\n redirects: nextConfig?.redirects ?? [],\n rewrites: nextConfig?.rewrites ?? { beforeFiles: [], afterFiles: [], fallback: [] },\n headers: nextConfig?.headers ?? [],\n i18n: nextConfig?.i18n ?? null,\n images: {\n deviceSizes: nextConfig?.images?.deviceSizes,\n imageSizes: nextConfig?.images?.imageSizes,\n dangerouslyAllowSVG: nextConfig?.images?.dangerouslyAllowSVG,\n contentDispositionType: nextConfig?.images?.contentDispositionType,\n contentSecurityPolicy: nextConfig?.images?.contentSecurityPolicy,\n },\n });\n\n // Generate instrumentation code if instrumentation.ts exists.\n // For production (Cloudflare Workers), instrumentation.ts is bundled into the\n // Worker and register() is called as a top-level await at module evaluation time —\n // before any request is handled. This mirrors App Router behavior (generateRscEntry)\n // and matches Next.js semantics: register() runs once on startup in the process\n // that handles requests.\n //\n // The onRequestError handler is stored on globalThis so it is visible across\n // all code within the Worker (same global scope).\n const instrumentationImportCode = instrumentationPath\n ? `import * as _instrumentation from ${JSON.stringify(instrumentationPath.replace(/\\\\/g, \"/\"))};`\n : \"\";\n\n const instrumentationInitCode = instrumentationPath\n ? `// Run instrumentation register() once at module evaluation time — before any\n// requests are handled. Matches Next.js semantics: register() is called once\n// on startup in the process that handles requests.\nif (typeof _instrumentation.register === \"function\") {\n await _instrumentation.register();\n}\n// Store the onRequestError handler on globalThis so it is visible to all\n// code within the Worker (same global scope).\nif (typeof _instrumentation.onRequestError === \"function\") {\n globalThis.__VINEXT_onRequestErrorHandler__ = _instrumentation.onRequestError;\n}`\n : \"\";\n\n // Generate middleware code if middleware.ts exists\n const middlewareImportCode = middlewarePath\n ? `import * as middlewareModule from ${JSON.stringify(middlewarePath.replace(/\\\\/g, \"/\"))};\nimport { NextRequest, NextFetchEvent } from \"next/server\";`\n : \"\";\n\n // The matcher config is read from the middleware module at import time.\n // We inline the matching + execution logic so the prod server can call it.\n const middlewareExportCode = middlewarePath\n ? `\n// --- Middleware support (generated from middleware-codegen.ts) ---\n${generateNormalizePathCode(\"es5\")}\n${generateRouteMatchNormalizationCode(\"es5\")}\n${generateSafeRegExpCode(\"es5\")}\n${generateMiddlewareMatcherCode(\"es5\")}\n\nexport async function runMiddleware(request, ctx) {\n if (ctx) return _runWithExecutionContext(ctx, () => _runMiddleware(request));\n return _runMiddleware(request);\n}\n\nasync function _runMiddleware(request) {\n var isProxy = ${middlewarePath ? JSON.stringify(isProxyFile(middlewarePath)) : \"false\"};\n var middlewareFn = isProxy\n ? (middlewareModule.proxy ?? middlewareModule.default)\n : (middlewareModule.middleware ?? middlewareModule.default);\n if (typeof middlewareFn !== \"function\") {\n var fileType = isProxy ? \"Proxy\" : \"Middleware\";\n var expectedExport = isProxy ? \"proxy\" : \"middleware\";\n throw new Error(\"The \" + fileType + \" file must export a function named \\`\" + expectedExport + \"\\` or a \\`default\\` function.\");\n }\n\n var config = middlewareModule.config;\n var matcher = config && config.matcher;\n var url = new URL(request.url);\n\n // Normalize pathname before matching to prevent path-confusion bypasses\n // (percent-encoding like /%61dmin, double slashes like /dashboard//settings).\n var decodedPathname;\n try { decodedPathname = __normalizePathnameForRouteMatchStrict(url.pathname); } catch (e) {\n return { continue: false, response: new Response(\"Bad Request\", { status: 400 }) };\n }\n var normalizedPathname = __normalizePath(decodedPathname);\n\n if (!matchesMiddleware(normalizedPathname, matcher, request, i18nConfig)) return { continue: true };\n\n // Construct a new Request with the decoded + normalized pathname so middleware\n // always sees the same canonical path that the router uses.\n var mwRequest = request;\n if (normalizedPathname !== url.pathname) {\n var mwUrl = new URL(url);\n mwUrl.pathname = normalizedPathname;\n mwRequest = new Request(mwUrl, request);\n }\n var __mwNextConfig = (vinextConfig.basePath || i18nConfig) ? { basePath: vinextConfig.basePath, i18n: i18nConfig || undefined } : undefined;\n var nextRequest = mwRequest instanceof NextRequest ? mwRequest : new NextRequest(mwRequest, __mwNextConfig ? { nextConfig: __mwNextConfig } : undefined);\n var fetchEvent = new NextFetchEvent({ page: normalizedPathname });\n var response;\n try { response = await middlewareFn(nextRequest, fetchEvent); }\n catch (e) {\n console.error(\"[vinext] Middleware error:\", e);\n return { continue: false, response: new Response(\"Internal Server Error\", { status: 500 }) };\n }\n var _mwCtx = _getRequestExecutionContext();\n if (_mwCtx && typeof _mwCtx.waitUntil === \"function\") { _mwCtx.waitUntil(fetchEvent.drainWaitUntil()); } else { fetchEvent.drainWaitUntil(); }\n\n if (!response) return { continue: true };\n\n if (response.headers.get(\"x-middleware-next\") === \"1\") {\n var rHeaders = new Headers();\n for (var [key, value] of response.headers) {\n // Keep x-middleware-request-* headers so the production server can\n // apply middleware-request header overrides before stripping internals\n // from the final client response.\n if (\n !key.startsWith(\"x-middleware-\") ||\n key === \"x-middleware-override-headers\" ||\n key.startsWith(\"x-middleware-request-\")\n ) rHeaders.append(key, value);\n }\n return { continue: true, responseHeaders: rHeaders };\n }\n\n if (response.status >= 300 && response.status < 400) {\n var location = response.headers.get(\"Location\") || response.headers.get(\"location\");\n if (location) {\n var rdHeaders = new Headers();\n for (var [rk, rv] of response.headers) {\n if (!rk.startsWith(\"x-middleware-\") && rk.toLowerCase() !== \"location\") rdHeaders.append(rk, rv);\n }\n return { continue: false, redirectUrl: location, redirectStatus: response.status, responseHeaders: rdHeaders };\n }\n }\n\n var rewriteUrl = response.headers.get(\"x-middleware-rewrite\");\n if (rewriteUrl) {\n var rwHeaders = new Headers();\n for (var [k, v] of response.headers) {\n if (!k.startsWith(\"x-middleware-\") || k === \"x-middleware-override-headers\" || k.startsWith(\"x-middleware-request-\")) rwHeaders.append(k, v);\n }\n var rewritePath;\n try { var parsed = new URL(rewriteUrl, request.url); rewritePath = parsed.pathname + parsed.search; }\n catch { rewritePath = rewriteUrl; }\n return { continue: true, rewriteUrl: rewritePath, rewriteStatus: response.status !== 200 ? response.status : undefined, responseHeaders: rwHeaders };\n }\n\n return { continue: false, response: response };\n}\n`\n : `\nexport async function runMiddleware() { return { continue: true }; }\n`;\n\n // The server entry is a self-contained module that uses Web-standard APIs\n // (Request/Response, renderToReadableStream) so it runs on Cloudflare Workers.\n return `\nimport React from \"react\";\nimport { renderToReadableStream } from \"react-dom/server.edge\";\nimport { resetSSRHead, getSSRHeadHTML } from \"next/head\";\nimport { flushPreloads } from \"next/dynamic\";\nimport { setSSRContext, wrapWithRouterContext } from \"next/router\";\nimport { getCacheHandler, _runWithCacheState } from \"next/cache\";\nimport { runWithPrivateCache } from \"vinext/cache-runtime\";\nimport { ensureFetchPatch, runWithFetchCache } from \"vinext/fetch-cache\";\nimport { runWithRequestContext as _runWithUnifiedCtx, createRequestContext as _createUnifiedCtx } from \"vinext/unified-request-context\";\nimport \"vinext/router-state\";\nimport { runWithServerInsertedHTMLState } from \"vinext/navigation-state\";\nimport { runWithHeadState } from \"vinext/head-state\";\nimport \"vinext/i18n-state\";\nimport { setI18nContext } from \"vinext/i18n-context\";\nimport { safeJsonStringify } from \"vinext/html\";\nimport { decode as decodeQueryString } from \"node:querystring\";\nimport { getSSRFontLinks as _getSSRFontLinks, getSSRFontStyles as _getSSRFontStylesGoogle, getSSRFontPreloads as _getSSRFontPreloadsGoogle } from \"next/font/google\";\nimport { getSSRFontStyles as _getSSRFontStylesLocal, getSSRFontPreloads as _getSSRFontPreloadsLocal } from \"next/font/local\";\nimport { parseCookies, sanitizeDestination as sanitizeDestinationLocal } from ${JSON.stringify(path.resolve(__dirname, \"../config/config-matchers.js\").replace(/\\\\/g, \"/\"))};\nimport { runWithExecutionContext as _runWithExecutionContext, getRequestExecutionContext as _getRequestExecutionContext } from ${JSON.stringify(_requestContextShimPath)};\nimport { buildRouteTrie as _buildRouteTrie, trieMatch as _trieMatch } from ${JSON.stringify(_routeTriePath)};\nimport { reportRequestError as _reportRequestError } from \"vinext/instrumentation\";\nimport { resolvePagesI18nRequest } from ${JSON.stringify(_pagesI18nPath)};\nimport { resolvePagesPageData as __resolvePagesPageData } from ${JSON.stringify(_pagesPageDataPath)};\nimport { renderPagesPageResponse as __renderPagesPageResponse } from ${JSON.stringify(_pagesPageResponsePath)};\n${instrumentationImportCode}\n${middlewareImportCode}\n\n${instrumentationInitCode}\n\n// i18n config (embedded at build time)\nconst i18nConfig = ${i18nConfigJson};\n\n// Build ID (embedded at build time)\nconst buildId = ${buildIdJson};\n\n// Full resolved config for production server (embedded at build time)\nexport const vinextConfig = ${vinextConfigJson};\n\nclass ApiBodyParseError extends Error {\n constructor(message, statusCode) {\n super(message);\n this.statusCode = statusCode;\n this.name = \"ApiBodyParseError\";\n }\n}\n\n// ISR cache helpers (inlined for the server entry)\nasync function isrGet(key) {\n const handler = getCacheHandler();\n const result = await handler.get(key);\n if (!result || !result.value) return null;\n return { value: result, isStale: result.cacheState === \"stale\" };\n}\nasync function isrSet(key, data, revalidateSeconds, tags) {\n const handler = getCacheHandler();\n await handler.set(key, data, { revalidate: revalidateSeconds, tags: tags || [] });\n}\nconst pendingRegenerations = new Map();\nfunction triggerBackgroundRegeneration(key, renderFn) {\n if (pendingRegenerations.has(key)) return;\n const promise = renderFn()\n .catch((err) => console.error(\"[vinext] ISR regen failed for \" + key + \":\", err))\n .finally(() => pendingRegenerations.delete(key));\n pendingRegenerations.set(key, promise);\n // Register with the Workers ExecutionContext so the isolate is kept alive\n // until the regeneration finishes, even after the Response has been sent.\n const ctx = _getRequestExecutionContext();\n if (ctx && typeof ctx.waitUntil === \"function\") ctx.waitUntil(promise);\n}\n\nfunction fnv1a64(input) {\n let h1 = 0x811c9dc5;\n for (let i = 0; i < input.length; i++) {\n h1 ^= input.charCodeAt(i);\n h1 = (h1 * 0x01000193) >>> 0;\n }\n let h2 = 0x050c5d1f;\n for (let i = 0; i < input.length; i++) {\n h2 ^= input.charCodeAt(i);\n h2 = (h2 * 0x01000193) >>> 0;\n }\n return h1.toString(36) + h2.toString(36);\n}\n// Keep prefix construction and hashing logic in sync with isrCacheKey() in server/isr-cache.ts.\n// buildId is a top-level const in the generated entry (see \"const buildId = ...\" above).\nfunction isrCacheKey(router, pathname) {\n const normalized = pathname === \"/\" ? \"/\" : pathname.replace(/\\\\/$/, \"\");\n const prefix = buildId ? router + \":\" + buildId : router;\n const key = prefix + \":\" + normalized;\n if (key.length <= 200) return key;\n return prefix + \":__hash:\" + fnv1a64(normalized);\n}\n\nfunction getMediaType(contentType) {\n var type = (contentType || \"text/plain\").split(\";\")[0];\n type = type && type.trim().toLowerCase();\n return type || \"text/plain\";\n}\n\nfunction isJsonMediaType(mediaType) {\n return mediaType === \"application/json\" || mediaType === \"application/ld+json\";\n}\n\nasync function renderToStringAsync(element) {\n const stream = await renderToReadableStream(element);\n await stream.allReady;\n return new Response(stream).text();\n}\n\nasync function renderIsrPassToStringAsync(element) {\n // The cache-fill render is a second render pass for the same request.\n // Reset render-scoped state so it cannot leak from the streamed response\n // render or affect async work that is still draining from that stream.\n // Keep request identity state (pathname/query/locale/executionContext)\n // intact: this second pass still belongs to the same request.\n return await runWithServerInsertedHTMLState(() =>\n runWithHeadState(() =>\n _runWithCacheState(() =>\n runWithPrivateCache(() => runWithFetchCache(async () => renderToStringAsync(element))),\n ),\n ),\n );\n}\n\n${pageImports.join(\"\\n\")}\n${apiImports.join(\"\\n\")}\n\n${appImportCode}\n${docImportCode}\n\nexport const pageRoutes = [\n${pageRouteEntries.join(\",\\n\")}\n];\nconst _pageRouteTrie = _buildRouteTrie(pageRoutes);\n\nconst apiRoutes = [\n${apiRouteEntries.join(\",\\n\")}\n];\nconst _apiRouteTrie = _buildRouteTrie(apiRoutes);\n\nfunction matchRoute(url, routes) {\n const pathname = url.split(\"?\")[0];\n let normalizedUrl = pathname === \"/\" ? \"/\" : pathname.replace(/\\\\/$/, \"\");\n // NOTE: Do NOT decodeURIComponent here. The pathname is already decoded at\n // the entry point. Decoding again would create a double-decode vector.\n const urlParts = normalizedUrl.split(\"/\").filter(Boolean);\n const trie = routes === pageRoutes ? _pageRouteTrie : _apiRouteTrie;\n return _trieMatch(trie, urlParts);\n}\n\nfunction parseQuery(url) {\n const qs = url.split(\"?\")[1];\n if (!qs) return {};\n const p = new URLSearchParams(qs);\n const q = {};\n for (const [k, v] of p) {\n if (k in q) {\n q[k] = Array.isArray(q[k]) ? q[k].concat(v) : [q[k], v];\n } else {\n q[k] = v;\n }\n }\n return q;\n}\n\nfunction patternToNextFormat(pattern) {\n return pattern\n .replace(/:([\\\\w]+)\\\\*/g, \"[[...$1]]\")\n .replace(/:([\\\\w]+)\\\\+/g, \"[...$1]\")\n .replace(/:([\\\\w]+)/g, \"[$1]\");\n}\n\nfunction collectAssetTags(manifest, moduleIds) {\n // Fall back to embedded manifest (set by vinext:cloudflare-build for Workers)\n const m = (manifest && Object.keys(manifest).length > 0)\n ? manifest\n : (typeof globalThis !== \"undefined\" && globalThis.__VINEXT_SSR_MANIFEST__) || null;\n const tags = [];\n const seen = new Set();\n\n // Load the set of lazy chunk filenames (only reachable via dynamic imports).\n // These should NOT get <link rel=\"modulepreload\"> or <script type=\"module\">\n // tags — they are fetched on demand when the dynamic import() executes (e.g.\n // chunks behind React.lazy() or next/dynamic boundaries).\n var lazyChunks = (typeof globalThis !== \"undefined\" && globalThis.__VINEXT_LAZY_CHUNKS__) || null;\n var lazySet = lazyChunks && lazyChunks.length > 0 ? new Set(lazyChunks) : null;\n\n // Inject the client entry script if embedded by vinext:cloudflare-build\n if (typeof globalThis !== \"undefined\" && globalThis.__VINEXT_CLIENT_ENTRY__) {\n const entry = globalThis.__VINEXT_CLIENT_ENTRY__;\n seen.add(entry);\n tags.push('<link rel=\"modulepreload\" href=\"/' + entry + '\" />');\n tags.push('<script type=\"module\" src=\"/' + entry + '\" crossorigin></script>');\n }\n if (m) {\n // Always inject shared chunks (framework, vinext runtime, entry) and\n // page-specific chunks. The manifest maps module file paths to their\n // associated JS/CSS assets.\n //\n // For page-specific injection, the module IDs may be absolute paths\n // while the manifest uses relative paths. Try both the original ID\n // and a suffix match to find the correct manifest entry.\n var allFiles = [];\n\n if (moduleIds && moduleIds.length > 0) {\n // Collect assets for the requested page modules\n for (var mi = 0; mi < moduleIds.length; mi++) {\n var id = moduleIds[mi];\n var files = m[id];\n if (!files) {\n // Absolute path didn't match — try matching by suffix.\n // Manifest keys are relative (e.g. \"pages/about.tsx\") while\n // moduleIds may be absolute (e.g. \"/home/.../pages/about.tsx\").\n for (var mk in m) {\n if (id.endsWith(\"/\" + mk) || id === mk) {\n files = m[mk];\n break;\n }\n }\n }\n if (files) {\n for (var fi = 0; fi < files.length; fi++) allFiles.push(files[fi]);\n }\n }\n\n // Also inject shared chunks that every page needs: framework,\n // vinext runtime, and the entry bootstrap. These are identified\n // by scanning all manifest values for chunk filenames containing\n // known prefixes.\n for (var key in m) {\n var vals = m[key];\n if (!vals) continue;\n for (var vi = 0; vi < vals.length; vi++) {\n var file = vals[vi];\n var basename = file.split(\"/\").pop() || \"\";\n if (\n basename.startsWith(\"framework-\") ||\n basename.startsWith(\"vinext-\") ||\n basename.includes(\"vinext-client-entry\") ||\n basename.includes(\"vinext-app-browser-entry\")\n ) {\n allFiles.push(file);\n }\n }\n }\n } else {\n // No specific modules — include all assets from manifest\n for (var akey in m) {\n var avals = m[akey];\n if (avals) {\n for (var ai = 0; ai < avals.length; ai++) allFiles.push(avals[ai]);\n }\n }\n }\n\n for (var ti = 0; ti < allFiles.length; ti++) {\n var tf = allFiles[ti];\n // Normalize: Vite's SSR manifest values include a leading '/'\n // (from base path), but we prepend '/' ourselves when building\n // href/src attributes. Strip any existing leading slash to avoid\n // producing protocol-relative URLs like \"//assets/chunk.js\".\n // This also ensures consistent keys for the seen-set dedup and\n // lazySet.has() checks (which use values without leading slash).\n if (tf.charAt(0) === '/') tf = tf.slice(1);\n if (seen.has(tf)) continue;\n seen.add(tf);\n if (tf.endsWith(\".css\")) {\n tags.push('<link rel=\"stylesheet\" href=\"/' + tf + '\" />');\n } else if (tf.endsWith(\".js\")) {\n // Skip lazy chunks — they are behind dynamic import() boundaries\n // (React.lazy, next/dynamic) and should only be fetched on demand.\n if (lazySet && lazySet.has(tf)) continue;\n tags.push('<link rel=\"modulepreload\" href=\"/' + tf + '\" />');\n tags.push('<script type=\"module\" src=\"/' + tf + '\" crossorigin></script>');\n }\n }\n }\n return tags.join(\"\\\\n \");\n}\n\n// i18n helpers\nfunction extractLocale(url) {\n if (!i18nConfig) return { locale: undefined, url, hadPrefix: false };\n const pathname = url.split(\"?\")[0];\n const parts = pathname.split(\"/\").filter(Boolean);\n const query = url.includes(\"?\") ? url.slice(url.indexOf(\"?\")) : \"\";\n if (parts.length > 0 && i18nConfig.locales.includes(parts[0])) {\n const locale = parts[0];\n const rest = \"/\" + parts.slice(1).join(\"/\");\n return { locale, url: (rest || \"/\") + query, hadPrefix: true };\n }\n return { locale: i18nConfig.defaultLocale, url, hadPrefix: false };\n}\n\nfunction detectLocaleFromHeaders(headers) {\n if (!i18nConfig) return null;\n const acceptLang = headers.get(\"accept-language\");\n if (!acceptLang) return null;\n const langs = acceptLang.split(\",\").map(function(part) {\n const pieces = part.trim().split(\";\");\n const q = pieces[1] ? parseFloat(pieces[1].replace(\"q=\", \"\")) : 1;\n return { lang: pieces[0].trim().toLowerCase(), q: q };\n }).sort(function(a, b) { return b.q - a.q; });\n for (let k = 0; k < langs.length; k++) {\n const lang = langs[k].lang;\n for (let j = 0; j < i18nConfig.locales.length; j++) {\n if (i18nConfig.locales[j].toLowerCase() === lang) return i18nConfig.locales[j];\n }\n const prefix = lang.split(\"-\")[0];\n for (let j = 0; j < i18nConfig.locales.length; j++) {\n const loc = i18nConfig.locales[j].toLowerCase();\n if (loc === prefix || loc.startsWith(prefix + \"-\")) return i18nConfig.locales[j];\n }\n }\n return null;\n}\n\nfunction parseCookieLocaleFromHeader(cookieHeader) {\n if (!i18nConfig || !cookieHeader) return null;\n const match = cookieHeader.match(/(?:^|;\\\\s*)NEXT_LOCALE=([^;]*)/);\n if (!match) return null;\n var value;\n try { value = decodeURIComponent(match[1].trim()); } catch (e) { return null; }\n if (i18nConfig.locales.indexOf(value) !== -1) return value;\n return null;\n}\n\n// Lightweight req/res facade for getServerSideProps and API routes.\n// Next.js pages expect ctx.req/ctx.res with Node-like shapes.\nfunction createReqRes(request, url, query, body) {\n const headersObj = {};\n for (const [k, v] of request.headers) headersObj[k.toLowerCase()] = v;\n\n const req = {\n method: request.method,\n url: url,\n headers: headersObj,\n query: query,\n body: body,\n cookies: parseCookies(request.headers.get(\"cookie\")),\n };\n\n let resStatusCode = 200;\n const resHeaders = {};\n // set-cookie needs array support (multiple Set-Cookie headers are common)\n const setCookieHeaders = [];\n let resBody = null;\n let ended = false;\n let resolveResponse;\n const responsePromise = new Promise(function(r) { resolveResponse = r; });\n\n const res = {\n get statusCode() { return resStatusCode; },\n set statusCode(code) { resStatusCode = code; },\n writeHead: function(code, headers) {\n resStatusCode = code;\n if (headers) {\n for (const [k, v] of Object.entries(headers)) {\n if (k.toLowerCase() === \"set-cookie\") {\n if (Array.isArray(v)) { for (const c of v) setCookieHeaders.push(c); }\n else { setCookieHeaders.push(v); }\n } else {\n resHeaders[k] = v;\n }\n }\n }\n return res;\n },\n setHeader: function(name, value) {\n if (name.toLowerCase() === \"set-cookie\") {\n if (Array.isArray(value)) { for (const c of value) setCookieHeaders.push(c); }\n else { setCookieHeaders.push(value); }\n } else {\n resHeaders[name.toLowerCase()] = value;\n }\n return res;\n },\n getHeader: function(name) {\n if (name.toLowerCase() === \"set-cookie\") return setCookieHeaders.length > 0 ? setCookieHeaders : undefined;\n return resHeaders[name.toLowerCase()];\n },\n end: function(data) {\n if (ended) return;\n ended = true;\n if (data !== undefined && data !== null) resBody = data;\n const h = new Headers(resHeaders);\n for (const c of setCookieHeaders) h.append(\"set-cookie\", c);\n resolveResponse(new Response(resBody, { status: resStatusCode, headers: h }));\n },\n status: function(code) { resStatusCode = code; return res; },\n json: function(data) {\n resHeaders[\"content-type\"] = \"application/json\";\n res.end(JSON.stringify(data));\n },\n send: function(data) {\n if (Buffer.isBuffer(data)) {\n if (!resHeaders[\"content-type\"]) resHeaders[\"content-type\"] = \"application/octet-stream\";\n resHeaders[\"content-length\"] = String(data.length);\n res.end(data);\n } else if (typeof data === \"object\" && data !== null) {\n res.json(data);\n } else {\n if (!resHeaders[\"content-type\"]) resHeaders[\"content-type\"] = \"text/plain\";\n res.end(String(data));\n }\n },\n redirect: function(statusOrUrl, url2) {\n if (typeof statusOrUrl === \"string\") { res.writeHead(307, { Location: statusOrUrl }); }\n else { res.writeHead(statusOrUrl, { Location: url2 }); }\n res.end();\n },\n getHeaders: function() {\n var h = Object.assign({}, resHeaders);\n if (setCookieHeaders.length > 0) h[\"set-cookie\"] = setCookieHeaders;\n return h;\n },\n get headersSent() { return ended; },\n };\n\n return { req, res, responsePromise };\n}\n\n/**\n * Read request body as text with a size limit.\n * Throws if the body exceeds maxBytes. This prevents DoS via chunked\n * transfer encoding where Content-Length is absent or spoofed.\n */\nasync function readBodyWithLimit(request, maxBytes) {\n if (!request.body) return \"\";\n var reader = request.body.getReader();\n var decoder = new TextDecoder();\n var chunks = [];\n var totalSize = 0;\n for (;;) {\n var result = await reader.read();\n if (result.done) break;\n totalSize += result.value.byteLength;\n if (totalSize > maxBytes) {\n reader.cancel();\n throw new Error(\"Request body too large\");\n }\n chunks.push(decoder.decode(result.value, { stream: true }));\n }\n chunks.push(decoder.decode());\n return chunks.join(\"\");\n}\n\nexport async function renderPage(request, url, manifest, ctx) {\n if (ctx) return _runWithExecutionContext(ctx, () => _renderPage(request, url, manifest));\n return _renderPage(request, url, manifest);\n}\n\nasync function _renderPage(request, url, manifest) {\n const localeInfo = i18nConfig\n ? resolvePagesI18nRequest(\n url,\n i18nConfig,\n request.headers,\n new URL(request.url).hostname,\n vinextConfig.basePath,\n vinextConfig.trailingSlash,\n )\n : { locale: undefined, url, hadPrefix: false, domainLocale: undefined, redirectUrl: undefined };\n const locale = localeInfo.locale;\n const routeUrl = localeInfo.url;\n const currentDefaultLocale = i18nConfig\n ? (localeInfo.domainLocale ? localeInfo.domainLocale.defaultLocale : i18nConfig.defaultLocale)\n : undefined;\n const domainLocales = i18nConfig ? i18nConfig.domains : undefined;\n\n if (localeInfo.redirectUrl) {\n return new Response(null, { status: 307, headers: { Location: localeInfo.redirectUrl } });\n }\n\n const match = matchRoute(routeUrl, pageRoutes);\n if (!match) {\n return new Response(\"<!DOCTYPE html><html><body><h1>404 - Page not found</h1></body></html>\",\n { status: 404, headers: { \"Content-Type\": \"text/html\" } });\n }\n\n\t const { route, params } = match;\n\t const __uCtx = _createUnifiedCtx({\n\t executionContext: _getRequestExecutionContext(),\n\t });\n\t return _runWithUnifiedCtx(__uCtx, async () => {\n\t ensureFetchPatch();\n\t try {\n\t const routePattern = patternToNextFormat(route.pattern);\n\t if (typeof setSSRContext === \"function\") {\n\t setSSRContext({\n\t pathname: routePattern,\n\t query: { ...params, ...parseQuery(routeUrl) },\n\t asPath: routeUrl,\n\t locale: locale,\n locales: i18nConfig ? i18nConfig.locales : undefined,\n defaultLocale: currentDefaultLocale,\n domainLocales: domainLocales,\n });\n }\n\n if (i18nConfig) {\n setI18nContext({\n locale: locale,\n locales: i18nConfig.locales,\n defaultLocale: currentDefaultLocale,\n domainLocales: domainLocales,\n hostname: new URL(request.url).hostname,\n });\n }\n\n const pageModule = route.module;\n const PageComponent = pageModule.default;\n\t if (!PageComponent) {\n\t return new Response(\"Page has no default export\", { status: 500 });\n\t }\n\t // Build font Link header early so it's available for ISR cached responses too.\n\t // Font preloads are module-level state populated at import time and persist across requests.\n\t var _fontLinkHeader = \"\";\n\t var _allFp = [];\n try {\n var _fpGoogle = typeof _getSSRFontPreloadsGoogle === \"function\" ? _getSSRFontPreloadsGoogle() : [];\n var _fpLocal = typeof _getSSRFontPreloadsLocal === \"function\" ? _getSSRFontPreloadsLocal() : [];\n _allFp = _fpGoogle.concat(_fpLocal);\n\t if (_allFp.length > 0) {\n\t _fontLinkHeader = _allFp.map(function(p) { return \"<\" + p.href + \">; rel=preload; as=font; type=\" + p.type + \"; crossorigin\"; }).join(\", \");\n\t }\n\t } catch (e) { /* font preloads not available */ }\n\t const query = parseQuery(routeUrl);\n\t const pageDataResult = await __resolvePagesPageData({\n\t applyRequestContexts() {\n\t if (typeof setSSRContext === \"function\") {\n\t setSSRContext({\n\t pathname: routePattern,\n\t query: { ...params, ...query },\n\t asPath: routeUrl,\n\t locale: locale,\n\t locales: i18nConfig ? i18nConfig.locales : undefined,\n\t defaultLocale: currentDefaultLocale,\n\t domainLocales: domainLocales,\n\t });\n\t }\n\t if (i18nConfig) {\n\t setI18nContext({\n\t locale: locale,\n\t locales: i18nConfig.locales,\n\t defaultLocale: currentDefaultLocale,\n\t domainLocales: domainLocales,\n\t hostname: new URL(request.url).hostname,\n\t });\n\t }\n\t },\n\t buildId,\n\t createGsspReqRes() {\n\t return createReqRes(request, routeUrl, query, undefined);\n\t },\n\t createPageElement(currentPageProps) {\n\t var currentElement = AppComponent\n\t ? React.createElement(AppComponent, { Component: PageComponent, pageProps: currentPageProps })\n\t : React.createElement(PageComponent, currentPageProps);\n\t return wrapWithRouterContext(currentElement);\n\t },\n\t fontLinkHeader: _fontLinkHeader,\n\t i18n: {\n\t locale: locale,\n\t locales: i18nConfig ? i18nConfig.locales : undefined,\n\t defaultLocale: currentDefaultLocale,\n\t domainLocales: domainLocales,\n\t },\n\t isrCacheKey,\n\t isrGet,\n\t isrSet,\n\t pageModule,\n\t params,\n\t query,\n\t renderIsrPassToStringAsync,\n\t route: {\n\t isDynamic: route.isDynamic,\n\t },\n\t routePattern,\n\t routeUrl,\n\t runInFreshUnifiedContext(callback) {\n\t var revalCtx = _createUnifiedCtx({\n\t executionContext: _getRequestExecutionContext(),\n\t });\n\t return _runWithUnifiedCtx(revalCtx, async () => {\n\t ensureFetchPatch();\n\t return callback();\n\t });\n\t },\n\t safeJsonStringify,\n\t sanitizeDestination: sanitizeDestinationLocal,\n\t triggerBackgroundRegeneration,\n\t });\n\t if (pageDataResult.kind === \"response\") {\n\t return pageDataResult.response;\n\t }\n\t let pageProps = pageDataResult.pageProps;\n\t var gsspRes = pageDataResult.gsspRes;\n\t let isrRevalidateSeconds = pageDataResult.isrRevalidateSeconds;\n\n\t const pageModuleIds = route.filePath ? [route.filePath] : [];\n\t const assetTags = collectAssetTags(manifest, pageModuleIds);\n\n return __renderPagesPageResponse({\n assetTags,\n buildId,\n clearSsrContext() {\n if (typeof setSSRContext === \"function\") setSSRContext(null);\n },\n createPageElement(currentPageProps) {\n var currentElement;\n if (AppComponent) {\n currentElement = React.createElement(AppComponent, { Component: PageComponent, pageProps: currentPageProps });\n } else {\n currentElement = React.createElement(PageComponent, currentPageProps);\n }\n return wrapWithRouterContext(currentElement);\n },\n DocumentComponent,\n flushPreloads: typeof flushPreloads === \"function\" ? flushPreloads : undefined,\n fontLinkHeader: _fontLinkHeader,\n fontPreloads: _allFp,\n getFontLinks() {\n try {\n return typeof _getSSRFontLinks === \"function\" ? _getSSRFontLinks() : [];\n } catch (e) {\n return [];\n }\n },\n getFontStyles() {\n try {\n var allFontStyles = [];\n if (typeof _getSSRFontStylesGoogle === \"function\") allFontStyles.push(..._getSSRFontStylesGoogle());\n if (typeof _getSSRFontStylesLocal === \"function\") allFontStyles.push(..._getSSRFontStylesLocal());\n return allFontStyles;\n } catch (e) {\n return [];\n }\n },\n\t getSSRHeadHTML: typeof getSSRHeadHTML === \"function\" ? getSSRHeadHTML : undefined,\n\t gsspRes,\n isrCacheKey,\n isrRevalidateSeconds,\n isrSet,\n i18n: {\n locale: locale,\n locales: i18nConfig ? i18nConfig.locales : undefined,\n defaultLocale: currentDefaultLocale,\n domainLocales: domainLocales,\n },\n pageProps,\n params,\n renderDocumentToString(element) {\n return renderToStringAsync(element);\n },\n renderIsrPassToStringAsync,\n renderToReadableStream(element) {\n return renderToReadableStream(element);\n },\n resetSSRHead: typeof resetSSRHead === \"function\" ? resetSSRHead : undefined,\n\t routePattern,\n routeUrl,\n safeJsonStringify,\n });\n } catch (e) {\n console.error(\"[vinext] SSR error:\", e);\n _reportRequestError(\n e instanceof Error ? e : new Error(String(e)),\n { path: url, method: request.method, headers: Object.fromEntries(request.headers.entries()) },\n { routerKind: \"Pages Router\", routePath: route.pattern, routeType: \"render\" },\n ).catch(() => { /* ignore reporting errors */ });\n return new Response(\"Internal Server Error\", { status: 500 });\n }\n });\n}\n\nexport async function handleApiRoute(request, url) {\n const match = matchRoute(url, apiRoutes);\n if (!match) {\n return new Response(\"404 - API route not found\", { status: 404 });\n }\n\n const { route, params } = match;\n const handler = route.module.default;\n if (typeof handler !== \"function\") {\n return new Response(\"API route does not export a default function\", { status: 500 });\n }\n\n const query = { ...params };\n const qs = url.split(\"?\")[1];\n if (qs) {\n for (const [k, v] of new URLSearchParams(qs)) {\n if (k in query) {\n // Multi-value: promote to array (Next.js returns string[] for duplicate keys)\n query[k] = Array.isArray(query[k]) ? query[k].concat(v) : [query[k], v];\n } else {\n query[k] = v;\n }\n }\n }\n\n // Parse request body (enforce 1MB limit to prevent memory exhaustion,\n // matching Next.js default bodyParser sizeLimit).\n // Check Content-Length first as a fast path, then enforce on the actual\n // stream to prevent bypasses via chunked transfer encoding.\n const contentLength = parseInt(request.headers.get(\"content-length\") || \"0\", 10);\n if (contentLength > 1 * 1024 * 1024) {\n return new Response(\"Request body too large\", { status: 413 });\n }\n try {\n let body;\n const mediaType = getMediaType(request.headers.get(\"content-type\"));\n let rawBody;\n try { rawBody = await readBodyWithLimit(request, 1 * 1024 * 1024); }\n catch { return new Response(\"Request body too large\", { status: 413 }); }\n if (!rawBody) {\n body = isJsonMediaType(mediaType)\n ? {}\n : mediaType === \"application/x-www-form-urlencoded\"\n ? decodeQueryString(rawBody)\n : undefined;\n } else if (isJsonMediaType(mediaType)) {\n try { body = JSON.parse(rawBody); }\n catch { throw new ApiBodyParseError(\"Invalid JSON\", 400); }\n } else if (mediaType === \"application/x-www-form-urlencoded\") {\n body = decodeQueryString(rawBody);\n } else {\n body = rawBody;\n }\n\n const { req, res, responsePromise } = createReqRes(request, url, query, body);\n await handler(req, res);\n // If handler didn't call res.end(), end it now.\n // The end() method is idempotent — safe to call twice.\n res.end();\n return await responsePromise;\n } catch (e) {\n if (e instanceof ApiBodyParseError) {\n return new Response(e.message, { status: e.statusCode, statusText: e.message });\n }\n console.error(\"[vinext] API error:\", e);\n _reportRequestError(\n e instanceof Error ? e : new Error(String(e)),\n { path: url, method: request.method, headers: Object.fromEntries(request.headers.entries()) },\n { routerKind: \"Pages Router\", routePath: route.pattern, routeType: \"route\" },\n );\n return new Response(\"Internal Server Error\", { status: 500 });\n }\n}\n\n${middlewareExportCode}\n`;\n}\n"],"mappings":"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"}
1
+ {"version":3,"file":"pages-server-entry.js","names":[],"sources":["../../src/entries/pages-server-entry.ts"],"sourcesContent":["/**\n * Pages Router server entry generator.\n *\n * Generates the virtual SSR server entry module (`virtual:vinext-server-entry`).\n * This is the entry point for `vite build --ssr`. It handles SSR, API routes,\n * middleware, ISR, and i18n for the Pages Router.\n *\n * Extracted from index.ts.\n */\nimport { resolveEntryPath } from \"./runtime-entry-module.js\";\nimport { pagesRouter, apiRouter, type Route } from \"../routing/pages-router.js\";\nimport { createValidFileMatcher } from \"../routing/file-matcher.js\";\nimport { type ResolvedNextConfig } from \"../config/next-config.js\";\nimport { isProxyFile } from \"../server/middleware.js\";\nimport {\n generateSafeRegExpCode,\n generateMiddlewareMatcherCode,\n generateNormalizePathCode,\n generateRouteMatchNormalizationCode,\n} from \"../server/middleware-codegen.js\";\nimport { findFileWithExts } from \"./pages-entry-helpers.js\";\n\nconst _requestContextShimPath = resolveEntryPath(\"../shims/request-context.js\", import.meta.url);\nconst _routeTriePath = resolveEntryPath(\"../routing/route-trie.js\", import.meta.url);\nconst _pagesI18nPath = resolveEntryPath(\"../server/pages-i18n.js\", import.meta.url);\nconst _pagesPageResponsePath = resolveEntryPath(\n \"../server/pages-page-response.js\",\n import.meta.url,\n);\nconst _pagesPageDataPath = resolveEntryPath(\"../server/pages-page-data.js\", import.meta.url);\nconst _pagesNodeCompatPath = resolveEntryPath(\"../server/pages-node-compat.js\", import.meta.url);\nconst _pagesApiRoutePath = resolveEntryPath(\"../server/pages-api-route.js\", import.meta.url);\nconst _isrCachePath = resolveEntryPath(\"../server/isr-cache.js\", import.meta.url);\n\n/**\n * Generate the virtual SSR server entry module.\n * This is the entry point for `vite build --ssr`.\n */\nexport async function generateServerEntry(\n pagesDir: string,\n nextConfig: ResolvedNextConfig,\n fileMatcher: ReturnType<typeof createValidFileMatcher>,\n middlewarePath: string | null,\n instrumentationPath: string | null,\n): Promise<string> {\n const pageRoutes = await pagesRouter(pagesDir, nextConfig?.pageExtensions, fileMatcher);\n const apiRoutes = await apiRouter(pagesDir, nextConfig?.pageExtensions, fileMatcher);\n\n // Generate import statements using absolute paths since virtual\n // modules don't have a real file location for relative resolution.\n const pageImports = pageRoutes.map((r: Route, i: number) => {\n const absPath = r.filePath.replace(/\\\\/g, \"/\");\n return `import * as page_${i} from ${JSON.stringify(absPath)};`;\n });\n\n const apiImports = apiRoutes.map((r: Route, i: number) => {\n const absPath = r.filePath.replace(/\\\\/g, \"/\");\n return `import * as api_${i} from ${JSON.stringify(absPath)};`;\n });\n\n // Build the route table — include filePath for SSR manifest lookup\n const pageRouteEntries = pageRoutes.map((r: Route, i: number) => {\n const absPath = r.filePath.replace(/\\\\/g, \"/\");\n return ` { pattern: ${JSON.stringify(r.pattern)}, patternParts: ${JSON.stringify(r.patternParts)}, isDynamic: ${r.isDynamic}, params: ${JSON.stringify(r.params)}, module: page_${i}, filePath: ${JSON.stringify(absPath)} }`;\n });\n\n const apiRouteEntries = apiRoutes.map(\n (r: Route, i: number) =>\n ` { pattern: ${JSON.stringify(r.pattern)}, patternParts: ${JSON.stringify(r.patternParts)}, isDynamic: ${r.isDynamic}, params: ${JSON.stringify(r.params)}, module: api_${i} }`,\n );\n\n // Check for _app and _document\n const appFilePath = findFileWithExts(pagesDir, \"_app\", fileMatcher);\n const docFilePath = findFileWithExts(pagesDir, \"_document\", fileMatcher);\n const appImportCode =\n appFilePath !== null\n ? `import { default as AppComponent } from ${JSON.stringify(appFilePath.replace(/\\\\/g, \"/\"))};`\n : `const AppComponent = null;`;\n\n const docImportCode =\n docFilePath !== null\n ? `import { default as DocumentComponent } from ${JSON.stringify(docFilePath.replace(/\\\\/g, \"/\"))};`\n : `const DocumentComponent = null;`;\n\n // Serialize i18n config for embedding in the server entry\n const i18nConfigJson = nextConfig?.i18n\n ? JSON.stringify({\n locales: nextConfig.i18n.locales,\n defaultLocale: nextConfig.i18n.defaultLocale,\n localeDetection: nextConfig.i18n.localeDetection,\n domains: nextConfig.i18n.domains,\n })\n : \"null\";\n\n // Embed the resolved build ID at build time\n const buildIdJson = JSON.stringify(nextConfig?.buildId ?? null);\n\n // Serialize the full resolved config for the production server.\n // This embeds redirects, rewrites, headers, basePath, trailingSlash\n // so prod-server.ts can apply them without loading next.config.js at runtime.\n const vinextConfigJson = JSON.stringify({\n basePath: nextConfig?.basePath ?? \"\",\n trailingSlash: nextConfig?.trailingSlash ?? false,\n redirects: nextConfig?.redirects ?? [],\n rewrites: nextConfig?.rewrites ?? { beforeFiles: [], afterFiles: [], fallback: [] },\n headers: nextConfig?.headers ?? [],\n i18n: nextConfig?.i18n ?? null,\n images: {\n deviceSizes: nextConfig?.images?.deviceSizes,\n imageSizes: nextConfig?.images?.imageSizes,\n dangerouslyAllowSVG: nextConfig?.images?.dangerouslyAllowSVG,\n contentDispositionType: nextConfig?.images?.contentDispositionType,\n contentSecurityPolicy: nextConfig?.images?.contentSecurityPolicy,\n },\n });\n\n // Generate instrumentation code if instrumentation.ts exists.\n // For production (Cloudflare Workers), instrumentation.ts is bundled into the\n // Worker and register() is called as a top-level await at module evaluation time —\n // before any request is handled. This mirrors App Router behavior (generateRscEntry)\n // and matches Next.js semantics: register() runs once on startup in the process\n // that handles requests.\n //\n // The onRequestError handler is stored on globalThis so it is visible across\n // all code within the Worker (same global scope).\n const instrumentationImportCode = instrumentationPath\n ? `import * as _instrumentation from ${JSON.stringify(instrumentationPath.replace(/\\\\/g, \"/\"))};`\n : \"\";\n\n const instrumentationInitCode = instrumentationPath\n ? `// Run instrumentation register() once at module evaluation time — before any\n// requests are handled. Matches Next.js semantics: register() is called once\n// on startup in the process that handles requests.\nif (typeof _instrumentation.register === \"function\") {\n await _instrumentation.register();\n}\n// Store the onRequestError handler on globalThis so it is visible to all\n// code within the Worker (same global scope).\nif (typeof _instrumentation.onRequestError === \"function\") {\n globalThis.__VINEXT_onRequestErrorHandler__ = _instrumentation.onRequestError;\n}`\n : \"\";\n\n // Generate middleware code if middleware.ts exists\n const middlewareImportCode = middlewarePath\n ? `import * as middlewareModule from ${JSON.stringify(middlewarePath.replace(/\\\\/g, \"/\"))};\nimport { NextRequest, NextFetchEvent } from \"next/server\";`\n : \"\";\n\n // The matcher config is read from the middleware module at import time.\n // We inline the matching + execution logic so the prod server can call it.\n const middlewareExportCode = middlewarePath\n ? `\n// --- Middleware support (generated from middleware-codegen.ts) ---\n${generateNormalizePathCode(\"es5\")}\n${generateRouteMatchNormalizationCode(\"es5\")}\n${generateSafeRegExpCode(\"es5\")}\n${generateMiddlewareMatcherCode(\"es5\")}\n\nexport async function runMiddleware(request, ctx) {\n if (ctx) return _runWithExecutionContext(ctx, () => _runMiddleware(request));\n return _runMiddleware(request);\n}\n\nasync function _runMiddleware(request) {\n var isProxy = ${middlewarePath ? JSON.stringify(isProxyFile(middlewarePath)) : \"false\"};\n var middlewareFn = isProxy\n ? (middlewareModule.proxy ?? middlewareModule.default)\n : (middlewareModule.middleware ?? middlewareModule.default);\n if (typeof middlewareFn !== \"function\") {\n var fileType = isProxy ? \"Proxy\" : \"Middleware\";\n var expectedExport = isProxy ? \"proxy\" : \"middleware\";\n throw new Error(\"The \" + fileType + \" file must export a function named \\`\" + expectedExport + \"\\` or a \\`default\\` function.\");\n }\n\n var config = middlewareModule.config;\n var matcher = config && config.matcher;\n var url = new URL(request.url);\n\n // Normalize pathname before matching to prevent path-confusion bypasses\n // (percent-encoding like /%61dmin, double slashes like /dashboard//settings).\n var decodedPathname;\n try { decodedPathname = __normalizePathnameForRouteMatchStrict(url.pathname); } catch (e) {\n return { continue: false, response: new Response(\"Bad Request\", { status: 400 }) };\n }\n var normalizedPathname = __normalizePath(decodedPathname);\n\n if (!matchesMiddleware(normalizedPathname, matcher, request, i18nConfig)) return { continue: true };\n\n // Construct a new Request with the decoded + normalized pathname so middleware\n // always sees the same canonical path that the router uses.\n var mwRequest = request;\n if (normalizedPathname !== url.pathname) {\n var mwUrl = new URL(url);\n mwUrl.pathname = normalizedPathname;\n mwRequest = new Request(mwUrl, request);\n }\n var __mwNextConfig = (vinextConfig.basePath || i18nConfig) ? { basePath: vinextConfig.basePath, i18n: i18nConfig || undefined } : undefined;\n var nextRequest = mwRequest instanceof NextRequest ? mwRequest : new NextRequest(mwRequest, __mwNextConfig ? { nextConfig: __mwNextConfig } : undefined);\n var fetchEvent = new NextFetchEvent({ page: normalizedPathname });\n var response;\n try { response = await middlewareFn(nextRequest, fetchEvent); }\n catch (e) {\n console.error(\"[vinext] Middleware error:\", e);\n var _mwCtxErr = _getRequestExecutionContext();\n if (_mwCtxErr && typeof _mwCtxErr.waitUntil === \"function\") { _mwCtxErr.waitUntil(fetchEvent.drainWaitUntil()); } else { fetchEvent.drainWaitUntil(); }\n return { continue: false, response: new Response(\"Internal Server Error\", { status: 500 }) };\n }\n var _mwCtx = _getRequestExecutionContext();\n if (_mwCtx && typeof _mwCtx.waitUntil === \"function\") { _mwCtx.waitUntil(fetchEvent.drainWaitUntil()); } else { fetchEvent.drainWaitUntil(); }\n\n if (!response) return { continue: true };\n\n if (response.headers.get(\"x-middleware-next\") === \"1\") {\n var rHeaders = new Headers();\n for (var [key, value] of response.headers) {\n // Keep x-middleware-request-* headers so the production server can\n // apply middleware-request header overrides before stripping internals\n // from the final client response.\n if (\n !key.startsWith(\"x-middleware-\") ||\n key === \"x-middleware-override-headers\" ||\n key.startsWith(\"x-middleware-request-\")\n ) rHeaders.append(key, value);\n }\n return { continue: true, responseHeaders: rHeaders };\n }\n\n if (response.status >= 300 && response.status < 400) {\n var location = response.headers.get(\"Location\") || response.headers.get(\"location\");\n if (location) {\n var rdHeaders = new Headers();\n for (var [rk, rv] of response.headers) {\n if (!rk.startsWith(\"x-middleware-\") && rk.toLowerCase() !== \"location\") rdHeaders.append(rk, rv);\n }\n return { continue: false, redirectUrl: location, redirectStatus: response.status, responseHeaders: rdHeaders };\n }\n }\n\n var rewriteUrl = response.headers.get(\"x-middleware-rewrite\");\n if (rewriteUrl) {\n var rwHeaders = new Headers();\n for (var [k, v] of response.headers) {\n if (!k.startsWith(\"x-middleware-\") || k === \"x-middleware-override-headers\" || k.startsWith(\"x-middleware-request-\")) rwHeaders.append(k, v);\n }\n var rewritePath;\n try { var parsed = new URL(rewriteUrl, request.url); rewritePath = parsed.pathname + parsed.search; }\n catch { rewritePath = rewriteUrl; }\n return { continue: true, rewriteUrl: rewritePath, rewriteStatus: response.status !== 200 ? response.status : undefined, responseHeaders: rwHeaders };\n }\n\n return { continue: false, response: response };\n}\n`\n : `\nexport async function runMiddleware() { return { continue: true }; }\n`;\n\n // The server entry is a self-contained module that uses Web-standard APIs\n // (Request/Response, renderToReadableStream) so it runs on Cloudflare Workers.\n return `\nimport React from \"react\";\nimport { renderToReadableStream } from \"react-dom/server.edge\";\nimport { resetSSRHead, getSSRHeadHTML } from \"next/head\";\nimport { flushPreloads } from \"next/dynamic\";\nimport { setSSRContext, wrapWithRouterContext } from \"next/router\";\nimport { _runWithCacheState } from \"next/cache\";\nimport { runWithPrivateCache } from \"vinext/cache-runtime\";\nimport { ensureFetchPatch, runWithFetchCache } from \"vinext/fetch-cache\";\nimport { runWithRequestContext as _runWithUnifiedCtx, createRequestContext as _createUnifiedCtx } from \"vinext/unified-request-context\";\nimport \"vinext/router-state\";\nimport { runWithServerInsertedHTMLState } from \"vinext/navigation-state\";\nimport { runWithHeadState } from \"vinext/head-state\";\nimport \"vinext/i18n-state\";\nimport { setI18nContext } from \"vinext/i18n-context\";\nimport { safeJsonStringify } from \"vinext/html\";\nimport { getSSRFontLinks as _getSSRFontLinks, getSSRFontStyles as _getSSRFontStylesGoogle, getSSRFontPreloads as _getSSRFontPreloadsGoogle } from \"next/font/google\";\nimport { getSSRFontStyles as _getSSRFontStylesLocal, getSSRFontPreloads as _getSSRFontPreloadsLocal } from \"next/font/local\";\nimport { sanitizeDestination as sanitizeDestinationLocal } from ${JSON.stringify(resolveEntryPath(\"../config/config-matchers.js\", import.meta.url))};\nimport { runWithExecutionContext as _runWithExecutionContext, getRequestExecutionContext as _getRequestExecutionContext } from ${JSON.stringify(_requestContextShimPath)};\nimport { buildRouteTrie as _buildRouteTrie, trieMatch as _trieMatch } from ${JSON.stringify(_routeTriePath)};\nimport { reportRequestError as _reportRequestError } from \"vinext/instrumentation\";\nimport { resolvePagesI18nRequest } from ${JSON.stringify(_pagesI18nPath)};\nimport { createPagesReqRes as __createPagesReqRes } from ${JSON.stringify(_pagesNodeCompatPath)};\nimport { handlePagesApiRoute as __handlePagesApiRoute } from ${JSON.stringify(_pagesApiRoutePath)};\nimport {\n isrGet as __sharedIsrGet,\n isrSet as __sharedIsrSet,\n isrCacheKey as __sharedIsrCacheKey,\n triggerBackgroundRegeneration as __sharedTriggerBackgroundRegeneration,\n} from ${JSON.stringify(_isrCachePath)};\nimport { resolvePagesPageData as __resolvePagesPageData } from ${JSON.stringify(_pagesPageDataPath)};\nimport { renderPagesPageResponse as __renderPagesPageResponse } from ${JSON.stringify(_pagesPageResponsePath)};\n${instrumentationImportCode}\n${middlewareImportCode}\n\n${instrumentationInitCode}\n\n// i18n config (embedded at build time)\nconst i18nConfig = ${i18nConfigJson};\n\n// Build ID (embedded at build time)\nconst buildId = ${buildIdJson};\n\n// Full resolved config for production server (embedded at build time)\nexport const vinextConfig = ${vinextConfigJson};\n\nfunction isrGet(key) {\n return __sharedIsrGet(key);\n}\nfunction isrSet(key, data, revalidateSeconds, tags) {\n return __sharedIsrSet(key, data, revalidateSeconds, tags);\n}\nfunction triggerBackgroundRegeneration(key, renderFn) {\n return __sharedTriggerBackgroundRegeneration(key, renderFn);\n}\nfunction isrCacheKey(router, pathname) {\n return __sharedIsrCacheKey(router, pathname, buildId || undefined);\n}\n\nasync function renderToStringAsync(element) {\n const stream = await renderToReadableStream(element);\n await stream.allReady;\n return new Response(stream).text();\n}\n\nasync function renderIsrPassToStringAsync(element) {\n // The cache-fill render is a second render pass for the same request.\n // Reset render-scoped state so it cannot leak from the streamed response\n // render or affect async work that is still draining from that stream.\n // Keep request identity state (pathname/query/locale/executionContext)\n // intact: this second pass still belongs to the same request.\n return await runWithServerInsertedHTMLState(() =>\n runWithHeadState(() =>\n _runWithCacheState(() =>\n runWithPrivateCache(() => runWithFetchCache(async () => renderToStringAsync(element))),\n ),\n ),\n );\n}\n\n${pageImports.join(\"\\n\")}\n${apiImports.join(\"\\n\")}\n\n${appImportCode}\n${docImportCode}\n\nexport const pageRoutes = [\n${pageRouteEntries.join(\",\\n\")}\n];\nconst _pageRouteTrie = _buildRouteTrie(pageRoutes);\n\nconst apiRoutes = [\n${apiRouteEntries.join(\",\\n\")}\n];\nconst _apiRouteTrie = _buildRouteTrie(apiRoutes);\n\nfunction matchRoute(url, routes) {\n const pathname = url.split(\"?\")[0];\n let normalizedUrl = pathname === \"/\" ? \"/\" : pathname.replace(/\\\\/$/, \"\");\n // NOTE: Do NOT decodeURIComponent here. The pathname is already decoded at\n // the entry point. Decoding again would create a double-decode vector.\n const urlParts = normalizedUrl.split(\"/\").filter(Boolean);\n const trie = routes === pageRoutes ? _pageRouteTrie : _apiRouteTrie;\n return _trieMatch(trie, urlParts);\n}\n\nfunction parseQuery(url) {\n const qs = url.split(\"?\")[1];\n if (!qs) return {};\n const p = new URLSearchParams(qs);\n const q = {};\n for (const [k, v] of p) {\n if (k in q) {\n q[k] = Array.isArray(q[k]) ? q[k].concat(v) : [q[k], v];\n } else {\n q[k] = v;\n }\n }\n return q;\n}\n\nfunction patternToNextFormat(pattern) {\n return pattern\n .replace(/:([\\\\w]+)\\\\*/g, \"[[...$1]]\")\n .replace(/:([\\\\w]+)\\\\+/g, \"[...$1]\")\n .replace(/:([\\\\w]+)/g, \"[$1]\");\n}\n\nfunction collectAssetTags(manifest, moduleIds) {\n // Fall back to embedded manifest (set by vinext:cloudflare-build for Workers)\n const m = (manifest && Object.keys(manifest).length > 0)\n ? manifest\n : (typeof globalThis !== \"undefined\" && globalThis.__VINEXT_SSR_MANIFEST__) || null;\n const tags = [];\n const seen = new Set();\n\n // Load the set of lazy chunk filenames (only reachable via dynamic imports).\n // These should NOT get <link rel=\"modulepreload\"> or <script type=\"module\">\n // tags — they are fetched on demand when the dynamic import() executes (e.g.\n // chunks behind React.lazy() or next/dynamic boundaries).\n var lazyChunks = (typeof globalThis !== \"undefined\" && globalThis.__VINEXT_LAZY_CHUNKS__) || null;\n var lazySet = lazyChunks && lazyChunks.length > 0 ? new Set(lazyChunks) : null;\n\n // Inject the client entry script if embedded by vinext:cloudflare-build\n if (typeof globalThis !== \"undefined\" && globalThis.__VINEXT_CLIENT_ENTRY__) {\n const entry = globalThis.__VINEXT_CLIENT_ENTRY__;\n seen.add(entry);\n tags.push('<link rel=\"modulepreload\" href=\"/' + entry + '\" />');\n tags.push('<script type=\"module\" src=\"/' + entry + '\" crossorigin></script>');\n }\n if (m) {\n // Always inject shared chunks (framework, vinext runtime, entry) and\n // page-specific chunks. The manifest maps module file paths to their\n // associated JS/CSS assets.\n //\n // For page-specific injection, the module IDs may be absolute paths\n // while the manifest uses relative paths. Try both the original ID\n // and a suffix match to find the correct manifest entry.\n var allFiles = [];\n\n if (moduleIds && moduleIds.length > 0) {\n // Collect assets for the requested page modules\n for (var mi = 0; mi < moduleIds.length; mi++) {\n var id = moduleIds[mi];\n var files = m[id];\n if (!files) {\n // Absolute path didn't match — try matching by suffix.\n // Manifest keys are relative (e.g. \"pages/about.tsx\") while\n // moduleIds may be absolute (e.g. \"/home/.../pages/about.tsx\").\n for (var mk in m) {\n if (id.endsWith(\"/\" + mk) || id === mk) {\n files = m[mk];\n break;\n }\n }\n }\n if (files) {\n for (var fi = 0; fi < files.length; fi++) allFiles.push(files[fi]);\n }\n }\n\n // Also inject shared chunks that every page needs: framework,\n // vinext runtime, and the entry bootstrap. These are identified\n // by scanning all manifest values for chunk filenames containing\n // known prefixes.\n for (var key in m) {\n var vals = m[key];\n if (!vals) continue;\n for (var vi = 0; vi < vals.length; vi++) {\n var file = vals[vi];\n var basename = file.split(\"/\").pop() || \"\";\n if (\n basename.startsWith(\"framework-\") ||\n basename.startsWith(\"vinext-\") ||\n basename.includes(\"vinext-client-entry\") ||\n basename.includes(\"vinext-app-browser-entry\")\n ) {\n allFiles.push(file);\n }\n }\n }\n } else {\n // No specific modules — include all assets from manifest\n for (var akey in m) {\n var avals = m[akey];\n if (avals) {\n for (var ai = 0; ai < avals.length; ai++) allFiles.push(avals[ai]);\n }\n }\n }\n\n for (var ti = 0; ti < allFiles.length; ti++) {\n var tf = allFiles[ti];\n // Normalize: Vite's SSR manifest values include a leading '/'\n // (from base path), but we prepend '/' ourselves when building\n // href/src attributes. Strip any existing leading slash to avoid\n // producing protocol-relative URLs like \"//assets/chunk.js\".\n // This also ensures consistent keys for the seen-set dedup and\n // lazySet.has() checks (which use values without leading slash).\n if (tf.charAt(0) === '/') tf = tf.slice(1);\n if (seen.has(tf)) continue;\n seen.add(tf);\n if (tf.endsWith(\".css\")) {\n tags.push('<link rel=\"stylesheet\" href=\"/' + tf + '\" />');\n } else if (tf.endsWith(\".js\")) {\n // Skip lazy chunks — they are behind dynamic import() boundaries\n // (React.lazy, next/dynamic) and should only be fetched on demand.\n if (lazySet && lazySet.has(tf)) continue;\n tags.push('<link rel=\"modulepreload\" href=\"/' + tf + '\" />');\n tags.push('<script type=\"module\" src=\"/' + tf + '\" crossorigin></script>');\n }\n }\n }\n return tags.join(\"\\\\n \");\n}\n\n// i18n helpers\nfunction extractLocale(url) {\n if (!i18nConfig) return { locale: undefined, url, hadPrefix: false };\n const pathname = url.split(\"?\")[0];\n const parts = pathname.split(\"/\").filter(Boolean);\n const query = url.includes(\"?\") ? url.slice(url.indexOf(\"?\")) : \"\";\n if (parts.length > 0 && i18nConfig.locales.includes(parts[0])) {\n const locale = parts[0];\n const rest = \"/\" + parts.slice(1).join(\"/\");\n return { locale, url: (rest || \"/\") + query, hadPrefix: true };\n }\n return { locale: i18nConfig.defaultLocale, url, hadPrefix: false };\n}\n\nfunction detectLocaleFromHeaders(headers) {\n if (!i18nConfig) return null;\n const acceptLang = headers.get(\"accept-language\");\n if (!acceptLang) return null;\n const langs = acceptLang.split(\",\").map(function(part) {\n const pieces = part.trim().split(\";\");\n const q = pieces[1] ? parseFloat(pieces[1].replace(\"q=\", \"\")) : 1;\n return { lang: pieces[0].trim().toLowerCase(), q: q };\n }).sort(function(a, b) { return b.q - a.q; });\n for (let k = 0; k < langs.length; k++) {\n const lang = langs[k].lang;\n for (let j = 0; j < i18nConfig.locales.length; j++) {\n if (i18nConfig.locales[j].toLowerCase() === lang) return i18nConfig.locales[j];\n }\n const prefix = lang.split(\"-\")[0];\n for (let j = 0; j < i18nConfig.locales.length; j++) {\n const loc = i18nConfig.locales[j].toLowerCase();\n if (loc === prefix || loc.startsWith(prefix + \"-\")) return i18nConfig.locales[j];\n }\n }\n return null;\n}\n\nfunction parseCookieLocaleFromHeader(cookieHeader) {\n if (!i18nConfig || !cookieHeader) return null;\n const match = cookieHeader.match(/(?:^|;\\\\s*)NEXT_LOCALE=([^;]*)/);\n if (!match) return null;\n var value;\n try { value = decodeURIComponent(match[1].trim()); } catch (e) { return null; }\n if (i18nConfig.locales.indexOf(value) !== -1) return value;\n return null;\n}\n\nexport async function renderPage(request, url, manifest, ctx) {\n if (ctx) return _runWithExecutionContext(ctx, () => _renderPage(request, url, manifest));\n return _renderPage(request, url, manifest);\n}\n\nasync function _renderPage(request, url, manifest) {\n const localeInfo = i18nConfig\n ? resolvePagesI18nRequest(\n url,\n i18nConfig,\n request.headers,\n new URL(request.url).hostname,\n vinextConfig.basePath,\n vinextConfig.trailingSlash,\n )\n : { locale: undefined, url, hadPrefix: false, domainLocale: undefined, redirectUrl: undefined };\n const locale = localeInfo.locale;\n const routeUrl = localeInfo.url;\n const currentDefaultLocale = i18nConfig\n ? (localeInfo.domainLocale ? localeInfo.domainLocale.defaultLocale : i18nConfig.defaultLocale)\n : undefined;\n const domainLocales = i18nConfig ? i18nConfig.domains : undefined;\n\n if (localeInfo.redirectUrl) {\n return new Response(null, { status: 307, headers: { Location: localeInfo.redirectUrl } });\n }\n\n const match = matchRoute(routeUrl, pageRoutes);\n if (!match) {\n return new Response(\"<!DOCTYPE html><html><body><h1>404 - Page not found</h1></body></html>\",\n { status: 404, headers: { \"Content-Type\": \"text/html\" } });\n }\n\n\t const { route, params } = match;\n\t const __uCtx = _createUnifiedCtx({\n\t executionContext: _getRequestExecutionContext(),\n\t });\n\t return _runWithUnifiedCtx(__uCtx, async () => {\n\t ensureFetchPatch();\n\t try {\n\t const routePattern = patternToNextFormat(route.pattern);\n\t if (typeof setSSRContext === \"function\") {\n\t setSSRContext({\n\t pathname: routePattern,\n\t query: { ...params, ...parseQuery(routeUrl) },\n\t asPath: routeUrl,\n\t locale: locale,\n locales: i18nConfig ? i18nConfig.locales : undefined,\n defaultLocale: currentDefaultLocale,\n domainLocales: domainLocales,\n });\n }\n\n if (i18nConfig) {\n setI18nContext({\n locale: locale,\n locales: i18nConfig.locales,\n defaultLocale: currentDefaultLocale,\n domainLocales: domainLocales,\n hostname: new URL(request.url).hostname,\n });\n }\n\n const pageModule = route.module;\n const PageComponent = pageModule.default;\n\t if (!PageComponent) {\n\t return new Response(\"Page has no default export\", { status: 500 });\n\t }\n\t // Build font Link header early so it's available for ISR cached responses too.\n\t // Font preloads are module-level state populated at import time and persist across requests.\n\t var _fontLinkHeader = \"\";\n\t var _allFp = [];\n try {\n var _fpGoogle = typeof _getSSRFontPreloadsGoogle === \"function\" ? _getSSRFontPreloadsGoogle() : [];\n var _fpLocal = typeof _getSSRFontPreloadsLocal === \"function\" ? _getSSRFontPreloadsLocal() : [];\n _allFp = _fpGoogle.concat(_fpLocal);\n\t if (_allFp.length > 0) {\n\t _fontLinkHeader = _allFp.map(function(p) { return \"<\" + p.href + \">; rel=preload; as=font; type=\" + p.type + \"; crossorigin\"; }).join(\", \");\n\t }\n\t } catch (e) { /* font preloads not available */ }\n\t const query = parseQuery(routeUrl);\n\t const pageDataResult = await __resolvePagesPageData({\n\t applyRequestContexts() {\n\t if (typeof setSSRContext === \"function\") {\n\t setSSRContext({\n\t pathname: routePattern,\n\t query: { ...params, ...query },\n\t asPath: routeUrl,\n\t locale: locale,\n\t locales: i18nConfig ? i18nConfig.locales : undefined,\n\t defaultLocale: currentDefaultLocale,\n\t domainLocales: domainLocales,\n\t });\n\t }\n\t if (i18nConfig) {\n\t setI18nContext({\n\t locale: locale,\n\t locales: i18nConfig.locales,\n\t defaultLocale: currentDefaultLocale,\n\t domainLocales: domainLocales,\n\t hostname: new URL(request.url).hostname,\n\t });\n\t }\n\t },\n\t buildId,\n\t createGsspReqRes() {\n\t return __createPagesReqRes({ body: undefined, query, request, url: routeUrl });\n\t },\n\t createPageElement(currentPageProps) {\n\t var currentElement = AppComponent\n\t ? React.createElement(AppComponent, { Component: PageComponent, pageProps: currentPageProps })\n\t : React.createElement(PageComponent, currentPageProps);\n\t return wrapWithRouterContext(currentElement);\n\t },\n\t fontLinkHeader: _fontLinkHeader,\n\t i18n: {\n\t locale: locale,\n\t locales: i18nConfig ? i18nConfig.locales : undefined,\n\t defaultLocale: currentDefaultLocale,\n\t domainLocales: domainLocales,\n\t },\n\t isrCacheKey,\n\t isrGet,\n\t isrSet,\n\t pageModule,\n\t params,\n\t query,\n\t renderIsrPassToStringAsync,\n\t route: {\n\t isDynamic: route.isDynamic,\n\t },\n\t routePattern,\n\t routeUrl,\n\t runInFreshUnifiedContext(callback) {\n\t var revalCtx = _createUnifiedCtx({\n\t executionContext: _getRequestExecutionContext(),\n\t });\n\t return _runWithUnifiedCtx(revalCtx, async () => {\n\t ensureFetchPatch();\n\t return callback();\n\t });\n\t },\n\t safeJsonStringify,\n\t sanitizeDestination: sanitizeDestinationLocal,\n\t triggerBackgroundRegeneration,\n\t });\n\t if (pageDataResult.kind === \"response\") {\n\t return pageDataResult.response;\n\t }\n\t let pageProps = pageDataResult.pageProps;\n\t var gsspRes = pageDataResult.gsspRes;\n\t let isrRevalidateSeconds = pageDataResult.isrRevalidateSeconds;\n\n\t const pageModuleIds = route.filePath ? [route.filePath] : [];\n\t const assetTags = collectAssetTags(manifest, pageModuleIds);\n\n return __renderPagesPageResponse({\n assetTags,\n buildId,\n clearSsrContext() {\n if (typeof setSSRContext === \"function\") setSSRContext(null);\n },\n createPageElement(currentPageProps) {\n var currentElement;\n if (AppComponent) {\n currentElement = React.createElement(AppComponent, { Component: PageComponent, pageProps: currentPageProps });\n } else {\n currentElement = React.createElement(PageComponent, currentPageProps);\n }\n return wrapWithRouterContext(currentElement);\n },\n DocumentComponent,\n flushPreloads: typeof flushPreloads === \"function\" ? flushPreloads : undefined,\n fontLinkHeader: _fontLinkHeader,\n fontPreloads: _allFp,\n getFontLinks() {\n try {\n return typeof _getSSRFontLinks === \"function\" ? _getSSRFontLinks() : [];\n } catch (e) {\n return [];\n }\n },\n getFontStyles() {\n try {\n var allFontStyles = [];\n if (typeof _getSSRFontStylesGoogle === \"function\") allFontStyles.push(..._getSSRFontStylesGoogle());\n if (typeof _getSSRFontStylesLocal === \"function\") allFontStyles.push(..._getSSRFontStylesLocal());\n return allFontStyles;\n } catch (e) {\n return [];\n }\n },\n\t getSSRHeadHTML: typeof getSSRHeadHTML === \"function\" ? getSSRHeadHTML : undefined,\n\t gsspRes,\n isrCacheKey,\n isrRevalidateSeconds,\n isrSet,\n i18n: {\n locale: locale,\n locales: i18nConfig ? i18nConfig.locales : undefined,\n defaultLocale: currentDefaultLocale,\n domainLocales: domainLocales,\n },\n pageProps,\n params,\n renderDocumentToString(element) {\n return renderToStringAsync(element);\n },\n renderIsrPassToStringAsync,\n renderToReadableStream(element) {\n return renderToReadableStream(element);\n },\n resetSSRHead: typeof resetSSRHead === \"function\" ? resetSSRHead : undefined,\n\t routePattern,\n routeUrl,\n safeJsonStringify,\n });\n } catch (e) {\n console.error(\"[vinext] SSR error:\", e);\n _reportRequestError(\n e instanceof Error ? e : new Error(String(e)),\n { path: url, method: request.method, headers: Object.fromEntries(request.headers.entries()) },\n { routerKind: \"Pages Router\", routePath: route.pattern, routeType: \"render\" },\n ).catch(() => { /* ignore reporting errors */ });\n return new Response(\"Internal Server Error\", { status: 500 });\n }\n });\n}\n\nexport async function handleApiRoute(request, url) {\n const match = matchRoute(url, apiRoutes);\n return __handlePagesApiRoute({\n match,\n request,\n url,\n reportRequestError(error, routePattern) {\n console.error(\"[vinext] API error:\", error);\n void _reportRequestError(\n error,\n { path: url, method: request.method, headers: Object.fromEntries(request.headers.entries()) },\n { routerKind: \"Pages Router\", routePath: routePattern, routeType: \"route\" },\n );\n },\n });\n}\n\n${middlewareExportCode}\n`;\n}\n"],"mappings":"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"}