vinext 0.0.52 → 0.0.54
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.
- package/README.md +1 -1
- package/dist/build/clean-output.d.ts +14 -0
- package/dist/build/clean-output.js +36 -0
- package/dist/build/clean-output.js.map +1 -0
- package/dist/build/inline-css.d.ts +7 -0
- package/dist/build/inline-css.js +50 -0
- package/dist/build/inline-css.js.map +1 -0
- package/dist/build/prerender.d.ts +6 -2
- package/dist/build/prerender.js +51 -12
- package/dist/build/prerender.js.map +1 -1
- package/dist/build/run-prerender.js +10 -1
- package/dist/build/run-prerender.js.map +1 -1
- package/dist/build/static-export.d.ts +5 -0
- package/dist/build/static-export.js +8 -3
- package/dist/build/static-export.js.map +1 -1
- package/dist/check.js +4 -0
- package/dist/check.js.map +1 -1
- package/dist/cli.js +19 -4
- package/dist/cli.js.map +1 -1
- package/dist/client/instrumentation-client-inject.d.ts +34 -0
- package/dist/client/instrumentation-client-inject.js +57 -0
- package/dist/client/instrumentation-client-inject.js.map +1 -0
- package/dist/client/navigation-runtime.d.ts +16 -2
- package/dist/client/navigation-runtime.js +16 -1
- package/dist/client/navigation-runtime.js.map +1 -1
- package/dist/client/vinext-next-data.d.ts +2 -1
- package/dist/client/vinext-next-data.js.map +1 -1
- package/dist/client/window-next.d.ts +17 -2
- package/dist/client/window-next.js.map +1 -1
- package/dist/cloudflare/tpr.js +1 -1
- package/dist/cloudflare/tpr.js.map +1 -1
- package/dist/config/config-matchers.js +2 -1
- package/dist/config/config-matchers.js.map +1 -1
- package/dist/config/next-config.d.ts +95 -4
- package/dist/config/next-config.js +173 -14
- package/dist/config/next-config.js.map +1 -1
- package/dist/deploy.js +42 -7
- package/dist/deploy.js.map +1 -1
- package/dist/entries/app-browser-entry.d.ts +11 -1
- package/dist/entries/app-browser-entry.js +16 -6
- package/dist/entries/app-browser-entry.js.map +1 -1
- package/dist/entries/app-rsc-entry.d.ts +12 -3
- package/dist/entries/app-rsc-entry.js +41 -8
- package/dist/entries/app-rsc-entry.js.map +1 -1
- package/dist/entries/app-rsc-manifest.d.ts +21 -1
- package/dist/entries/app-rsc-manifest.js +6 -4
- package/dist/entries/app-rsc-manifest.js.map +1 -1
- package/dist/entries/pages-client-entry.d.ts +4 -1
- package/dist/entries/pages-client-entry.js +40 -3
- package/dist/entries/pages-client-entry.js.map +1 -1
- package/dist/entries/pages-server-entry.js +292 -34
- package/dist/entries/pages-server-entry.js.map +1 -1
- package/dist/entries/runtime-entry-module.d.ts +1 -10
- package/dist/entries/runtime-entry-module.js +2 -12
- package/dist/entries/runtime-entry-module.js.map +1 -1
- package/dist/index.js +91 -10
- package/dist/index.js.map +1 -1
- package/dist/plugins/fonts.js +25 -2
- package/dist/plugins/fonts.js.map +1 -1
- package/dist/plugins/remove-console.d.ts +16 -0
- package/dist/plugins/remove-console.js +176 -0
- package/dist/plugins/remove-console.js.map +1 -0
- package/dist/routing/app-route-graph.d.ts +24 -1
- package/dist/routing/app-route-graph.js +52 -4
- package/dist/routing/app-route-graph.js.map +1 -1
- package/dist/routing/app-router.d.ts +2 -2
- package/dist/routing/app-router.js +2 -2
- package/dist/routing/app-router.js.map +1 -1
- package/dist/routing/file-matcher.d.ts +21 -1
- package/dist/routing/file-matcher.js +39 -1
- package/dist/routing/file-matcher.js.map +1 -1
- package/dist/routing/pages-router.d.ts +1 -1
- package/dist/routing/pages-router.js +10 -3
- package/dist/routing/pages-router.js.map +1 -1
- package/dist/routing/route-trie.js +13 -18
- package/dist/routing/route-trie.js.map +1 -1
- package/dist/routing/utils.d.ts +11 -1
- package/dist/routing/utils.js +15 -1
- package/dist/routing/utils.js.map +1 -1
- package/dist/server/api-handler.js +19 -10
- package/dist/server/api-handler.js.map +1 -1
- package/dist/server/app-browser-action-result.d.ts +16 -1
- package/dist/server/app-browser-action-result.js +15 -1
- package/dist/server/app-browser-action-result.js.map +1 -1
- package/dist/server/app-browser-entry.js +47 -28
- package/dist/server/app-browser-entry.js.map +1 -1
- package/dist/server/app-browser-navigation-controller.d.ts +2 -0
- package/dist/server/app-browser-navigation-controller.js +4 -0
- package/dist/server/app-browser-navigation-controller.js.map +1 -1
- package/dist/server/app-elements-wire.d.ts +13 -4
- package/dist/server/app-elements-wire.js +10 -1
- package/dist/server/app-elements-wire.js.map +1 -1
- package/dist/server/app-elements.d.ts +2 -2
- package/dist/server/app-elements.js +2 -2
- package/dist/server/app-elements.js.map +1 -1
- package/dist/server/app-fallback-renderer.d.ts +27 -8
- package/dist/server/app-fallback-renderer.js +19 -8
- package/dist/server/app-fallback-renderer.js.map +1 -1
- package/dist/server/app-history-state.js +6 -2
- package/dist/server/app-history-state.js.map +1 -1
- package/dist/server/app-inline-css-client.d.ts +7 -0
- package/dist/server/app-inline-css-client.js +37 -0
- package/dist/server/app-inline-css-client.js.map +1 -0
- package/dist/server/app-interception-context-header.d.ts +33 -0
- package/dist/server/app-interception-context-header.js +44 -0
- package/dist/server/app-interception-context-header.js.map +1 -0
- package/dist/server/app-mounted-slots-header.d.ts +19 -0
- package/dist/server/app-mounted-slots-header.js +40 -1
- package/dist/server/app-mounted-slots-header.js.map +1 -1
- package/dist/server/app-optimistic-routing.js +26 -18
- package/dist/server/app-optimistic-routing.js.map +1 -1
- package/dist/server/app-page-boundary-render.d.ts +1 -0
- package/dist/server/app-page-boundary-render.js +2 -0
- package/dist/server/app-page-boundary-render.js.map +1 -1
- package/dist/server/app-page-boundary.d.ts +22 -1
- package/dist/server/app-page-boundary.js +30 -3
- package/dist/server/app-page-boundary.js.map +1 -1
- package/dist/server/app-page-cache.d.ts +9 -3
- package/dist/server/app-page-cache.js +14 -8
- package/dist/server/app-page-cache.js.map +1 -1
- package/dist/server/app-page-dispatch.d.ts +13 -1
- package/dist/server/app-page-dispatch.js +136 -82
- package/dist/server/app-page-dispatch.js.map +1 -1
- package/dist/server/app-page-element-builder.d.ts +2 -1
- package/dist/server/app-page-element-builder.js +17 -30
- package/dist/server/app-page-element-builder.js.map +1 -1
- package/dist/server/app-page-execution.d.ts +1 -0
- package/dist/server/app-page-execution.js +2 -0
- package/dist/server/app-page-execution.js.map +1 -1
- package/dist/server/app-page-head.d.ts +1 -0
- package/dist/server/app-page-head.js +8 -0
- package/dist/server/app-page-head.js.map +1 -1
- package/dist/server/app-page-render-identity.d.ts +22 -0
- package/dist/server/app-page-render-identity.js +42 -0
- package/dist/server/app-page-render-identity.js.map +1 -0
- package/dist/server/app-page-render-observation.js +1 -1
- package/dist/server/app-page-render.d.ts +9 -1
- package/dist/server/app-page-render.js +8 -2
- package/dist/server/app-page-render.js.map +1 -1
- package/dist/server/app-page-request.d.ts +6 -3
- package/dist/server/app-page-request.js +5 -2
- package/dist/server/app-page-request.js.map +1 -1
- package/dist/server/app-page-response.d.ts +11 -1
- package/dist/server/app-page-response.js +16 -4
- package/dist/server/app-page-response.js.map +1 -1
- package/dist/server/app-page-route-wiring.d.ts +16 -0
- package/dist/server/app-page-route-wiring.js +25 -10
- package/dist/server/app-page-route-wiring.js.map +1 -1
- package/dist/server/app-page-stream.d.ts +12 -0
- package/dist/server/app-page-stream.js +3 -0
- package/dist/server/app-page-stream.js.map +1 -1
- package/dist/server/app-route-handler-dispatch.d.ts +1 -0
- package/dist/server/app-route-handler-dispatch.js +3 -0
- package/dist/server/app-route-handler-dispatch.js.map +1 -1
- package/dist/server/app-route-handler-execution.d.ts +1 -0
- package/dist/server/app-route-handler-execution.js +1 -0
- package/dist/server/app-route-handler-execution.js.map +1 -1
- package/dist/server/app-route-handler-response.js +38 -6
- package/dist/server/app-route-handler-response.js.map +1 -1
- package/dist/server/app-rsc-handler.d.ts +16 -3
- package/dist/server/app-rsc-handler.js +60 -11
- package/dist/server/app-rsc-handler.js.map +1 -1
- package/dist/server/app-rsc-request-normalization.d.ts +2 -1
- package/dist/server/app-rsc-request-normalization.js +6 -4
- package/dist/server/app-rsc-request-normalization.js.map +1 -1
- package/dist/server/app-segment-config.d.ts +4 -1
- package/dist/server/app-segment-config.js +6 -1
- package/dist/server/app-segment-config.js.map +1 -1
- package/dist/server/app-server-action-execution.d.ts +22 -3
- package/dist/server/app-server-action-execution.js +46 -7
- package/dist/server/app-server-action-execution.js.map +1 -1
- package/dist/server/app-ssr-entry.d.ts +6 -0
- package/dist/server/app-ssr-entry.js +57 -6
- package/dist/server/app-ssr-entry.js.map +1 -1
- package/dist/server/app-ssr-error-meta.js +3 -3
- package/dist/server/app-ssr-error-meta.js.map +1 -1
- package/dist/server/app-ssr-stream.d.ts +25 -1
- package/dist/server/app-ssr-stream.js +237 -19
- package/dist/server/app-ssr-stream.js.map +1 -1
- package/dist/server/app-static-generation.d.ts +1 -0
- package/dist/server/app-static-generation.js +2 -1
- package/dist/server/app-static-generation.js.map +1 -1
- package/dist/server/client-trace-metadata.d.ts +31 -0
- package/dist/server/client-trace-metadata.js +83 -0
- package/dist/server/client-trace-metadata.js.map +1 -0
- package/dist/server/cookie-utils.d.ts +13 -0
- package/dist/server/cookie-utils.js +20 -0
- package/dist/server/cookie-utils.js.map +1 -0
- package/dist/server/default-not-found-module.d.ts +20 -0
- package/dist/server/default-not-found-module.js +20 -0
- package/dist/server/default-not-found-module.js.map +1 -0
- package/dist/server/dev-server.d.ts +8 -1
- package/dist/server/dev-server.js +56 -11
- package/dist/server/dev-server.js.map +1 -1
- package/dist/server/headers.d.ts +5 -1
- package/dist/server/headers.js +5 -1
- package/dist/server/headers.js.map +1 -1
- package/dist/server/html.d.ts +2 -1
- package/dist/server/html.js +6 -1
- package/dist/server/html.js.map +1 -1
- package/dist/server/image-optimization.d.ts +13 -4
- package/dist/server/image-optimization.js +15 -4
- package/dist/server/image-optimization.js.map +1 -1
- package/dist/server/isr-cache.d.ts +7 -5
- package/dist/server/isr-cache.js +17 -6
- package/dist/server/isr-cache.js.map +1 -1
- package/dist/server/middleware-runtime.js +1 -2
- package/dist/server/middleware-runtime.js.map +1 -1
- package/dist/server/middleware.js +1 -1
- package/dist/server/middleware.js.map +1 -1
- package/dist/server/pages-api-route.d.ts +18 -0
- package/dist/server/pages-api-route.js +3 -1
- package/dist/server/pages-api-route.js.map +1 -1
- package/dist/server/pages-body-parser-config.d.ts +60 -0
- package/dist/server/pages-body-parser-config.js +79 -0
- package/dist/server/pages-body-parser-config.js.map +1 -0
- package/dist/server/pages-data-route.js +1 -0
- package/dist/server/pages-data-route.js.map +1 -1
- package/dist/server/pages-default-404.d.ts +31 -0
- package/dist/server/pages-default-404.js +40 -0
- package/dist/server/pages-default-404.js.map +1 -0
- package/dist/server/pages-document-initial-props.d.ts +7 -0
- package/dist/server/pages-document-initial-props.js +14 -0
- package/dist/server/pages-document-initial-props.js.map +1 -0
- package/dist/server/pages-node-compat.d.ts +10 -0
- package/dist/server/pages-node-compat.js +12 -1
- package/dist/server/pages-node-compat.js.map +1 -1
- package/dist/server/pages-page-data.d.ts +40 -0
- package/dist/server/pages-page-data.js +19 -14
- package/dist/server/pages-page-data.js.map +1 -1
- package/dist/server/pages-page-method.d.ts +48 -0
- package/dist/server/pages-page-method.js +19 -0
- package/dist/server/pages-page-method.js.map +1 -0
- package/dist/server/pages-page-response.d.ts +8 -0
- package/dist/server/pages-page-response.js +21 -11
- package/dist/server/pages-page-response.js.map +1 -1
- package/dist/server/pages-serializable-props.d.ts +25 -0
- package/dist/server/pages-serializable-props.js +69 -0
- package/dist/server/pages-serializable-props.js.map +1 -0
- package/dist/server/prerender-route-params.d.ts +14 -0
- package/dist/server/prerender-route-params.js +94 -0
- package/dist/server/prerender-route-params.js.map +1 -0
- package/dist/server/prod-server.d.ts +3 -23
- package/dist/server/prod-server.js +43 -57
- package/dist/server/prod-server.js.map +1 -1
- package/dist/server/proxy-trust.d.ts +41 -0
- package/dist/server/proxy-trust.js +70 -0
- package/dist/server/proxy-trust.js.map +1 -0
- package/dist/server/request-pipeline.d.ts +3 -3
- package/dist/server/request-pipeline.js +5 -4
- package/dist/server/request-pipeline.js.map +1 -1
- package/dist/server/seed-cache.js +12 -6
- package/dist/server/seed-cache.js.map +1 -1
- package/dist/server/server-action-not-found.js +3 -2
- package/dist/server/server-action-not-found.js.map +1 -1
- package/dist/server/static-file-cache.js +2 -1
- package/dist/server/static-file-cache.js.map +1 -1
- package/dist/server/streaming-metadata.d.ts +5 -0
- package/dist/server/streaming-metadata.js +10 -0
- package/dist/server/streaming-metadata.js.map +1 -0
- package/dist/shims/app-router-scroll-state.d.ts +14 -0
- package/dist/shims/app-router-scroll-state.js +51 -0
- package/dist/shims/app-router-scroll-state.js.map +1 -0
- package/dist/shims/app-router-scroll.d.ts +28 -0
- package/dist/shims/app-router-scroll.js +115 -0
- package/dist/shims/app-router-scroll.js.map +1 -0
- package/dist/shims/before-interactive-context.d.ts +30 -0
- package/dist/shims/before-interactive-context.js +10 -0
- package/dist/shims/before-interactive-context.js.map +1 -0
- package/dist/shims/cache-runtime.d.ts +1 -1
- package/dist/shims/cache-runtime.js +14 -1
- package/dist/shims/cache-runtime.js.map +1 -1
- package/dist/shims/cache.d.ts +6 -0
- package/dist/shims/cache.js +7 -0
- package/dist/shims/cache.js.map +1 -1
- package/dist/shims/default-not-found.d.ts +12 -0
- package/dist/shims/default-not-found.js +61 -0
- package/dist/shims/default-not-found.js.map +1 -0
- package/dist/shims/error.js +3 -0
- package/dist/shims/error.js.map +1 -1
- package/dist/shims/font-local.d.ts +5 -0
- package/dist/shims/font-local.js +6 -2
- package/dist/shims/font-local.js.map +1 -1
- package/dist/shims/head.js +4 -4
- package/dist/shims/head.js.map +1 -1
- package/dist/shims/headers.d.ts +13 -2
- package/dist/shims/headers.js +73 -22
- package/dist/shims/headers.js.map +1 -1
- package/dist/shims/image.d.ts +1 -1
- package/dist/shims/image.js +4 -4
- package/dist/shims/image.js.map +1 -1
- package/dist/shims/internal/app-route-detection.d.ts +37 -0
- package/dist/shims/internal/app-route-detection.js +69 -0
- package/dist/shims/internal/app-route-detection.js.map +1 -0
- package/dist/shims/internal/pages-data-target.d.ts +58 -0
- package/dist/shims/internal/pages-data-target.js +91 -0
- package/dist/shims/internal/pages-data-target.js.map +1 -0
- package/dist/shims/internal/pages-data-url.d.ts +42 -0
- package/dist/shims/internal/pages-data-url.js +73 -0
- package/dist/shims/internal/pages-data-url.js.map +1 -0
- package/dist/shims/link.d.ts +18 -2
- package/dist/shims/link.js +129 -15
- package/dist/shims/link.js.map +1 -1
- package/dist/shims/metadata.d.ts +9 -7
- package/dist/shims/metadata.js +70 -7
- package/dist/shims/metadata.js.map +1 -1
- package/dist/shims/navigation.d.ts +1 -2
- package/dist/shims/navigation.js +94 -20
- package/dist/shims/navigation.js.map +1 -1
- package/dist/shims/router.d.ts +5 -0
- package/dist/shims/router.js +389 -80
- package/dist/shims/router.js.map +1 -1
- package/dist/shims/script.d.ts +11 -1
- package/dist/shims/script.js +158 -15
- package/dist/shims/script.js.map +1 -1
- package/dist/shims/server.js +1 -0
- package/dist/shims/server.js.map +1 -1
- package/dist/shims/url-utils.d.ts +2 -1
- package/dist/shims/url-utils.js +15 -4
- package/dist/shims/url-utils.js.map +1 -1
- package/dist/utils/html-limited-bots.d.ts +5 -0
- package/dist/utils/html-limited-bots.js +15 -0
- package/dist/utils/html-limited-bots.js.map +1 -0
- package/dist/utils/path.d.ts +13 -0
- package/dist/utils/path.js +16 -0
- package/dist/utils/path.js.map +1 -0
- package/dist/utils/query.d.ts +6 -0
- package/dist/utils/query.js +10 -1
- package/dist/utils/query.js.map +1 -1
- package/package.json +1 -1
package/dist/shims/script.js
CHANGED
|
@@ -1,6 +1,8 @@
|
|
|
1
1
|
"use client";
|
|
2
2
|
import { escapeInlineContent } from "./head.js";
|
|
3
|
+
import { hasAppNavigationRuntimeBootstrap } from "../client/navigation-runtime.js";
|
|
3
4
|
import { useScriptNonce } from "./script-nonce-context.js";
|
|
5
|
+
import { useBeforeInteractiveRegister } from "./before-interactive-context.js";
|
|
4
6
|
import React, { useEffect, useRef } from "react";
|
|
5
7
|
import * as ReactDOM from "react-dom";
|
|
6
8
|
//#region src/shims/script.tsx
|
|
@@ -21,14 +23,56 @@ const loadingScripts = /* @__PURE__ */ new Map();
|
|
|
21
23
|
function getClientAutoNonce() {
|
|
22
24
|
if (typeof document === "undefined") return void 0;
|
|
23
25
|
const existingNonceElement = document.querySelector("[nonce]");
|
|
24
|
-
if (!
|
|
25
|
-
return existingNonceElement.nonce || existingNonceElement.getAttribute("nonce") || void 0;
|
|
26
|
+
if (!existingNonceElement) return void 0;
|
|
27
|
+
if (typeof HTMLElement !== "undefined" && existingNonceElement instanceof HTMLElement) return existingNonceElement.nonce || existingNonceElement.getAttribute("nonce") || void 0;
|
|
28
|
+
return existingNonceElement.getAttribute("nonce") || void 0;
|
|
26
29
|
}
|
|
27
30
|
function resolveScriptNonce(explicitNonce, contextualNonce) {
|
|
28
31
|
if (typeof explicitNonce === "string" && explicitNonce.length > 0) return explicitNonce;
|
|
29
|
-
if (typeof
|
|
32
|
+
if (typeof contextualNonce === "string" && contextualNonce.length > 0) return contextualNonce;
|
|
33
|
+
if (typeof window === "undefined") return;
|
|
30
34
|
return getClientAutoNonce();
|
|
31
35
|
}
|
|
36
|
+
/**
|
|
37
|
+
* Insert `<link rel="stylesheet">` tags into `document.head` for each entry
|
|
38
|
+
* in `stylesheets`. Used by the imperative client-side load path
|
|
39
|
+
* (`handleClientScriptLoad`) when `ReactDOM.preinit` is not available
|
|
40
|
+
* (e.g. pre-Float React or hosts that strip it). Mirrors Next.js's
|
|
41
|
+
* `insertStylesheets` Pages-Router fallback at
|
|
42
|
+
* `.nextjs-ref/packages/next/src/client/script.tsx:48-59`.
|
|
43
|
+
*
|
|
44
|
+
* The `ReactDOM.preinit` path is preferred where available — it dedupes
|
|
45
|
+
* across mounts and respects React Float's hoisting order. This DOM
|
|
46
|
+
* fallback is best-effort: no dedupe, no ordering guarantee.
|
|
47
|
+
*/
|
|
48
|
+
function insertClientStylesheets(stylesheets) {
|
|
49
|
+
if (!stylesheets || stylesheets.length === 0) return;
|
|
50
|
+
if (typeof document === "undefined") return;
|
|
51
|
+
if (typeof ReactDOM.preinit === "function") {
|
|
52
|
+
for (const href of stylesheets) ReactDOM.preinit(href, { as: "style" });
|
|
53
|
+
return;
|
|
54
|
+
}
|
|
55
|
+
const head = document.head;
|
|
56
|
+
if (!head) return;
|
|
57
|
+
for (const href of stylesheets) {
|
|
58
|
+
const link = document.createElement("link");
|
|
59
|
+
link.rel = "stylesheet";
|
|
60
|
+
link.type = "text/css";
|
|
61
|
+
link.href = href;
|
|
62
|
+
head.appendChild(link);
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
/**
|
|
66
|
+
* Emit `<link rel="stylesheet">` tags during SSR for each entry in
|
|
67
|
+
* `stylesheets` via `ReactDOM.preinit`. React Float hoists these into
|
|
68
|
+
* `<head>` in the streamed HTML. Mirrors the App-Router branch of
|
|
69
|
+
* Next.js's Script component at `.nextjs-ref/packages/next/src/client/script.tsx:309-313`.
|
|
70
|
+
*/
|
|
71
|
+
function preinitStylesheetsForSSR(stylesheets) {
|
|
72
|
+
if (!stylesheets || stylesheets.length === 0) return;
|
|
73
|
+
if (typeof ReactDOM.preinit !== "function") return;
|
|
74
|
+
for (const href of stylesheets) ReactDOM.preinit(href, { as: "style" });
|
|
75
|
+
}
|
|
32
76
|
function buildBeforeInteractiveScriptProps(options) {
|
|
33
77
|
const scriptProps = { ...options.rest };
|
|
34
78
|
if (options.src) scriptProps.src = options.src;
|
|
@@ -37,6 +81,81 @@ function buildBeforeInteractiveScriptProps(options) {
|
|
|
37
81
|
if (options.dangerouslySetInnerHTML) scriptProps.dangerouslySetInnerHTML = { __html: escapeInlineContent(options.dangerouslySetInnerHTML.__html, "script") };
|
|
38
82
|
return scriptProps;
|
|
39
83
|
}
|
|
84
|
+
/**
|
|
85
|
+
* Extract the inline script content for a `beforeInteractive` Script element
|
|
86
|
+
* with no `src`. Returns `null` when the element has neither a string-shaped
|
|
87
|
+
* `children` value nor a valid `dangerouslySetInnerHTML.__html` payload — in
|
|
88
|
+
* that case the caller should fall through to React's regular rendering path.
|
|
89
|
+
*
|
|
90
|
+
* The returned string is the raw author-supplied JavaScript content. Callers
|
|
91
|
+
* are responsible for passing it through `escapeInlineContent(..., "script")`
|
|
92
|
+
* before emitting it inside a `<script>` tag (we keep that escape adjacent
|
|
93
|
+
* to the emit point so the rule is obvious at the boundary).
|
|
94
|
+
*/
|
|
95
|
+
function extractBeforeInteractiveInlineContent(children, dangerouslySetInnerHTML) {
|
|
96
|
+
if (dangerouslySetInnerHTML && typeof dangerouslySetInnerHTML.__html === "string" && dangerouslySetInnerHTML.__html.length > 0) return dangerouslySetInnerHTML.__html;
|
|
97
|
+
if (typeof children === "string" && children.length > 0) return children;
|
|
98
|
+
if (Array.isArray(children) && children.every((c) => typeof c === "string")) {
|
|
99
|
+
const joined = children.join("");
|
|
100
|
+
return joined.length > 0 ? joined : null;
|
|
101
|
+
}
|
|
102
|
+
return null;
|
|
103
|
+
}
|
|
104
|
+
/**
|
|
105
|
+
* Map of React DOM prop names to their HTML attribute equivalents. Mirrors
|
|
106
|
+
* Next.js's `set-attributes-from-props.ts`:
|
|
107
|
+
* .nextjs-ref/packages/next/src/client/set-attributes-from-props.ts
|
|
108
|
+
* HTML parses attribute names case-insensitively, so without this translation
|
|
109
|
+
* `className="foo"` round-trips as `classname="foo"` and CSS selectors on
|
|
110
|
+
* `.foo` never match. Same hazard for `htmlFor`/`for`, `httpEquiv`/`http-equiv`,
|
|
111
|
+
* `acceptCharset`/`accept-charset`.
|
|
112
|
+
*/
|
|
113
|
+
const REACT_TO_HTML_ATTR = {
|
|
114
|
+
acceptCharset: "accept-charset",
|
|
115
|
+
className: "class",
|
|
116
|
+
htmlFor: "for",
|
|
117
|
+
httpEquiv: "http-equiv"
|
|
118
|
+
};
|
|
119
|
+
/**
|
|
120
|
+
* Convert the residual `<Script>` props into a plain string-attributes record
|
|
121
|
+
* for emission inside a hoisted `<script>` tag. Drops React-only props
|
|
122
|
+
* (event handlers, children, etc.) and reserved keys already handled by the
|
|
123
|
+
* pre-head-injection emitter (id, nonce). Skips `undefined`/`null` so they
|
|
124
|
+
* round-trip as "attribute absent" rather than `attr="undefined"`.
|
|
125
|
+
*
|
|
126
|
+
* React DOM prop names (className, htmlFor, etc.) are translated to their
|
|
127
|
+
* HTML attribute names so the output parses correctly — see comment on
|
|
128
|
+
* `REACT_TO_HTML_ATTR`.
|
|
129
|
+
*/
|
|
130
|
+
function collectBeforeInteractiveAttributes(rest) {
|
|
131
|
+
const RESERVED = new Set([
|
|
132
|
+
"id",
|
|
133
|
+
"nonce",
|
|
134
|
+
"src",
|
|
135
|
+
"children",
|
|
136
|
+
"strategy",
|
|
137
|
+
"dangerouslySetInnerHTML",
|
|
138
|
+
"onLoad",
|
|
139
|
+
"onReady",
|
|
140
|
+
"onError",
|
|
141
|
+
"stylesheets"
|
|
142
|
+
]);
|
|
143
|
+
const out = {};
|
|
144
|
+
for (const [key, value] of Object.entries(rest)) {
|
|
145
|
+
if (RESERVED.has(key)) continue;
|
|
146
|
+
if (value === void 0 || value === null || value === false) continue;
|
|
147
|
+
const attrName = REACT_TO_HTML_ATTR[key] ?? key;
|
|
148
|
+
if (typeof value === "boolean") {
|
|
149
|
+
out[attrName] = true;
|
|
150
|
+
continue;
|
|
151
|
+
}
|
|
152
|
+
if (typeof value === "string" || typeof value === "number") {
|
|
153
|
+
out[attrName] = String(value);
|
|
154
|
+
continue;
|
|
155
|
+
}
|
|
156
|
+
}
|
|
157
|
+
return out;
|
|
158
|
+
}
|
|
40
159
|
function setBooleanScriptAttribute(el, attr, value) {
|
|
41
160
|
const enabled = value !== false && value !== "false" && Boolean(value);
|
|
42
161
|
switch (attr) {
|
|
@@ -69,8 +188,9 @@ function setScriptAttributes(el, rest) {
|
|
|
69
188
|
}
|
|
70
189
|
}
|
|
71
190
|
function loadClientScript(props, options) {
|
|
72
|
-
const { src, id, onLoad, onReady, onError, strategy = "afterInteractive", children, dangerouslySetInnerHTML, ...rest } = props;
|
|
191
|
+
const { src, id, onLoad, onReady, onError, strategy = "afterInteractive", children, dangerouslySetInnerHTML, stylesheets, ...rest } = props;
|
|
73
192
|
if (typeof window === "undefined") return;
|
|
193
|
+
insertClientStylesheets(stylesheets);
|
|
74
194
|
const key = id ?? src ?? "";
|
|
75
195
|
if (key && loadedScripts.has(key)) {
|
|
76
196
|
if (options.fireReadyWhenAlreadyLoaded) onReady?.();
|
|
@@ -137,16 +257,21 @@ function initScriptLoader(scripts) {
|
|
|
137
257
|
for (const script of scripts) handleClientScriptLoad(script);
|
|
138
258
|
}
|
|
139
259
|
function Script(props) {
|
|
140
|
-
const { src, id, strategy = "afterInteractive", onLoad, onReady, onError, children, dangerouslySetInnerHTML, ...rest } = props;
|
|
260
|
+
const { src, id, strategy = "afterInteractive", onLoad, onReady, onError, children, dangerouslySetInnerHTML, stylesheets, ...rest } = props;
|
|
141
261
|
const hasMounted = useRef(false);
|
|
142
262
|
const key = id ?? src ?? "";
|
|
143
263
|
const contextualNonce = useScriptNonce();
|
|
144
264
|
const resolvedNonce = resolveScriptNonce(rest.nonce, contextualNonce);
|
|
265
|
+
const registerBeforeInteractive = useBeforeInteractiveRegister();
|
|
145
266
|
useEffect(() => {
|
|
146
267
|
if (hasMounted.current) return;
|
|
147
268
|
hasMounted.current = true;
|
|
148
|
-
if (strategy === "beforeInteractive")
|
|
269
|
+
if (strategy === "beforeInteractive") {
|
|
270
|
+
insertClientStylesheets(stylesheets);
|
|
271
|
+
return;
|
|
272
|
+
}
|
|
149
273
|
if (key && loadedScripts.has(key)) {
|
|
274
|
+
insertClientStylesheets(stylesheets);
|
|
150
275
|
onReady?.();
|
|
151
276
|
return;
|
|
152
277
|
}
|
|
@@ -164,6 +289,7 @@ function Script(props) {
|
|
|
164
289
|
onError,
|
|
165
290
|
children,
|
|
166
291
|
dangerouslySetInnerHTML,
|
|
292
|
+
stylesheets,
|
|
167
293
|
...rest
|
|
168
294
|
}, {
|
|
169
295
|
resolvedNonce,
|
|
@@ -186,11 +312,13 @@ function Script(props) {
|
|
|
186
312
|
onError,
|
|
187
313
|
children,
|
|
188
314
|
dangerouslySetInnerHTML,
|
|
315
|
+
stylesheets,
|
|
189
316
|
key,
|
|
190
317
|
resolvedNonce,
|
|
191
318
|
rest
|
|
192
319
|
]);
|
|
193
320
|
if (typeof window === "undefined") {
|
|
321
|
+
preinitStylesheetsForSSR(stylesheets);
|
|
194
322
|
if (src && typeof ReactDOM.preload === "function" && (strategy === "afterInteractive" || strategy === "beforeInteractive")) {
|
|
195
323
|
const integrity = typeof rest.integrity === "string" ? rest.integrity : void 0;
|
|
196
324
|
const preloadOptions = {
|
|
@@ -201,22 +329,37 @@ function Script(props) {
|
|
|
201
329
|
if (integrity !== void 0) preloadOptions.integrity = integrity;
|
|
202
330
|
ReactDOM.preload(src, preloadOptions);
|
|
203
331
|
}
|
|
204
|
-
if (strategy === "beforeInteractive")
|
|
332
|
+
if (strategy === "beforeInteractive") {
|
|
333
|
+
const inlineContent = src ? null : extractBeforeInteractiveInlineContent(children, dangerouslySetInnerHTML);
|
|
334
|
+
if (inlineContent !== null && registerBeforeInteractive) {
|
|
335
|
+
registerBeforeInteractive({
|
|
336
|
+
id,
|
|
337
|
+
innerHTML: escapeInlineContent(inlineContent, "script"),
|
|
338
|
+
nonce: resolvedNonce,
|
|
339
|
+
attributes: collectBeforeInteractiveAttributes(rest)
|
|
340
|
+
});
|
|
341
|
+
return null;
|
|
342
|
+
}
|
|
343
|
+
return React.createElement("script", buildBeforeInteractiveScriptProps({
|
|
344
|
+
src,
|
|
345
|
+
id,
|
|
346
|
+
rest,
|
|
347
|
+
resolvedNonce,
|
|
348
|
+
dangerouslySetInnerHTML
|
|
349
|
+
}), children);
|
|
350
|
+
}
|
|
351
|
+
return null;
|
|
352
|
+
}
|
|
353
|
+
if (strategy === "beforeInteractive") {
|
|
354
|
+
if ((src ? null : extractBeforeInteractiveInlineContent(children, dangerouslySetInnerHTML)) !== null && hasAppNavigationRuntimeBootstrap()) return null;
|
|
355
|
+
return React.createElement("script", buildBeforeInteractiveScriptProps({
|
|
205
356
|
src,
|
|
206
357
|
id,
|
|
207
358
|
rest,
|
|
208
359
|
resolvedNonce,
|
|
209
360
|
dangerouslySetInnerHTML
|
|
210
361
|
}), children);
|
|
211
|
-
return null;
|
|
212
362
|
}
|
|
213
|
-
if (strategy === "beforeInteractive") return React.createElement("script", buildBeforeInteractiveScriptProps({
|
|
214
|
-
src,
|
|
215
|
-
id,
|
|
216
|
-
rest,
|
|
217
|
-
resolvedNonce,
|
|
218
|
-
dangerouslySetInnerHTML
|
|
219
|
-
}), children);
|
|
220
363
|
return null;
|
|
221
364
|
}
|
|
222
365
|
//#endregion
|
package/dist/shims/script.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"script.js","names":[],"sources":["../../src/shims/script.tsx"],"sourcesContent":["\"use client\";\n\n/**\n * next/script shim\n *\n * Provides the <Script> component for loading third-party scripts with\n * configurable loading strategies.\n *\n * Strategies:\n * - \"beforeInteractive\": rendered as a <script> tag in SSR output\n * - \"afterInteractive\" (default): loaded client-side after hydration\n * - \"lazyOnload\": deferred until window.load + requestIdleCallback\n * - \"worker\": sets type=\"text/partytown\" (requires Partytown setup)\n */\nimport React, { useEffect, useRef } from \"react\";\nimport * as ReactDOM from \"react-dom\";\nimport { escapeInlineContent } from \"./head.js\";\nimport { useScriptNonce } from \"./script-nonce-context.js\";\n\nexport type ScriptProps = {\n /** Script source URL */\n src?: string;\n /** Loading strategy. Default: \"afterInteractive\" */\n strategy?: \"beforeInteractive\" | \"afterInteractive\" | \"lazyOnload\" | \"worker\";\n /** Unique identifier for the script */\n id?: string;\n /** Called when the script has loaded */\n onLoad?: (e: Event) => void;\n /** Called when the script is ready (after load, and on every re-render if already loaded) */\n onReady?: () => void;\n /** Called on script load error */\n onError?: (e: Event) => void;\n /** Inline script content */\n children?: React.ReactNode;\n /** Dangerous inner HTML */\n dangerouslySetInnerHTML?: { __html: string };\n /** Script type attribute */\n type?: string;\n /** Async attribute */\n async?: boolean;\n /** Defer attribute */\n defer?: boolean;\n /** Crossorigin attribute */\n crossOrigin?: string;\n /** Nonce for CSP */\n nonce?: string;\n /** Integrity hash */\n integrity?: string;\n /** Additional attributes */\n [key: string]: unknown;\n};\n\n// Track scripts that have already been loaded, plus remote scripts currently\n// loading, to avoid duplicate DOM insertion when same-src components mount\n// before the first load event fires.\nconst loadedScripts = new Set<string>();\nconst loadingScripts = new Map<string, Promise<Event>>();\n\nfunction getClientAutoNonce(): string | undefined {\n if (typeof document === \"undefined\") return undefined;\n\n const existingNonceElement = document.querySelector(\"[nonce]\");\n if (!(existingNonceElement instanceof HTMLElement)) {\n return undefined;\n }\n\n return existingNonceElement.nonce || existingNonceElement.getAttribute(\"nonce\") || undefined;\n}\n\nfunction resolveScriptNonce(explicitNonce: unknown, contextualNonce?: string): string | undefined {\n if (typeof explicitNonce === \"string\" && explicitNonce.length > 0) {\n return explicitNonce;\n }\n\n if (typeof window === \"undefined\") {\n return contextualNonce;\n }\n\n return getClientAutoNonce();\n}\n\nfunction buildBeforeInteractiveScriptProps(options: {\n src?: string;\n id?: string;\n rest: Record<string, unknown>;\n resolvedNonce?: string;\n dangerouslySetInnerHTML?: { __html: string };\n}): Record<string, unknown> {\n const scriptProps: Record<string, unknown> = { ...options.rest };\n if (options.src) scriptProps.src = options.src;\n if (options.id) scriptProps.id = options.id;\n if (options.resolvedNonce) {\n scriptProps.nonce = options.resolvedNonce;\n }\n if (options.dangerouslySetInnerHTML) {\n scriptProps.dangerouslySetInnerHTML = {\n __html: escapeInlineContent(options.dangerouslySetInnerHTML.__html, \"script\"),\n };\n }\n return scriptProps;\n}\n\nfunction setBooleanScriptAttribute(el: HTMLScriptElement, attr: string, value: unknown): boolean {\n const enabled = value !== false && value !== \"false\" && Boolean(value);\n\n switch (attr) {\n case \"async\":\n el.async = enabled;\n break;\n case \"defer\":\n el.defer = enabled;\n break;\n case \"noModule\":\n case \"nomodule\":\n el.noModule = enabled;\n break;\n default:\n return false;\n }\n\n if (!enabled) {\n // Dynamic script elements start in the browser's force-async state.\n // Setting and removing the attribute mirrors Next.js and clears that state.\n el.setAttribute(attr, \"\");\n el.removeAttribute(attr);\n }\n\n return true;\n}\n\nfunction setScriptAttributes(el: HTMLScriptElement, rest: Record<string, unknown>): void {\n for (const [attr, value] of Object.entries(rest)) {\n if (attr === \"dangerouslySetInnerHTML\") continue;\n if (value === undefined) continue;\n if (setBooleanScriptAttribute(el, attr, value)) continue;\n if (attr === \"className\" && typeof value === \"string\") {\n el.setAttribute(\"class\", value);\n } else if (typeof value === \"string\") {\n el.setAttribute(attr, value);\n } else if (typeof value === \"boolean\" && value) {\n el.setAttribute(attr, \"\");\n }\n }\n}\n\nfunction loadClientScript(\n props: ScriptProps,\n options: {\n resolvedNonce?: string;\n fireReadyWhenAlreadyLoaded: boolean;\n },\n): void {\n const {\n src,\n id,\n onLoad,\n onReady,\n onError,\n strategy = \"afterInteractive\",\n children,\n dangerouslySetInnerHTML,\n ...rest\n } = props;\n if (typeof window === \"undefined\") return;\n\n const key = id ?? src ?? \"\";\n if (key && loadedScripts.has(key)) {\n if (options.fireReadyWhenAlreadyLoaded) {\n onReady?.();\n }\n return;\n }\n\n if (src) {\n const existingLoad = loadingScripts.get(src);\n if (existingLoad) {\n void existingLoad.then(\n (event) => {\n if (key) loadedScripts.add(key);\n onLoad?.(event);\n onReady?.();\n },\n (event) => onError?.(event),\n );\n return;\n }\n }\n\n const el = document.createElement(\"script\");\n if (src) el.src = src;\n if (id) el.id = id;\n\n setScriptAttributes(el, rest);\n if (options.resolvedNonce && !el.getAttribute(\"nonce\")) {\n el.setAttribute(\"nonce\", options.resolvedNonce);\n }\n\n if (strategy === \"worker\") {\n el.setAttribute(\"type\", \"text/partytown\");\n }\n\n const markLoaded = () => {\n if (key) loadedScripts.add(key);\n onReady?.();\n };\n\n if (dangerouslySetInnerHTML?.__html) {\n // Intentional: mirrors the Next.js <Script> API where dangerouslySetInnerHTML\n // is developer-supplied inline script content (not user input). The prop name\n // itself signals developer awareness of the XSS risk, consistent with React's\n // design. User-supplied data must never flow into this prop.\n el.innerHTML = dangerouslySetInnerHTML.__html;\n markLoaded();\n } else if (children && typeof children === \"string\") {\n el.textContent = children;\n markLoaded();\n } else if (src) {\n const loadPromise = new Promise<Event>((resolve, reject) => {\n el.addEventListener(\"load\", (event) => {\n resolve(event);\n if (key) loadedScripts.add(key);\n onLoad?.(event);\n onReady?.();\n });\n el.addEventListener(\"error\", (event) => {\n reject(event);\n onError?.(event);\n });\n });\n loadPromise.catch(() => undefined).finally(() => loadingScripts.delete(src));\n loadingScripts.set(src, loadPromise);\n }\n\n document.body.appendChild(el);\n}\n\n/**\n * Load a script imperatively (outside of React).\n */\nexport function handleClientScriptLoad(props: ScriptProps): void {\n loadClientScript(props, {\n resolvedNonce: resolveScriptNonce(props.nonce),\n fireReadyWhenAlreadyLoaded: false,\n });\n}\n\n/**\n * Initialize multiple scripts at once (called during app bootstrap).\n */\nexport function initScriptLoader(scripts: ScriptProps[]): void {\n for (const script of scripts) {\n handleClientScriptLoad(script);\n }\n}\n\nfunction Script(props: ScriptProps): React.ReactElement | null {\n const {\n src,\n id,\n strategy = \"afterInteractive\",\n onLoad,\n onReady,\n onError,\n children,\n dangerouslySetInnerHTML,\n ...rest\n } = props;\n\n const hasMounted = useRef(false);\n const key = id ?? src ?? \"\";\n const contextualNonce = useScriptNonce();\n const resolvedNonce = resolveScriptNonce(rest.nonce, contextualNonce);\n\n // Client path: load scripts via useEffect based on strategy.\n // useEffect never runs during SSR, so it's safe to call unconditionally.\n useEffect(() => {\n if (hasMounted.current) return;\n hasMounted.current = true;\n\n if (strategy === \"beforeInteractive\") {\n return;\n }\n\n // Already loaded — just fire onReady\n if (key && loadedScripts.has(key)) {\n onReady?.();\n return;\n }\n\n const load = () => {\n if (key && loadedScripts.has(key)) {\n onReady?.();\n return;\n }\n\n loadClientScript(\n {\n src,\n id,\n strategy,\n onLoad,\n onReady,\n onError,\n children,\n dangerouslySetInnerHTML,\n ...rest,\n },\n { resolvedNonce, fireReadyWhenAlreadyLoaded: true },\n );\n };\n\n if (strategy === \"lazyOnload\") {\n // Wait for window load, then use idle callback\n if (document.readyState === \"complete\") {\n if (typeof requestIdleCallback === \"function\") {\n requestIdleCallback(load);\n } else {\n setTimeout(load, 1);\n }\n } else {\n window.addEventListener(\"load\", () => {\n if (typeof requestIdleCallback === \"function\") {\n requestIdleCallback(load);\n } else {\n setTimeout(load, 1);\n }\n });\n }\n } else {\n // \"afterInteractive\" (default), \"beforeInteractive\" (client re-mount), \"worker\"\n load();\n }\n }, [\n src,\n id,\n strategy,\n onLoad,\n onReady,\n onError,\n children,\n dangerouslySetInnerHTML,\n key,\n resolvedNonce,\n rest,\n ]);\n\n // SSR path: only \"beforeInteractive\" renders a <script> tag server-side\n if (typeof window === \"undefined\") {\n // React Float preload — emits <link rel=\"preload\" as=\"script\" /> in <head>\n // so the script is fetched while HTML streams. Mirrors Next.js's App Router\n // behavior at .nextjs-ref/packages/next/src/client/script.tsx:298-376:\n // - afterInteractive with src: preload only (no <script> tag in SSR)\n // - beforeInteractive with src: preload + <script> tag\n // - inline scripts (no src): no preload\n // Calling ReactDOM.preload during SSR is safe in both routers; React only\n // hoists the link when it has a real <head> to hoist into.\n if (\n src &&\n typeof ReactDOM.preload === \"function\" &&\n (strategy === \"afterInteractive\" || strategy === \"beforeInteractive\")\n ) {\n const integrity = typeof rest.integrity === \"string\" ? rest.integrity : undefined;\n const crossOrigin =\n rest.crossOrigin === \"anonymous\" || rest.crossOrigin === \"use-credentials\"\n ? rest.crossOrigin\n : undefined;\n const preloadOptions: ReactDOM.PreloadOptions = {\n as: \"script\",\n crossOrigin,\n };\n if (resolvedNonce !== undefined) {\n preloadOptions.nonce = resolvedNonce;\n }\n if (integrity !== undefined) {\n preloadOptions.integrity = integrity;\n }\n ReactDOM.preload(src, preloadOptions);\n }\n\n if (strategy === \"beforeInteractive\") {\n return React.createElement(\n \"script\",\n buildBeforeInteractiveScriptProps({\n src,\n id,\n rest,\n resolvedNonce,\n dangerouslySetInnerHTML,\n }),\n children,\n );\n }\n // Other strategies don't render during SSR\n return null;\n }\n\n if (strategy === \"beforeInteractive\") {\n return React.createElement(\n \"script\",\n buildBeforeInteractiveScriptProps({\n src,\n id,\n rest,\n resolvedNonce,\n dangerouslySetInnerHTML,\n }),\n children,\n );\n }\n\n // The component itself renders nothing — scripts are injected imperatively\n return null;\n}\n\nexport default Script;\n"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"script.js","names":[],"sources":["../../src/shims/script.tsx"],"sourcesContent":["\"use client\";\n\n/**\n * next/script shim\n *\n * Provides the <Script> component for loading third-party scripts with\n * configurable loading strategies.\n *\n * Strategies:\n * - \"beforeInteractive\": rendered as a <script> tag in SSR output\n * - \"afterInteractive\" (default): loaded client-side after hydration\n * - \"lazyOnload\": deferred until window.load + requestIdleCallback\n * - \"worker\": sets type=\"text/partytown\" (requires Partytown setup)\n */\nimport React, { useEffect, useRef } from \"react\";\nimport * as ReactDOM from \"react-dom\";\nimport { hasAppNavigationRuntimeBootstrap } from \"../client/navigation-runtime.js\";\nimport { escapeInlineContent } from \"./head.js\";\nimport { useScriptNonce } from \"./script-nonce-context.js\";\nimport {\n useBeforeInteractiveRegister,\n type BeforeInteractiveInlineScript,\n} from \"./before-interactive-context.js\";\n\nexport type ScriptProps = {\n /** Script source URL */\n src?: string;\n /** Loading strategy. Default: \"afterInteractive\" */\n strategy?: \"beforeInteractive\" | \"afterInteractive\" | \"lazyOnload\" | \"worker\";\n /** Unique identifier for the script */\n id?: string;\n /** Called when the script has loaded */\n onLoad?: (e: Event) => void;\n /** Called when the script is ready (after load, and on every re-render if already loaded) */\n onReady?: () => void;\n /** Called on script load error */\n onError?: (e: Event) => void;\n /** Inline script content */\n children?: React.ReactNode;\n /** Dangerous inner HTML */\n dangerouslySetInnerHTML?: { __html: string };\n /** Script type attribute */\n type?: string;\n /** Async attribute */\n async?: boolean;\n /** Defer attribute */\n defer?: boolean;\n /** Crossorigin attribute */\n crossOrigin?: string;\n /** Nonce for CSP */\n nonce?: string;\n /** Integrity hash */\n integrity?: string;\n /**\n * Associated stylesheets to load alongside the script. Emitted as\n * `<link rel=\"stylesheet\" href=\"...\">` on SSR (via `ReactDOM.preinit`)\n * and inserted into `<head>` on the client load path.\n *\n * Mirrors Next.js App Router behaviour at\n * `.nextjs-ref/packages/next/src/client/script.tsx` (`insertStylesheets`\n * and the `appDir` block).\n */\n stylesheets?: string[];\n /** Additional attributes */\n [key: string]: unknown;\n};\n\n// Track scripts that have already been loaded, plus remote scripts currently\n// loading, to avoid duplicate DOM insertion when same-src components mount\n// before the first load event fires.\nconst loadedScripts = new Set<string>();\nconst loadingScripts = new Map<string, Promise<Event>>();\n\nfunction getClientAutoNonce(): string | undefined {\n if (typeof document === \"undefined\") return undefined;\n\n const existingNonceElement = document.querySelector(\"[nonce]\");\n if (!existingNonceElement) return undefined;\n\n // `HTMLElement` is not defined in some SSR/edge runtimes that polyfill\n // `document` but stop short of the full DOM surface. Guarding the\n // constructor before `instanceof` keeps SSR from crashing in those hosts;\n // when the constructor *is* present we still prefer the typed `.nonce`\n // property because browsers strip the `nonce` attribute from serialised\n // HTML for CSP reasons.\n if (typeof HTMLElement !== \"undefined\" && existingNonceElement instanceof HTMLElement) {\n return existingNonceElement.nonce || existingNonceElement.getAttribute(\"nonce\") || undefined;\n }\n\n return existingNonceElement.getAttribute(\"nonce\") || undefined;\n}\n\nfunction resolveScriptNonce(explicitNonce: unknown, contextualNonce?: string): string | undefined {\n if (typeof explicitNonce === \"string\" && explicitNonce.length > 0) {\n return explicitNonce;\n }\n\n if (typeof contextualNonce === \"string\" && contextualNonce.length > 0) {\n return contextualNonce;\n }\n\n if (typeof window === \"undefined\") {\n return undefined;\n }\n\n return getClientAutoNonce();\n}\n\n/**\n * Insert `<link rel=\"stylesheet\">` tags into `document.head` for each entry\n * in `stylesheets`. Used by the imperative client-side load path\n * (`handleClientScriptLoad`) when `ReactDOM.preinit` is not available\n * (e.g. pre-Float React or hosts that strip it). Mirrors Next.js's\n * `insertStylesheets` Pages-Router fallback at\n * `.nextjs-ref/packages/next/src/client/script.tsx:48-59`.\n *\n * The `ReactDOM.preinit` path is preferred where available — it dedupes\n * across mounts and respects React Float's hoisting order. This DOM\n * fallback is best-effort: no dedupe, no ordering guarantee.\n */\nfunction insertClientStylesheets(stylesheets: string[] | undefined): void {\n if (!stylesheets || stylesheets.length === 0) return;\n if (typeof document === \"undefined\") return;\n\n // Prefer ReactDOM.preinit when available — it dedupes via React Float\n // and matches Next.js's app-router behaviour.\n if (typeof ReactDOM.preinit === \"function\") {\n for (const href of stylesheets) {\n ReactDOM.preinit(href, { as: \"style\" });\n }\n return;\n }\n\n const head = document.head;\n if (!head) return;\n for (const href of stylesheets) {\n const link = document.createElement(\"link\");\n link.rel = \"stylesheet\";\n link.type = \"text/css\";\n link.href = href;\n head.appendChild(link);\n }\n}\n\n/**\n * Emit `<link rel=\"stylesheet\">` tags during SSR for each entry in\n * `stylesheets` via `ReactDOM.preinit`. React Float hoists these into\n * `<head>` in the streamed HTML. Mirrors the App-Router branch of\n * Next.js's Script component at `.nextjs-ref/packages/next/src/client/script.tsx:309-313`.\n */\nfunction preinitStylesheetsForSSR(stylesheets: string[] | undefined): void {\n if (!stylesheets || stylesheets.length === 0) return;\n if (typeof ReactDOM.preinit !== \"function\") return;\n for (const href of stylesheets) {\n ReactDOM.preinit(href, { as: \"style\" });\n }\n}\n\nfunction buildBeforeInteractiveScriptProps(options: {\n src?: string;\n id?: string;\n rest: Record<string, unknown>;\n resolvedNonce?: string;\n dangerouslySetInnerHTML?: { __html: string };\n}): Record<string, unknown> {\n const scriptProps: Record<string, unknown> = { ...options.rest };\n if (options.src) scriptProps.src = options.src;\n if (options.id) scriptProps.id = options.id;\n if (options.resolvedNonce) {\n scriptProps.nonce = options.resolvedNonce;\n }\n if (options.dangerouslySetInnerHTML) {\n scriptProps.dangerouslySetInnerHTML = {\n __html: escapeInlineContent(options.dangerouslySetInnerHTML.__html, \"script\"),\n };\n }\n return scriptProps;\n}\n\n/**\n * Extract the inline script content for a `beforeInteractive` Script element\n * with no `src`. Returns `null` when the element has neither a string-shaped\n * `children` value nor a valid `dangerouslySetInnerHTML.__html` payload — in\n * that case the caller should fall through to React's regular rendering path.\n *\n * The returned string is the raw author-supplied JavaScript content. Callers\n * are responsible for passing it through `escapeInlineContent(..., \"script\")`\n * before emitting it inside a `<script>` tag (we keep that escape adjacent\n * to the emit point so the rule is obvious at the boundary).\n */\nfunction extractBeforeInteractiveInlineContent(\n children: React.ReactNode,\n dangerouslySetInnerHTML?: { __html: string },\n): string | null {\n if (\n dangerouslySetInnerHTML &&\n typeof dangerouslySetInnerHTML.__html === \"string\" &&\n dangerouslySetInnerHTML.__html.length > 0\n ) {\n return dangerouslySetInnerHTML.__html;\n }\n if (typeof children === \"string\" && children.length > 0) {\n return children;\n }\n if (Array.isArray(children) && children.every((c) => typeof c === \"string\")) {\n const joined = (children as string[]).join(\"\");\n return joined.length > 0 ? joined : null;\n }\n return null;\n}\n\n/**\n * Map of React DOM prop names to their HTML attribute equivalents. Mirrors\n * Next.js's `set-attributes-from-props.ts`:\n * .nextjs-ref/packages/next/src/client/set-attributes-from-props.ts\n * HTML parses attribute names case-insensitively, so without this translation\n * `className=\"foo\"` round-trips as `classname=\"foo\"` and CSS selectors on\n * `.foo` never match. Same hazard for `htmlFor`/`for`, `httpEquiv`/`http-equiv`,\n * `acceptCharset`/`accept-charset`.\n */\nconst REACT_TO_HTML_ATTR: Record<string, string> = {\n acceptCharset: \"accept-charset\",\n className: \"class\",\n htmlFor: \"for\",\n httpEquiv: \"http-equiv\",\n};\n\n/**\n * Convert the residual `<Script>` props into a plain string-attributes record\n * for emission inside a hoisted `<script>` tag. Drops React-only props\n * (event handlers, children, etc.) and reserved keys already handled by the\n * pre-head-injection emitter (id, nonce). Skips `undefined`/`null` so they\n * round-trip as \"attribute absent\" rather than `attr=\"undefined\"`.\n *\n * React DOM prop names (className, htmlFor, etc.) are translated to their\n * HTML attribute names so the output parses correctly — see comment on\n * `REACT_TO_HTML_ATTR`.\n */\nfunction collectBeforeInteractiveAttributes(\n rest: Record<string, unknown>,\n): Record<string, string | boolean> {\n const RESERVED = new Set([\n \"id\",\n \"nonce\",\n \"src\",\n \"children\",\n \"strategy\",\n \"dangerouslySetInnerHTML\",\n \"onLoad\",\n \"onReady\",\n \"onError\",\n \"stylesheets\",\n ]);\n const out: Record<string, string | boolean> = {};\n for (const [key, value] of Object.entries(rest)) {\n if (RESERVED.has(key)) continue;\n if (value === undefined || value === null || value === false) continue;\n const attrName = REACT_TO_HTML_ATTR[key] ?? key;\n if (typeof value === \"boolean\") {\n out[attrName] = true;\n continue;\n }\n if (typeof value === \"string\" || typeof value === \"number\") {\n out[attrName] = String(value);\n continue;\n }\n // Skip anything else (functions, objects) — they cannot serialise into an\n // HTML attribute and only the developer-controlled string/boolean shape\n // is expected for native `<script>` attributes here.\n }\n return out;\n}\n\nfunction setBooleanScriptAttribute(el: HTMLScriptElement, attr: string, value: unknown): boolean {\n const enabled = value !== false && value !== \"false\" && Boolean(value);\n\n switch (attr) {\n case \"async\":\n el.async = enabled;\n break;\n case \"defer\":\n el.defer = enabled;\n break;\n case \"noModule\":\n case \"nomodule\":\n el.noModule = enabled;\n break;\n default:\n return false;\n }\n\n if (!enabled) {\n // Dynamic script elements start in the browser's force-async state.\n // Setting and removing the attribute mirrors Next.js and clears that state.\n el.setAttribute(attr, \"\");\n el.removeAttribute(attr);\n }\n\n return true;\n}\n\nfunction setScriptAttributes(el: HTMLScriptElement, rest: Record<string, unknown>): void {\n for (const [attr, value] of Object.entries(rest)) {\n if (attr === \"dangerouslySetInnerHTML\") continue;\n if (value === undefined) continue;\n if (setBooleanScriptAttribute(el, attr, value)) continue;\n if (attr === \"className\" && typeof value === \"string\") {\n el.setAttribute(\"class\", value);\n } else if (typeof value === \"string\") {\n el.setAttribute(attr, value);\n } else if (typeof value === \"boolean\" && value) {\n el.setAttribute(attr, \"\");\n }\n }\n}\n\nfunction loadClientScript(\n props: ScriptProps,\n options: {\n resolvedNonce?: string;\n fireReadyWhenAlreadyLoaded: boolean;\n },\n): void {\n const {\n src,\n id,\n onLoad,\n onReady,\n onError,\n strategy = \"afterInteractive\",\n children,\n dangerouslySetInnerHTML,\n stylesheets,\n ...rest\n } = props;\n if (typeof window === \"undefined\") return;\n\n // Insert associated stylesheets into <head> regardless of whether the\n // script was already loaded — the script's onReady handlers may already\n // assume the stylesheet is present. `insertClientStylesheets` dedupes\n // via ReactDOM.preinit where available.\n insertClientStylesheets(stylesheets);\n\n const key = id ?? src ?? \"\";\n if (key && loadedScripts.has(key)) {\n if (options.fireReadyWhenAlreadyLoaded) {\n onReady?.();\n }\n return;\n }\n\n if (src) {\n const existingLoad = loadingScripts.get(src);\n if (existingLoad) {\n void existingLoad.then(\n (event) => {\n if (key) loadedScripts.add(key);\n onLoad?.(event);\n onReady?.();\n },\n (event) => onError?.(event),\n );\n return;\n }\n }\n\n const el = document.createElement(\"script\");\n if (src) el.src = src;\n if (id) el.id = id;\n\n setScriptAttributes(el, rest);\n if (options.resolvedNonce && !el.getAttribute(\"nonce\")) {\n el.setAttribute(\"nonce\", options.resolvedNonce);\n }\n\n if (strategy === \"worker\") {\n el.setAttribute(\"type\", \"text/partytown\");\n }\n\n const markLoaded = () => {\n if (key) loadedScripts.add(key);\n onReady?.();\n };\n\n if (dangerouslySetInnerHTML?.__html) {\n // Intentional: mirrors the Next.js <Script> API where dangerouslySetInnerHTML\n // is developer-supplied inline script content (not user input). The prop name\n // itself signals developer awareness of the XSS risk, consistent with React's\n // design. User-supplied data must never flow into this prop.\n el.innerHTML = dangerouslySetInnerHTML.__html;\n markLoaded();\n } else if (children && typeof children === \"string\") {\n el.textContent = children;\n markLoaded();\n } else if (src) {\n const loadPromise = new Promise<Event>((resolve, reject) => {\n el.addEventListener(\"load\", (event) => {\n resolve(event);\n if (key) loadedScripts.add(key);\n onLoad?.(event);\n onReady?.();\n });\n el.addEventListener(\"error\", (event) => {\n reject(event);\n onError?.(event);\n });\n });\n loadPromise.catch(() => undefined).finally(() => loadingScripts.delete(src));\n loadingScripts.set(src, loadPromise);\n }\n\n document.body.appendChild(el);\n}\n\n/**\n * Load a script imperatively (outside of React).\n */\nexport function handleClientScriptLoad(props: ScriptProps): void {\n loadClientScript(props, {\n resolvedNonce: resolveScriptNonce(props.nonce),\n fireReadyWhenAlreadyLoaded: false,\n });\n}\n\n/**\n * Initialize multiple scripts at once (called during app bootstrap).\n */\nexport function initScriptLoader(scripts: ScriptProps[]): void {\n for (const script of scripts) {\n handleClientScriptLoad(script);\n }\n}\n\nfunction Script(props: ScriptProps): React.ReactElement | null {\n const {\n src,\n id,\n strategy = \"afterInteractive\",\n onLoad,\n onReady,\n onError,\n children,\n dangerouslySetInnerHTML,\n stylesheets,\n ...rest\n } = props;\n\n const hasMounted = useRef(false);\n const key = id ?? src ?? \"\";\n const contextualNonce = useScriptNonce();\n const resolvedNonce = resolveScriptNonce(rest.nonce, contextualNonce);\n // Available only during SSR — the provider lives in app-ssr-entry.ts. When\n // missing (Pages Router SSR, raw renderToString, client render) we keep the\n // inline `<script>` element in source order.\n const registerBeforeInteractive = useBeforeInteractiveRegister();\n\n // Client path: load scripts via useEffect based on strategy.\n // useEffect never runs during SSR, so it's safe to call unconditionally.\n useEffect(() => {\n if (hasMounted.current) return;\n hasMounted.current = true;\n\n if (strategy === \"beforeInteractive\") {\n // The script itself is loaded by Next.js's bootstrap before hydration,\n // but the associated stylesheets still need to land in <head> on the\n // client. ReactDOM.preinit (called inside insertClientStylesheets)\n // dedupes against any SSR-emitted <link rel=\"stylesheet\">, so this is\n // safe even when the server already hoisted them via React Float.\n insertClientStylesheets(stylesheets);\n return;\n }\n\n // Already loaded — just fire onReady\n if (key && loadedScripts.has(key)) {\n // Stylesheets must still be inserted on subsequent mounts of the same\n // script. loadClientScript handles this for the fresh-load path; the\n // already-loaded shortcut needs it explicitly.\n insertClientStylesheets(stylesheets);\n onReady?.();\n return;\n }\n\n const load = () => {\n if (key && loadedScripts.has(key)) {\n onReady?.();\n return;\n }\n\n loadClientScript(\n {\n src,\n id,\n strategy,\n onLoad,\n onReady,\n onError,\n children,\n dangerouslySetInnerHTML,\n stylesheets,\n ...rest,\n },\n { resolvedNonce, fireReadyWhenAlreadyLoaded: true },\n );\n };\n\n if (strategy === \"lazyOnload\") {\n // Wait for window load, then use idle callback\n if (document.readyState === \"complete\") {\n if (typeof requestIdleCallback === \"function\") {\n requestIdleCallback(load);\n } else {\n setTimeout(load, 1);\n }\n } else {\n window.addEventListener(\"load\", () => {\n if (typeof requestIdleCallback === \"function\") {\n requestIdleCallback(load);\n } else {\n setTimeout(load, 1);\n }\n });\n }\n } else {\n // \"afterInteractive\" (default), \"beforeInteractive\" (client re-mount), \"worker\"\n load();\n }\n }, [\n src,\n id,\n strategy,\n onLoad,\n onReady,\n onError,\n children,\n dangerouslySetInnerHTML,\n stylesheets,\n key,\n resolvedNonce,\n rest,\n ]);\n\n // SSR path: only \"beforeInteractive\" renders a <script> tag server-side\n if (typeof window === \"undefined\") {\n // Emit associated stylesheets as <link rel=\"stylesheet\"> via React Float.\n // ReactDOM.preinit dedupes across mounts and hoists the link into <head>\n // regardless of strategy — matches Next.js's app-router branch at\n // `.nextjs-ref/packages/next/src/client/script.tsx:309-313`.\n preinitStylesheetsForSSR(stylesheets);\n\n // React Float preload — emits <link rel=\"preload\" as=\"script\" /> in <head>\n // so the script is fetched while HTML streams. Mirrors Next.js's App Router\n // behavior at .nextjs-ref/packages/next/src/client/script.tsx:298-376:\n // - afterInteractive with src: preload only (no <script> tag in SSR)\n // - beforeInteractive with src: preload + <script> tag\n // - inline scripts (no src): no preload\n // Calling ReactDOM.preload during SSR is safe in both routers; React only\n // hoists the link when it has a real <head> to hoist into.\n if (\n src &&\n typeof ReactDOM.preload === \"function\" &&\n (strategy === \"afterInteractive\" || strategy === \"beforeInteractive\")\n ) {\n const integrity = typeof rest.integrity === \"string\" ? rest.integrity : undefined;\n const crossOrigin =\n rest.crossOrigin === \"anonymous\" || rest.crossOrigin === \"use-credentials\"\n ? rest.crossOrigin\n : undefined;\n const preloadOptions: ReactDOM.PreloadOptions = {\n as: \"script\",\n crossOrigin,\n };\n if (resolvedNonce !== undefined) {\n preloadOptions.nonce = resolvedNonce;\n }\n if (integrity !== undefined) {\n preloadOptions.integrity = integrity;\n }\n ReactDOM.preload(src, preloadOptions);\n }\n\n if (strategy === \"beforeInteractive\") {\n // Inline beforeInteractive scripts (no src) need to run BEFORE any\n // stylesheets, modulepreload links, or other resource hints React Float\n // hoists into <head>. React Fizz emits user-rendered head children\n // AFTER the hoisted resources, so leaving the script in source order\n // breaks the no-flash dark-mode pattern. We instead capture the inline\n // content through BeforeInteractiveContext and the SSR pipeline emits\n // it immediately after `<head>` opens — guaranteeing it precedes every\n // React-emitted hint in the streamed HTML.\n const inlineContent = src\n ? null\n : extractBeforeInteractiveInlineContent(children, dangerouslySetInnerHTML);\n if (inlineContent !== null && registerBeforeInteractive) {\n const inline: BeforeInteractiveInlineScript = {\n id,\n // Escape `</script>` sequences exactly as the inline render path does\n // (see buildBeforeInteractiveScriptProps); keep the escape colocated\n // with the emit boundary so it never gets accidentally skipped.\n innerHTML: escapeInlineContent(inlineContent, \"script\"),\n nonce: resolvedNonce,\n attributes: collectBeforeInteractiveAttributes(rest),\n };\n registerBeforeInteractive(inline);\n return null;\n }\n\n return React.createElement(\n \"script\",\n buildBeforeInteractiveScriptProps({\n src,\n id,\n rest,\n resolvedNonce,\n dangerouslySetInnerHTML,\n }),\n children,\n );\n }\n // Other strategies don't render during SSR\n return null;\n }\n\n if (strategy === \"beforeInteractive\") {\n // On the client, only suppress the `<script>` render for inline\n // beforeInteractive Scripts in App Router pages. The pre-head splice\n // in app-ssr-entry/app-ssr-stream already put the tag in the DOM, so\n // rendering it again would either duplicate the script (for Scripts\n // outside `<head>`) or cause a hydration mismatch (positions differ).\n //\n // For Pages Router and any other SSR path that didn't run through\n // app-ssr-entry, the server rendered the `<script>` inline in source\n // order, so the client must match. We detect \"App Router\" via the\n // navigation runtime that the App Router bootstrap installs before\n // calling hydrateRoot — it is the most reliable runtime signal we\n // can read from inside a `\"use client\"` shim.\n //\n // External-`src` beforeInteractive scripts always keep rendering\n // inline. They are not captured by the pre-head splice and must mount\n // through React so their `src` attribute is fetched on the client.\n const inlineContent = src\n ? null\n : extractBeforeInteractiveInlineContent(children, dangerouslySetInnerHTML);\n if (inlineContent !== null && hasAppNavigationRuntimeBootstrap()) {\n return null;\n }\n\n return React.createElement(\n \"script\",\n buildBeforeInteractiveScriptProps({\n src,\n id,\n rest,\n resolvedNonce,\n dangerouslySetInnerHTML,\n }),\n children,\n );\n }\n\n // The component itself renders nothing — scripts are injected imperatively\n return null;\n}\n\nexport default Script;\n"],"mappings":";;;;;;;;;;;;;;;;;;;;AAsEA,MAAM,gCAAgB,IAAI,KAAa;AACvC,MAAM,iCAAiB,IAAI,KAA6B;AAExD,SAAS,qBAAyC;CAChD,IAAI,OAAO,aAAa,aAAa,OAAO,KAAA;CAE5C,MAAM,uBAAuB,SAAS,cAAc,UAAU;CAC9D,IAAI,CAAC,sBAAsB,OAAO,KAAA;CAQlC,IAAI,OAAO,gBAAgB,eAAe,gCAAgC,aACxE,OAAO,qBAAqB,SAAS,qBAAqB,aAAa,QAAQ,IAAI,KAAA;CAGrF,OAAO,qBAAqB,aAAa,QAAQ,IAAI,KAAA;;AAGvD,SAAS,mBAAmB,eAAwB,iBAA8C;CAChG,IAAI,OAAO,kBAAkB,YAAY,cAAc,SAAS,GAC9D,OAAO;CAGT,IAAI,OAAO,oBAAoB,YAAY,gBAAgB,SAAS,GAClE,OAAO;CAGT,IAAI,OAAO,WAAW,aACpB;CAGF,OAAO,oBAAoB;;;;;;;;;;;;;;AAe7B,SAAS,wBAAwB,aAAyC;CACxE,IAAI,CAAC,eAAe,YAAY,WAAW,GAAG;CAC9C,IAAI,OAAO,aAAa,aAAa;CAIrC,IAAI,OAAO,SAAS,YAAY,YAAY;EAC1C,KAAK,MAAM,QAAQ,aACjB,SAAS,QAAQ,MAAM,EAAE,IAAI,SAAS,CAAC;EAEzC;;CAGF,MAAM,OAAO,SAAS;CACtB,IAAI,CAAC,MAAM;CACX,KAAK,MAAM,QAAQ,aAAa;EAC9B,MAAM,OAAO,SAAS,cAAc,OAAO;EAC3C,KAAK,MAAM;EACX,KAAK,OAAO;EACZ,KAAK,OAAO;EACZ,KAAK,YAAY,KAAK;;;;;;;;;AAU1B,SAAS,yBAAyB,aAAyC;CACzE,IAAI,CAAC,eAAe,YAAY,WAAW,GAAG;CAC9C,IAAI,OAAO,SAAS,YAAY,YAAY;CAC5C,KAAK,MAAM,QAAQ,aACjB,SAAS,QAAQ,MAAM,EAAE,IAAI,SAAS,CAAC;;AAI3C,SAAS,kCAAkC,SAMf;CAC1B,MAAM,cAAuC,EAAE,GAAG,QAAQ,MAAM;CAChE,IAAI,QAAQ,KAAK,YAAY,MAAM,QAAQ;CAC3C,IAAI,QAAQ,IAAI,YAAY,KAAK,QAAQ;CACzC,IAAI,QAAQ,eACV,YAAY,QAAQ,QAAQ;CAE9B,IAAI,QAAQ,yBACV,YAAY,0BAA0B,EACpC,QAAQ,oBAAoB,QAAQ,wBAAwB,QAAQ,SAAS,EAC9E;CAEH,OAAO;;;;;;;;;;;;;AAcT,SAAS,sCACP,UACA,yBACe;CACf,IACE,2BACA,OAAO,wBAAwB,WAAW,YAC1C,wBAAwB,OAAO,SAAS,GAExC,OAAO,wBAAwB;CAEjC,IAAI,OAAO,aAAa,YAAY,SAAS,SAAS,GACpD,OAAO;CAET,IAAI,MAAM,QAAQ,SAAS,IAAI,SAAS,OAAO,MAAM,OAAO,MAAM,SAAS,EAAE;EAC3E,MAAM,SAAU,SAAsB,KAAK,GAAG;EAC9C,OAAO,OAAO,SAAS,IAAI,SAAS;;CAEtC,OAAO;;;;;;;;;;;AAYT,MAAM,qBAA6C;CACjD,eAAe;CACf,WAAW;CACX,SAAS;CACT,WAAW;CACZ;;;;;;;;;;;;AAaD,SAAS,mCACP,MACkC;CAClC,MAAM,WAAW,IAAI,IAAI;EACvB;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACD,CAAC;CACF,MAAM,MAAwC,EAAE;CAChD,KAAK,MAAM,CAAC,KAAK,UAAU,OAAO,QAAQ,KAAK,EAAE;EAC/C,IAAI,SAAS,IAAI,IAAI,EAAE;EACvB,IAAI,UAAU,KAAA,KAAa,UAAU,QAAQ,UAAU,OAAO;EAC9D,MAAM,WAAW,mBAAmB,QAAQ;EAC5C,IAAI,OAAO,UAAU,WAAW;GAC9B,IAAI,YAAY;GAChB;;EAEF,IAAI,OAAO,UAAU,YAAY,OAAO,UAAU,UAAU;GAC1D,IAAI,YAAY,OAAO,MAAM;GAC7B;;;CAMJ,OAAO;;AAGT,SAAS,0BAA0B,IAAuB,MAAc,OAAyB;CAC/F,MAAM,UAAU,UAAU,SAAS,UAAU,WAAW,QAAQ,MAAM;CAEtE,QAAQ,MAAR;EACE,KAAK;GACH,GAAG,QAAQ;GACX;EACF,KAAK;GACH,GAAG,QAAQ;GACX;EACF,KAAK;EACL,KAAK;GACH,GAAG,WAAW;GACd;EACF,SACE,OAAO;;CAGX,IAAI,CAAC,SAAS;EAGZ,GAAG,aAAa,MAAM,GAAG;EACzB,GAAG,gBAAgB,KAAK;;CAG1B,OAAO;;AAGT,SAAS,oBAAoB,IAAuB,MAAqC;CACvF,KAAK,MAAM,CAAC,MAAM,UAAU,OAAO,QAAQ,KAAK,EAAE;EAChD,IAAI,SAAS,2BAA2B;EACxC,IAAI,UAAU,KAAA,GAAW;EACzB,IAAI,0BAA0B,IAAI,MAAM,MAAM,EAAE;EAChD,IAAI,SAAS,eAAe,OAAO,UAAU,UAC3C,GAAG,aAAa,SAAS,MAAM;OAC1B,IAAI,OAAO,UAAU,UAC1B,GAAG,aAAa,MAAM,MAAM;OACvB,IAAI,OAAO,UAAU,aAAa,OACvC,GAAG,aAAa,MAAM,GAAG;;;AAK/B,SAAS,iBACP,OACA,SAIM;CACN,MAAM,EACJ,KACA,IACA,QACA,SACA,SACA,WAAW,oBACX,UACA,yBACA,aACA,GAAG,SACD;CACJ,IAAI,OAAO,WAAW,aAAa;CAMnC,wBAAwB,YAAY;CAEpC,MAAM,MAAM,MAAM,OAAO;CACzB,IAAI,OAAO,cAAc,IAAI,IAAI,EAAE;EACjC,IAAI,QAAQ,4BACV,WAAW;EAEb;;CAGF,IAAI,KAAK;EACP,MAAM,eAAe,eAAe,IAAI,IAAI;EAC5C,IAAI,cAAc;GAChB,aAAkB,MACf,UAAU;IACT,IAAI,KAAK,cAAc,IAAI,IAAI;IAC/B,SAAS,MAAM;IACf,WAAW;OAEZ,UAAU,UAAU,MAAM,CAC5B;GACD;;;CAIJ,MAAM,KAAK,SAAS,cAAc,SAAS;CAC3C,IAAI,KAAK,GAAG,MAAM;CAClB,IAAI,IAAI,GAAG,KAAK;CAEhB,oBAAoB,IAAI,KAAK;CAC7B,IAAI,QAAQ,iBAAiB,CAAC,GAAG,aAAa,QAAQ,EACpD,GAAG,aAAa,SAAS,QAAQ,cAAc;CAGjD,IAAI,aAAa,UACf,GAAG,aAAa,QAAQ,iBAAiB;CAG3C,MAAM,mBAAmB;EACvB,IAAI,KAAK,cAAc,IAAI,IAAI;EAC/B,WAAW;;CAGb,IAAI,yBAAyB,QAAQ;EAKnC,GAAG,YAAY,wBAAwB;EACvC,YAAY;QACP,IAAI,YAAY,OAAO,aAAa,UAAU;EACnD,GAAG,cAAc;EACjB,YAAY;QACP,IAAI,KAAK;EACd,MAAM,cAAc,IAAI,SAAgB,SAAS,WAAW;GAC1D,GAAG,iBAAiB,SAAS,UAAU;IACrC,QAAQ,MAAM;IACd,IAAI,KAAK,cAAc,IAAI,IAAI;IAC/B,SAAS,MAAM;IACf,WAAW;KACX;GACF,GAAG,iBAAiB,UAAU,UAAU;IACtC,OAAO,MAAM;IACb,UAAU,MAAM;KAChB;IACF;EACF,YAAY,YAAY,KAAA,EAAU,CAAC,cAAc,eAAe,OAAO,IAAI,CAAC;EAC5E,eAAe,IAAI,KAAK,YAAY;;CAGtC,SAAS,KAAK,YAAY,GAAG;;;;;AAM/B,SAAgB,uBAAuB,OAA0B;CAC/D,iBAAiB,OAAO;EACtB,eAAe,mBAAmB,MAAM,MAAM;EAC9C,4BAA4B;EAC7B,CAAC;;;;;AAMJ,SAAgB,iBAAiB,SAA8B;CAC7D,KAAK,MAAM,UAAU,SACnB,uBAAuB,OAAO;;AAIlC,SAAS,OAAO,OAA+C;CAC7D,MAAM,EACJ,KACA,IACA,WAAW,oBACX,QACA,SACA,SACA,UACA,yBACA,aACA,GAAG,SACD;CAEJ,MAAM,aAAa,OAAO,MAAM;CAChC,MAAM,MAAM,MAAM,OAAO;CACzB,MAAM,kBAAkB,gBAAgB;CACxC,MAAM,gBAAgB,mBAAmB,KAAK,OAAO,gBAAgB;CAIrE,MAAM,4BAA4B,8BAA8B;CAIhE,gBAAgB;EACd,IAAI,WAAW,SAAS;EACxB,WAAW,UAAU;EAErB,IAAI,aAAa,qBAAqB;GAMpC,wBAAwB,YAAY;GACpC;;EAIF,IAAI,OAAO,cAAc,IAAI,IAAI,EAAE;GAIjC,wBAAwB,YAAY;GACpC,WAAW;GACX;;EAGF,MAAM,aAAa;GACjB,IAAI,OAAO,cAAc,IAAI,IAAI,EAAE;IACjC,WAAW;IACX;;GAGF,iBACE;IACE;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA,GAAG;IACJ,EACD;IAAE;IAAe,4BAA4B;IAAM,CACpD;;EAGH,IAAI,aAAa,cAEf,IAAI,SAAS,eAAe,YAC1B,IAAI,OAAO,wBAAwB,YACjC,oBAAoB,KAAK;OAEzB,WAAW,MAAM,EAAE;OAGrB,OAAO,iBAAiB,cAAc;GACpC,IAAI,OAAO,wBAAwB,YACjC,oBAAoB,KAAK;QAEzB,WAAW,MAAM,EAAE;IAErB;OAIJ,MAAM;IAEP;EACD;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACD,CAAC;CAGF,IAAI,OAAO,WAAW,aAAa;EAKjC,yBAAyB,YAAY;EAUrC,IACE,OACA,OAAO,SAAS,YAAY,eAC3B,aAAa,sBAAsB,aAAa,sBACjD;GACA,MAAM,YAAY,OAAO,KAAK,cAAc,WAAW,KAAK,YAAY,KAAA;GAKxE,MAAM,iBAA0C;IAC9C,IAAI;IACJ,aALA,KAAK,gBAAgB,eAAe,KAAK,gBAAgB,oBACrD,KAAK,cACL,KAAA;IAIL;GACD,IAAI,kBAAkB,KAAA,GACpB,eAAe,QAAQ;GAEzB,IAAI,cAAc,KAAA,GAChB,eAAe,YAAY;GAE7B,SAAS,QAAQ,KAAK,eAAe;;EAGvC,IAAI,aAAa,qBAAqB;GASpC,MAAM,gBAAgB,MAClB,OACA,sCAAsC,UAAU,wBAAwB;GAC5E,IAAI,kBAAkB,QAAQ,2BAA2B;IAUvD,0BAA0B;KARxB;KAIA,WAAW,oBAAoB,eAAe,SAAS;KACvD,OAAO;KACP,YAAY,mCAAmC,KAAK;KAEtB,CAAC;IACjC,OAAO;;GAGT,OAAO,MAAM,cACX,UACA,kCAAkC;IAChC;IACA;IACA;IACA;IACA;IACD,CAAC,EACF,SACD;;EAGH,OAAO;;CAGT,IAAI,aAAa,qBAAqB;EAoBpC,KAHsB,MAClB,OACA,sCAAsC,UAAU,wBAAwB,MACtD,QAAQ,kCAAkC,EAC9D,OAAO;EAGT,OAAO,MAAM,cACX,UACA,kCAAkC;GAChC;GACA;GACA;GACA;GACA;GACD,CAAC,EACF,SACD;;CAIH,OAAO"}
|
package/dist/shims/server.js
CHANGED
|
@@ -46,6 +46,7 @@ var NextRequest = class extends Request {
|
|
|
46
46
|
_url;
|
|
47
47
|
_cookies;
|
|
48
48
|
constructor(input, init) {
|
|
49
|
+
validateURL(typeof input !== "string" && "url" in input ? input.url : String(input));
|
|
49
50
|
const { nextConfig: _nextConfig, ...requestInit } = init ?? {};
|
|
50
51
|
if (input instanceof Request) {
|
|
51
52
|
const requestInput = requestInit.body === void 0 && input.body && !input.bodyUsed ? input.clone() : input;
|
package/dist/shims/server.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"server.js","names":[],"sources":["../../src/shims/server.ts"],"sourcesContent":["/**\n * next/server shim\n *\n * Provides NextRequest, NextResponse, and related types that work with\n * standard Web APIs (Request/Response). This means they work on Node,\n * Cloudflare Workers, Deno, and any WinterCG-compatible runtime.\n *\n * This is a pragmatic subset — we implement the most commonly used APIs\n * rather than bug-for-bug parity with Next.js internals.\n */\n\nimport {\n MIDDLEWARE_NEXT_HEADER,\n MIDDLEWARE_REWRITE_HEADER,\n MIDDLEWARE_SET_COOKIE_HEADER,\n} from \"../server/headers.js\";\nimport { encodeMiddlewareRequestHeaders } from \"../server/middleware-request-headers.js\";\nimport { serializeSetCookie, validateCookieName } from \"./internal/cookie-serialize.js\";\nimport { parseCookieHeader } from \"./internal/parse-cookie-header.js\";\nimport { getRequestExecutionContext } from \"./request-context.js\";\nimport { assertSafeNavigationUrl } from \"./url-safety.js\";\nimport { stripBasePath } from \"../utils/base-path.js\";\n\n// ---------------------------------------------------------------------------\n// Inlined cache-scope guard for after()\n//\n// We cannot statically import throwIfInsideCacheScope from headers.ts here\n// because headers.ts contains the \"use cache\" directive string in its error\n// message, which causes Vite's use-cache transform to include it in the module\n// graph. If headers.ts is pulled in via static import from server.ts, the\n// transform fires on it in Pages Router fixtures that lack @vitejs/plugin-rsc.\n//\n// The connection() function in this file avoids the same problem by using\n// `await import(\"./headers.js\")` (dynamic import, async function). after()\n// must remain synchronous, so we inline the check using the same Symbol.for\n// keys that cache-runtime.ts and cache.ts register their ALS instances with.\n// ---------------------------------------------------------------------------\n\nconst _USE_CACHE_ALS_KEY = Symbol.for(\"vinext.cacheRuntime.contextAls\");\nconst _UNSTABLE_CACHE_ALS_KEY = Symbol.for(\"vinext.unstableCache.als\");\nconst _g = globalThis as unknown as Record<PropertyKey, unknown>;\n\n/**\n * Record an invalid dynamic usage error on the request context so it survives\n * user try/catch and can be forwarded to the dev overlay on client-side navigations.\n */\nfunction _recordInvalidDynamicUsageError(error: Error): void {\n try {\n const _unifiedAls = _g[Symbol.for(\"vinext.unifiedRequestContext.als\")] as\n | { getStore(): unknown }\n | undefined;\n const ctx = _unifiedAls?.getStore() as Record<string, unknown> | undefined;\n if (ctx) ctx.invalidDynamicUsageError = error;\n } catch {\n // Ignore — best-effort recording for dev diagnostics\n }\n}\n\nfunction _throwIfInsideCacheScope(apiName: string): void {\n const cacheAls = _g[_USE_CACHE_ALS_KEY] as { getStore(): unknown } | undefined;\n if (cacheAls?.getStore() != null) {\n const error = new Error(\n `\\`${apiName}\\` cannot be called inside \"use cache\". ` +\n `If you need this data inside a cached function, call \\`${apiName}\\` ` +\n \"outside and pass the required data as an argument.\",\n );\n _recordInvalidDynamicUsageError(error);\n throw error;\n }\n const unstableAls = _g[_UNSTABLE_CACHE_ALS_KEY] as { getStore(): unknown } | undefined;\n if (unstableAls?.getStore() === true) {\n const error = new Error(\n `\\`${apiName}\\` cannot be called inside a function cached with \\`unstable_cache()\\`. ` +\n `If you need this data inside a cached function, call \\`${apiName}\\` ` +\n \"outside and pass the required data as an argument.\",\n );\n _recordInvalidDynamicUsageError(error);\n throw error;\n }\n}\n\n// ---------------------------------------------------------------------------\n// NextRequest\n// ---------------------------------------------------------------------------\n\nexport class NextRequest extends Request {\n private _nextUrl: NextURL;\n private _url: string;\n private _cookies: RequestCookies;\n\n constructor(\n input: URL | RequestInfo,\n init?: RequestInit & {\n nextConfig?: {\n basePath?: string;\n i18n?: { locales: string[]; defaultLocale: string };\n trailingSlash?: boolean;\n };\n },\n ) {\n // Strip nextConfig before passing to super() — it's vinext-internal,\n // not a valid RequestInit property.\n const { nextConfig: _nextConfig, ...requestInit } = init ?? {};\n if (input instanceof Request) {\n // Keep caller-owned request bodies readable after wrapping. Middleware and\n // route-handler plumbing may need the source Request after this wrapper runs.\n const requestInput =\n requestInit.body === undefined && input.body && !input.bodyUsed ? input.clone() : input;\n super(requestInput, requestInit);\n } else {\n super(input, requestInit);\n }\n const url =\n typeof input === \"string\"\n ? new URL(input, \"http://localhost\")\n : input instanceof URL\n ? input\n : new URL(input.url, \"http://localhost\");\n const urlConfig: NextURLConfig | undefined = _nextConfig\n ? {\n basePath: _nextConfig.basePath,\n nextConfig: { i18n: _nextConfig.i18n, trailingSlash: _nextConfig.trailingSlash },\n }\n : undefined;\n this._nextUrl = new NextURL(url, undefined, urlConfig);\n this._url = process.env.__NEXT_NO_MIDDLEWARE_URL_NORMALIZE\n ? url.toString()\n : this._nextUrl.toString();\n this._cookies = new RequestCookies(this.headers);\n }\n\n get nextUrl(): NextURL {\n return this._nextUrl;\n }\n\n get url(): string {\n return this._url;\n }\n\n get cookies(): RequestCookies {\n return this._cookies;\n }\n\n /**\n * Client IP address. Prefers Cloudflare's trusted CF-Connecting-IP header\n * over the spoofable X-Forwarded-For. Returns undefined if unavailable.\n */\n get ip(): string | undefined {\n return (\n this.headers.get(\"cf-connecting-ip\") ??\n this.headers.get(\"x-real-ip\") ??\n this.headers.get(\"x-forwarded-for\")?.split(\",\")[0]?.trim() ??\n undefined\n );\n }\n\n /**\n * Geolocation data. Platform-dependent (e.g., Cloudflare, Vercel).\n * Returns undefined if not available.\n */\n get geo():\n | { city?: string; country?: string; region?: string; latitude?: string; longitude?: string }\n | undefined {\n // Check Cloudflare-style headers, Vercel-style headers\n const country =\n this.headers.get(\"cf-ipcountry\") ?? this.headers.get(\"x-vercel-ip-country\") ?? undefined;\n if (!country) return undefined;\n return {\n country,\n city: this.headers.get(\"cf-ipcity\") ?? this.headers.get(\"x-vercel-ip-city\") ?? undefined,\n region:\n this.headers.get(\"cf-region\") ??\n this.headers.get(\"x-vercel-ip-country-region\") ??\n undefined,\n latitude:\n this.headers.get(\"cf-iplatitude\") ?? this.headers.get(\"x-vercel-ip-latitude\") ?? undefined,\n longitude:\n this.headers.get(\"cf-iplongitude\") ??\n this.headers.get(\"x-vercel-ip-longitude\") ??\n undefined,\n };\n }\n\n /**\n * The build ID of the Next.js application.\n * Delegates to `nextUrl.buildId` to match Next.js API surface.\n * Can be used in middleware to detect deployment skew between client and server.\n */\n get buildId(): string | undefined {\n return this._nextUrl.buildId;\n }\n}\n\n// ---------------------------------------------------------------------------\n// NextResponse\n// ---------------------------------------------------------------------------\n\n/** Valid HTTP redirect status codes, matching Next.js's REDIRECTS set. */\nconst REDIRECT_STATUSES = new Set([301, 302, 303, 307, 308]);\n\nfunction validateURL(url: string | URL | NextURL): string {\n assertSafeNavigationUrl(String(url));\n try {\n return String(new URL(String(url)));\n } catch (error) {\n throw new Error(\n `URL is malformed \"${String(\n url,\n )}\". Please use only absolute URLs - https://nextjs.org/docs/messages/middleware-relative-urls`,\n { cause: error },\n );\n }\n}\n\nexport class NextResponse<_Body = unknown> extends Response {\n private _cookies: ResponseCookies;\n\n constructor(body?: BodyInit | null, init?: ResponseInit) {\n super(body, init);\n this._cookies = new MiddlewareResponseCookies(this.headers);\n }\n\n get cookies(): ResponseCookies {\n return this._cookies;\n }\n\n /**\n * Create a JSON response.\n */\n static json<JsonBody>(body: JsonBody, init?: ResponseInit): NextResponse<JsonBody> {\n const headers = new Headers(init?.headers);\n if (!headers.has(\"content-type\")) {\n headers.set(\"content-type\", \"application/json\");\n }\n return new NextResponse(JSON.stringify(body), {\n ...init,\n headers,\n }) as NextResponse<JsonBody>;\n }\n\n /**\n * Create a redirect response.\n */\n static redirect(url: string | URL | NextURL, init?: number | ResponseInit): NextResponse {\n const status = typeof init === \"number\" ? init : (init?.status ?? 307);\n if (!REDIRECT_STATUSES.has(status)) {\n throw new RangeError(`Failed to execute \"redirect\" on \"response\": Invalid status code`);\n }\n const headers = new Headers(typeof init === \"object\" ? init?.headers : undefined);\n headers.set(\"Location\", validateURL(url));\n return new NextResponse(null, { status, headers });\n }\n\n /**\n * Create a rewrite response (middleware pattern).\n * Sets the x-middleware-rewrite header.\n */\n static rewrite(destination: string | URL | NextURL, init?: MiddlewareResponseInit): NextResponse {\n const headers = new Headers(init?.headers);\n headers.set(MIDDLEWARE_REWRITE_HEADER, validateURL(destination));\n if (init?.request?.headers) {\n encodeMiddlewareRequestHeaders(headers, init.request.headers);\n }\n return new NextResponse(null, { ...init, headers });\n }\n\n /**\n * Continue to the next handler (middleware pattern).\n * Sets the x-middleware-next header.\n */\n static next(init?: MiddlewareResponseInit): NextResponse {\n const headers = new Headers(init?.headers);\n headers.set(MIDDLEWARE_NEXT_HEADER, \"1\");\n if (init?.request?.headers) {\n encodeMiddlewareRequestHeaders(headers, init.request.headers);\n }\n return new NextResponse(null, { ...init, headers });\n }\n}\n\n// ---------------------------------------------------------------------------\n// NextURL — lightweight URL wrapper with pathname helpers\n// ---------------------------------------------------------------------------\n\nexport type NextURLConfig = {\n basePath?: string;\n nextConfig?: {\n i18n?: {\n locales: string[];\n defaultLocale: string;\n };\n /**\n * When true, `href`/`toString()` formats non-root, non-file-like pathnames\n * with a trailing slash. Matches Next.js's `formatNextPathnameInfo` so that\n * `NextResponse.redirect(request.nextUrl)` and `NextResponse.rewrite(url)`\n * honour the user's `trailingSlash` config.\n */\n trailingSlash?: boolean;\n };\n};\n\nexport class NextURL {\n /** Internal URL stores the pathname WITHOUT basePath or locale prefix. */\n private _url: URL;\n private _basePath: string;\n private _trailingSlash: boolean;\n private _locale: string | undefined;\n private _defaultLocale: string | undefined;\n private _locales: string[] | undefined;\n\n constructor(input: string | URL, base?: string | URL, config?: NextURLConfig) {\n this._url = new URL(input.toString(), base);\n this._basePath = config?.basePath ?? \"\";\n this._trailingSlash = config?.nextConfig?.trailingSlash ?? false;\n this._stripBasePath();\n const i18n = config?.nextConfig?.i18n;\n if (i18n) {\n this._locales = [...i18n.locales];\n this._defaultLocale = i18n.defaultLocale;\n this._analyzeLocale(this._locales);\n }\n }\n\n /** Strip basePath prefix from the internal pathname. */\n private _stripBasePath(): void {\n if (!this._basePath) return;\n this._url.pathname = stripBasePath(this._url.pathname, this._basePath);\n }\n\n /** Extract locale from pathname, stripping it from the internal URL. */\n private _analyzeLocale(locales: string[]): void {\n const segments = this._url.pathname.split(\"/\");\n const candidate = segments[1]?.toLowerCase();\n const match = locales.find((l) => l.toLowerCase() === candidate);\n if (match) {\n this._locale = match;\n this._url.pathname = \"/\" + segments.slice(2).join(\"/\");\n } else {\n this._locale = this._defaultLocale;\n }\n }\n\n /**\n * Reconstruct the full pathname with basePath + locale prefix and apply\n * the configured trailingSlash policy.\n * Mirrors Next.js's internal formatNextPathnameInfo().\n */\n private _formatPathname(): string {\n // Build prefix: basePath + locale (skip defaultLocale — Next.js omits it)\n let prefix = this._basePath;\n if (this._locale && this._locale !== this._defaultLocale) {\n prefix += \"/\" + this._locale;\n }\n const inner = this._url.pathname;\n const composed = !prefix ? inner : inner === \"/\" ? prefix : prefix + inner;\n return this._applyTrailingSlash(composed);\n }\n\n /**\n * Apply the configured trailingSlash policy to a composed pathname. Matches\n * Next.js's `formatNextPathnameInfo`: when `trailingSlash` is true, add a\n * trailing slash unless the path is empty/root; when false, strip a trailing\n * slash unless the path is empty/root.\n */\n private _applyTrailingSlash(pathname: string): string {\n // Never strip or add a slash to the root path.\n if (pathname === \"\" || pathname === \"/\") return pathname;\n if (this._trailingSlash) {\n return pathname.endsWith(\"/\") ? pathname : pathname + \"/\";\n }\n return pathname.endsWith(\"/\") ? pathname.slice(0, -1) : pathname;\n }\n\n get href(): string {\n const formatted = this._formatPathname();\n if (formatted === this._url.pathname) return this._url.href;\n // Replace pathname in href via string slicing — avoids URL allocation.\n // URL.href is always <origin+auth><pathname><search><hash>.\n const { href, pathname, search, hash } = this._url;\n const baseEnd = href.length - pathname.length - search.length - hash.length;\n return href.slice(0, baseEnd) + formatted + search + hash;\n }\n set href(value: string) {\n this._url.href = value;\n this._stripBasePath();\n if (this._locales) this._analyzeLocale(this._locales);\n }\n\n get origin(): string {\n return this._url.origin;\n }\n\n get protocol(): string {\n return this._url.protocol;\n }\n set protocol(value: string) {\n this._url.protocol = value;\n }\n\n get username(): string {\n return this._url.username;\n }\n set username(value: string) {\n this._url.username = value;\n }\n\n get password(): string {\n return this._url.password;\n }\n set password(value: string) {\n this._url.password = value;\n }\n\n get host(): string {\n return this._url.host;\n }\n set host(value: string) {\n this._url.host = value;\n }\n\n get hostname(): string {\n return this._url.hostname;\n }\n set hostname(value: string) {\n this._url.hostname = value;\n }\n\n get port(): string {\n return this._url.port;\n }\n set port(value: string) {\n this._url.port = value;\n }\n\n /** Returns the pathname WITHOUT basePath or locale prefix. */\n get pathname(): string {\n return this._url.pathname;\n }\n set pathname(value: string) {\n this._url.pathname = value;\n }\n\n get search(): string {\n return this._url.search;\n }\n set search(value: string) {\n this._url.search = value;\n }\n\n get searchParams(): URLSearchParams {\n return this._url.searchParams;\n }\n\n get hash(): string {\n return this._url.hash;\n }\n set hash(value: string) {\n this._url.hash = value;\n }\n\n get basePath(): string {\n return this._basePath;\n }\n set basePath(value: string) {\n this._basePath = value === \"\" ? \"\" : value.startsWith(\"/\") ? value : \"/\" + value;\n }\n\n get locale(): string {\n return this._locale ?? \"\";\n }\n set locale(value: string | undefined) {\n if (this._locales) {\n if (!value) {\n this._locale = this._defaultLocale;\n return;\n }\n if (!this._locales.includes(value)) {\n throw new TypeError(\n `The locale \"${value}\" is not in the configured locales: ${this._locales.join(\", \")}`,\n );\n }\n }\n this._locale = this._locales ? value : this._locale;\n }\n\n get defaultLocale(): string | undefined {\n return this._defaultLocale;\n }\n\n get locales(): string[] | undefined {\n return this._locales ? [...this._locales] : undefined;\n }\n\n clone(): NextURL {\n const nextConfig: NonNullable<NextURLConfig[\"nextConfig\"]> = {};\n if (this._locales) {\n nextConfig.i18n = { locales: [...this._locales], defaultLocale: this._defaultLocale! };\n }\n if (this._trailingSlash) {\n nextConfig.trailingSlash = true;\n }\n const config: NextURLConfig = {\n basePath: this._basePath,\n nextConfig: Object.keys(nextConfig).length > 0 ? nextConfig : undefined,\n };\n // Pass the full href (with locale/basePath re-added) so the constructor\n // can re-analyze and extract locale correctly.\n return new NextURL(this.href, undefined, config);\n }\n\n toString(): string {\n return this.href;\n }\n\n /**\n * The build ID of the Next.js application.\n * Set from `generateBuildId` in next.config.js, or a random UUID if not configured.\n * Can be used in middleware to detect deployment skew between client and server.\n * Matches the Next.js API: `request.nextUrl.buildId`.\n */\n get buildId(): string | undefined {\n return process.env.__VINEXT_BUILD_ID ?? undefined;\n }\n}\n\n// ---------------------------------------------------------------------------\n// Cookie helpers (minimal implementations)\n// ---------------------------------------------------------------------------\n\ntype CookieEntry = {\n name: string;\n value: string;\n};\n\nexport class RequestCookies {\n private _headers: Headers;\n private _parsed: Map<string, string>;\n\n constructor(headers: Headers) {\n this._headers = headers;\n this._parsed = parseCookieHeader(headers.get(\"cookie\") ?? \"\");\n }\n\n get(name: string): CookieEntry | undefined {\n const value = this._parsed.get(name);\n return value !== undefined ? { name, value } : undefined;\n }\n\n getAll(nameOrOptions?: string | CookieEntry): CookieEntry[] {\n const name = typeof nameOrOptions === \"string\" ? nameOrOptions : nameOrOptions?.name;\n return [...this._parsed.entries()]\n .filter(([cookieName]) => name === undefined || cookieName === name)\n .map(([cookieName, value]) => ({ name: cookieName, value }));\n }\n\n has(name: string): boolean {\n return this._parsed.has(name);\n }\n\n set(nameOrOptions: string | CookieEntry, value?: string): this {\n let cookieName: string;\n let cookieValue: string;\n if (typeof nameOrOptions === \"string\") {\n cookieName = nameOrOptions;\n cookieValue = value ?? \"\";\n } else {\n cookieName = nameOrOptions.name;\n cookieValue = nameOrOptions.value;\n }\n validateCookieName(cookieName);\n this._parsed.set(cookieName, cookieValue);\n this._syncHeader();\n return this;\n }\n\n delete(names: string | string[]): boolean | boolean[] {\n if (Array.isArray(names)) {\n const results = names.map((name) => {\n validateCookieName(name);\n return this._parsed.delete(name);\n });\n this._syncHeader();\n return results;\n }\n validateCookieName(names);\n const result = this._parsed.delete(names);\n this._syncHeader();\n return result;\n }\n\n clear(): this {\n this._parsed.clear();\n this._syncHeader();\n return this;\n }\n\n get size(): number {\n return this._parsed.size;\n }\n\n toString(): string {\n return this._serialize();\n }\n\n private _serialize(): string {\n return [...this._parsed.entries()].map(([n, v]) => `${n}=${encodeURIComponent(v)}`).join(\"; \");\n }\n\n private _syncHeader(): void {\n if (this._parsed.size === 0) {\n this._headers.delete(\"cookie\");\n } else {\n this._headers.set(\"cookie\", this._serialize());\n }\n }\n\n [Symbol.iterator](): IterableIterator<[string, CookieEntry]> {\n const entries = this.getAll().map((c) => [c.name, c] as [string, CookieEntry]);\n return entries[Symbol.iterator]();\n }\n}\n\n// Keep this error message in sync with headers.ts. This adapter backs\n// NextRequest cookies, while headers.ts owns the next/headers cookies object.\nclass ReadonlyRequestCookiesError extends Error {\n constructor() {\n super(\n \"Cookies can only be modified in a Server Action or Route Handler. Read more: https://nextjs.org/docs/app/api-reference/functions/cookies#options\",\n );\n }\n\n static callable(this: void): never {\n throw new ReadonlyRequestCookiesError();\n }\n}\n\nconst REQUEST_HEADERS_MUTATING_METHODS = new Set([\"set\", \"delete\", \"append\"]);\n\n// Keep this error message in sync with headers.ts. This adapter backs\n// NextRequest headers in force-static route handlers, while headers.ts owns the\n// next/headers object.\nclass ReadonlyRequestHeadersError extends Error {\n constructor() {\n super(\n \"Headers cannot be modified. Read more: https://nextjs.org/docs/app/api-reference/functions/headers\",\n );\n }\n\n static callable(this: void): never {\n throw new ReadonlyRequestHeadersError();\n }\n}\n\nexport function sealRequestHeaders(headers: Headers): Headers {\n return new Proxy<Headers>(headers, {\n get(target, prop) {\n if (typeof prop === \"string\" && REQUEST_HEADERS_MUTATING_METHODS.has(prop)) {\n return ReadonlyRequestHeadersError.callable;\n }\n\n const value = Reflect.get(target, prop, target);\n return typeof value === \"function\" ? value.bind(target) : value;\n },\n });\n}\n\nexport function sealRequestCookies(cookies: RequestCookies): RequestCookies {\n return new Proxy<RequestCookies>(cookies, {\n get(target, prop) {\n if (prop === \"set\" || prop === \"delete\" || prop === \"clear\") {\n return ReadonlyRequestCookiesError.callable;\n }\n\n const value = Reflect.get(target, prop, target);\n return typeof value === \"function\" ? value.bind(target) : value;\n },\n });\n}\n\nexport class ResponseCookies {\n private _headers: Headers;\n /** Internal map keyed by cookie name — single source of truth. */\n private _parsed: Map<string, { serialized: string; entry: CookieEntry }> = new Map();\n\n constructor(headers: Headers) {\n this._headers = headers;\n\n // Hydrate internal map from any existing Set-Cookie headers\n for (const header of headers.getSetCookie()) {\n const eq = header.indexOf(\"=\");\n if (eq === -1) continue;\n const cookieName = header.slice(0, eq);\n const semi = header.indexOf(\";\", eq);\n const raw = header.slice(eq + 1, semi === -1 ? undefined : semi);\n let value: string;\n try {\n value = decodeURIComponent(raw);\n } catch {\n value = raw;\n }\n this._parsed.set(cookieName, { serialized: header, entry: { name: cookieName, value } });\n }\n }\n\n set(\n ...args:\n | [name: string, value: string, options?: CookieOptions]\n | [options: CookieOptions & { name: string; value: string }]\n ): this {\n const [name, value, opts] = parseCookieSetArgs(args);\n validateCookieName(name);\n\n const serialized = serializeSetCookie(name, value, opts);\n this._parsed.set(name, { serialized, entry: { name, value } });\n this._syncHeaders();\n return this;\n }\n\n get(...args: [name: string] | [options: { name: string }]): CookieEntry | undefined {\n const key = typeof args[0] === \"string\" ? args[0] : args[0].name;\n return this._parsed.get(key)?.entry;\n }\n\n has(name: string): boolean {\n return this._parsed.has(name);\n }\n\n getAll(...args: [name: string] | [options: { name: string }] | []): CookieEntry[] {\n const all = [...this._parsed.values()].map((v) => v.entry);\n if (args.length === 0) return all;\n const key = typeof args[0] === \"string\" ? args[0] : args[0].name;\n return all.filter((c) => c.name === key);\n }\n\n delete(\n ...args:\n | [name: string]\n | [options: Omit<CookieOptions & { name: string }, \"maxAge\" | \"expires\">]\n ): this {\n const [name, opts] =\n typeof args[0] === \"string\" ? [args[0], undefined] : [args[0].name, args[0]];\n return this.set({\n name,\n value: \"\",\n expires: new Date(0),\n path: opts?.path,\n domain: opts?.domain,\n httpOnly: opts?.httpOnly,\n secure: opts?.secure,\n sameSite: opts?.sameSite,\n });\n }\n\n [Symbol.iterator](): IterableIterator<[string, CookieEntry]> {\n const entries: [string, CookieEntry][] = [...this._parsed.values()].map((v) => [\n v.entry.name,\n v.entry,\n ]);\n return entries[Symbol.iterator]();\n }\n\n /** Delete all Set-Cookie headers and re-append from the internal map. */\n private _syncHeaders(): void {\n this._headers.delete(\"Set-Cookie\");\n for (const { serialized } of this._parsed.values()) {\n this._headers.append(\"Set-Cookie\", serialized);\n }\n }\n}\n\nclass MiddlewareResponseCookies extends ResponseCookies {\n private _responseHeaders: Headers;\n\n constructor(headers: Headers) {\n super(headers);\n this._responseHeaders = headers;\n }\n\n override set(\n ...args:\n | [name: string, value: string, options?: CookieOptions]\n | [options: CookieOptions & { name: string; value: string }]\n ): this {\n super.set(...args);\n this._syncMiddlewareCookieHeader();\n return this;\n }\n\n override delete(\n ...args:\n | [name: string]\n | [options: Omit<CookieOptions & { name: string }, \"maxAge\" | \"expires\">]\n ): this {\n super.delete(...args);\n this._syncMiddlewareCookieHeader();\n return this;\n }\n\n private _syncMiddlewareCookieHeader(): void {\n const cookies = this._responseHeaders.getSetCookie();\n if (cookies.length === 0) {\n this._responseHeaders.delete(MIDDLEWARE_SET_COOKIE_HEADER);\n return;\n }\n\n this._responseHeaders.set(MIDDLEWARE_SET_COOKIE_HEADER, cookies.join(\",\"));\n }\n}\n\ntype CookieOptions = {\n path?: string;\n domain?: string;\n maxAge?: number;\n expires?: Date;\n httpOnly?: boolean;\n secure?: boolean;\n sameSite?: \"Strict\" | \"Lax\" | \"None\";\n};\n\n/**\n * Parse the overloaded arguments for ResponseCookies.set():\n * - (name, value, options?) — positional form\n * - ({ name, value, ...options }) — object form\n */\nfunction parseCookieSetArgs(\n args:\n | [name: string, value: string, options?: CookieOptions]\n | [options: CookieOptions & { name: string; value: string }],\n): [string, string, CookieOptions | undefined] {\n if (typeof args[0] === \"string\") {\n return [args[0], args[1] as string, args[2] as CookieOptions | undefined];\n }\n const { name, value, ...opts } = args[0];\n return [name, value, opts as CookieOptions];\n}\n\n// ---------------------------------------------------------------------------\n// Types\n// ---------------------------------------------------------------------------\n\nexport type MiddlewareResponseInit = {\n request?: {\n headers?: Headers;\n };\n} & ResponseInit;\n\nexport type NextMiddlewareResult = NextResponse | Response | null | undefined | void;\n\nexport type NextMiddleware = (\n request: NextRequest,\n event: NextFetchEvent,\n) => NextMiddlewareResult | Promise<NextMiddlewareResult>;\n\n/**\n * Minimal NextFetchEvent — extends FetchEvent where available,\n * otherwise provides the waitUntil pattern standalone.\n */\nexport class NextFetchEvent {\n sourcePage: string;\n private _waitUntilPromises: Promise<unknown>[] = [];\n\n constructor(params: { page: string }) {\n this.sourcePage = params.page;\n }\n\n waitUntil(promise: Promise<unknown>): void {\n this._waitUntilPromises.push(promise);\n }\n\n get waitUntilPromises(): Promise<unknown>[] {\n return this._waitUntilPromises;\n }\n\n /** Drain all waitUntil promises. Returns a single promise that settles when all are done. */\n drainWaitUntil(): Promise<PromiseSettledResult<unknown>[]> {\n return Promise.allSettled(this._waitUntilPromises);\n }\n}\n\n// ---------------------------------------------------------------------------\n// Utility exports\n// ---------------------------------------------------------------------------\n\n/**\n * Parse user agent string. Minimal implementation — for full UA parsing,\n * apps should use a dedicated library like `ua-parser-js`.\n */\nexport function userAgentFromString(ua: string | undefined): UserAgent {\n const input = ua ?? \"\";\n return {\n isBot: /bot|crawler|spider|crawling/i.test(input),\n ua: input,\n browser: {},\n device: {},\n engine: {},\n os: {},\n cpu: {},\n };\n}\n\nexport function userAgent({ headers }: { headers: Headers }): UserAgent {\n return userAgentFromString(headers.get(\"user-agent\") ?? undefined);\n}\n\nexport type UserAgent = {\n isBot: boolean;\n ua: string;\n browser: { name?: string; version?: string; major?: string };\n device: { model?: string; type?: string; vendor?: string };\n engine: { name?: string; version?: string };\n os: { name?: string; version?: string };\n cpu: { architecture?: string };\n};\n\n/**\n * after() — schedule work after the response is sent.\n *\n * Uses the platform's `waitUntil` (via the per-request ExecutionContext) when\n * available so the task survives past the response on Cloudflare Workers.\n * Falls back to a fire-and-forget microtask on runtimes without an execution\n * context (e.g. Node.js dev server).\n *\n * Throws when called inside a cached scope — request-specific\n * side-effects must not leak into cached results.\n */\nexport function after<T>(task: Promise<T> | (() => T | Promise<T>)): void {\n _throwIfInsideCacheScope(\"after()\");\n\n const promise = typeof task === \"function\" ? Promise.resolve().then(task) : task;\n // NOTE: vinext runs function tasks concurrently with response streaming (next microtask),\n // whereas Next.js queues them to run strictly after the response is sent via onClose.\n // This is a known simplification — function tasks here are not guaranteed to run\n // after the response completes, only after the current synchronous execution.\n //\n // `.catch()` is attached synchronously in the same tick as `promise` is created, so\n // there is no window where a pre-rejected `task` promise could trigger an\n // `unhandledrejection` event before the handler is in place.\n const guarded = promise.catch((err) => {\n console.error(\"[vinext] after() task failed:\", err);\n });\n\n // TODO: Next.js throws when after() is called outside a request context or when\n // waitUntil is unavailable, preventing silent task loss. vinext falls back to\n // fire-and-forget here, which is correct for the Node.js dev server (where\n // getRequestExecutionContext() always returns null). On Workers, a misconfigured\n // entry that omits runWithExecutionContext would silently drop tasks — consider\n // a one-time console.warn on the fallback path, gated to production only (e.g.\n // `process.env.NODE_ENV === 'production'` or `typeof caches !== 'undefined'` for\n // a Workers runtime check) with a module-level `let _warned = false` guard so it\n // fires at most once and doesn't spam the dev-server console.\n getRequestExecutionContext()?.waitUntil(guarded);\n}\n\n/**\n * connection() — signals that the response requires a live connection\n * (not a static/cached response). Opts the page out of ISR caching\n * and sets Cache-Control: no-store on the response.\n */\nexport async function connection(): Promise<void> {\n const { markDynamicUsage, markRenderRequestApiUsage, throwIfInsideCacheScope } =\n await import(\"./headers.js\");\n markRenderRequestApiUsage(\"connection\");\n throwIfInsideCacheScope(\"connection()\");\n markDynamicUsage();\n}\n\n/**\n * URLPattern re-export — used in middleware for route matching.\n * Available natively in Node 20+, Cloudflare Workers, Deno.\n * Falls back to urlpattern-polyfill if the global is not available.\n */\nexport const URLPattern: typeof globalThis.URLPattern =\n globalThis.URLPattern ??\n (() => {\n throw new Error(\n \"URLPattern is not available in this runtime. \" +\n \"Install the `urlpattern-polyfill` package or upgrade to Node 20+.\",\n );\n });\n"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"server.js","names":[],"sources":["../../src/shims/server.ts"],"sourcesContent":["/**\n * next/server shim\n *\n * Provides NextRequest, NextResponse, and related types that work with\n * standard Web APIs (Request/Response). This means they work on Node,\n * Cloudflare Workers, Deno, and any WinterCG-compatible runtime.\n *\n * This is a pragmatic subset — we implement the most commonly used APIs\n * rather than bug-for-bug parity with Next.js internals.\n */\n\nimport {\n MIDDLEWARE_NEXT_HEADER,\n MIDDLEWARE_REWRITE_HEADER,\n MIDDLEWARE_SET_COOKIE_HEADER,\n} from \"../server/headers.js\";\nimport { encodeMiddlewareRequestHeaders } from \"../server/middleware-request-headers.js\";\nimport { serializeSetCookie, validateCookieName } from \"./internal/cookie-serialize.js\";\nimport { parseCookieHeader } from \"./internal/parse-cookie-header.js\";\nimport { getRequestExecutionContext } from \"./request-context.js\";\nimport { assertSafeNavigationUrl } from \"./url-safety.js\";\nimport { stripBasePath } from \"../utils/base-path.js\";\n\n// ---------------------------------------------------------------------------\n// Inlined cache-scope guard for after()\n//\n// We cannot statically import throwIfInsideCacheScope from headers.ts here\n// because headers.ts contains the \"use cache\" directive string in its error\n// message, which causes Vite's use-cache transform to include it in the module\n// graph. If headers.ts is pulled in via static import from server.ts, the\n// transform fires on it in Pages Router fixtures that lack @vitejs/plugin-rsc.\n//\n// The connection() function in this file avoids the same problem by using\n// `await import(\"./headers.js\")` (dynamic import, async function). after()\n// must remain synchronous, so we inline the check using the same Symbol.for\n// keys that cache-runtime.ts and cache.ts register their ALS instances with.\n// ---------------------------------------------------------------------------\n\nconst _USE_CACHE_ALS_KEY = Symbol.for(\"vinext.cacheRuntime.contextAls\");\nconst _UNSTABLE_CACHE_ALS_KEY = Symbol.for(\"vinext.unstableCache.als\");\nconst _g = globalThis as unknown as Record<PropertyKey, unknown>;\n\n/**\n * Record an invalid dynamic usage error on the request context so it survives\n * user try/catch and can be forwarded to the dev overlay on client-side navigations.\n */\nfunction _recordInvalidDynamicUsageError(error: Error): void {\n try {\n const _unifiedAls = _g[Symbol.for(\"vinext.unifiedRequestContext.als\")] as\n | { getStore(): unknown }\n | undefined;\n const ctx = _unifiedAls?.getStore() as Record<string, unknown> | undefined;\n if (ctx) ctx.invalidDynamicUsageError = error;\n } catch {\n // Ignore — best-effort recording for dev diagnostics\n }\n}\n\nfunction _throwIfInsideCacheScope(apiName: string): void {\n const cacheAls = _g[_USE_CACHE_ALS_KEY] as { getStore(): unknown } | undefined;\n if (cacheAls?.getStore() != null) {\n const error = new Error(\n `\\`${apiName}\\` cannot be called inside \"use cache\". ` +\n `If you need this data inside a cached function, call \\`${apiName}\\` ` +\n \"outside and pass the required data as an argument.\",\n );\n _recordInvalidDynamicUsageError(error);\n throw error;\n }\n const unstableAls = _g[_UNSTABLE_CACHE_ALS_KEY] as { getStore(): unknown } | undefined;\n if (unstableAls?.getStore() === true) {\n const error = new Error(\n `\\`${apiName}\\` cannot be called inside a function cached with \\`unstable_cache()\\`. ` +\n `If you need this data inside a cached function, call \\`${apiName}\\` ` +\n \"outside and pass the required data as an argument.\",\n );\n _recordInvalidDynamicUsageError(error);\n throw error;\n }\n}\n\n// ---------------------------------------------------------------------------\n// NextRequest\n// ---------------------------------------------------------------------------\n\nexport class NextRequest extends Request {\n private _nextUrl: NextURL;\n private _url: string;\n private _cookies: RequestCookies;\n\n constructor(\n input: URL | RequestInfo,\n init?: RequestInit & {\n nextConfig?: {\n basePath?: string;\n i18n?: { locales: string[]; defaultLocale: string };\n trailingSlash?: boolean;\n };\n },\n ) {\n // Match Next.js: reject relative URLs with the canonical error before any\n // fallback URL parsing kicks in. Next.js calls `validateURL(url)` at the\n // top of its NextRequest constructor; we mirror that here so middleware\n // tests asserting on the error message text get the documented string.\n // Reuse the local `validateURL` helper so the message format stays in lockstep\n // with NextResponse, and so `javascript:` / `data:` URIs are blocked too.\n const rawUrl = typeof input !== \"string\" && \"url\" in input ? input.url : String(input);\n validateURL(rawUrl);\n // Strip nextConfig before passing to super() — it's vinext-internal,\n // not a valid RequestInit property.\n const { nextConfig: _nextConfig, ...requestInit } = init ?? {};\n if (input instanceof Request) {\n // Keep caller-owned request bodies readable after wrapping. Middleware and\n // route-handler plumbing may need the source Request after this wrapper runs.\n const requestInput =\n requestInit.body === undefined && input.body && !input.bodyUsed ? input.clone() : input;\n super(requestInput, requestInit);\n } else {\n super(input, requestInit);\n }\n const url =\n typeof input === \"string\"\n ? new URL(input, \"http://localhost\")\n : input instanceof URL\n ? input\n : new URL(input.url, \"http://localhost\");\n const urlConfig: NextURLConfig | undefined = _nextConfig\n ? {\n basePath: _nextConfig.basePath,\n nextConfig: { i18n: _nextConfig.i18n, trailingSlash: _nextConfig.trailingSlash },\n }\n : undefined;\n this._nextUrl = new NextURL(url, undefined, urlConfig);\n this._url = process.env.__NEXT_NO_MIDDLEWARE_URL_NORMALIZE\n ? url.toString()\n : this._nextUrl.toString();\n this._cookies = new RequestCookies(this.headers);\n }\n\n get nextUrl(): NextURL {\n return this._nextUrl;\n }\n\n get url(): string {\n return this._url;\n }\n\n get cookies(): RequestCookies {\n return this._cookies;\n }\n\n /**\n * Client IP address. Prefers Cloudflare's trusted CF-Connecting-IP header\n * over the spoofable X-Forwarded-For. Returns undefined if unavailable.\n */\n get ip(): string | undefined {\n return (\n this.headers.get(\"cf-connecting-ip\") ??\n this.headers.get(\"x-real-ip\") ??\n this.headers.get(\"x-forwarded-for\")?.split(\",\")[0]?.trim() ??\n undefined\n );\n }\n\n /**\n * Geolocation data. Platform-dependent (e.g., Cloudflare, Vercel).\n * Returns undefined if not available.\n */\n get geo():\n | { city?: string; country?: string; region?: string; latitude?: string; longitude?: string }\n | undefined {\n // Check Cloudflare-style headers, Vercel-style headers\n const country =\n this.headers.get(\"cf-ipcountry\") ?? this.headers.get(\"x-vercel-ip-country\") ?? undefined;\n if (!country) return undefined;\n return {\n country,\n city: this.headers.get(\"cf-ipcity\") ?? this.headers.get(\"x-vercel-ip-city\") ?? undefined,\n region:\n this.headers.get(\"cf-region\") ??\n this.headers.get(\"x-vercel-ip-country-region\") ??\n undefined,\n latitude:\n this.headers.get(\"cf-iplatitude\") ?? this.headers.get(\"x-vercel-ip-latitude\") ?? undefined,\n longitude:\n this.headers.get(\"cf-iplongitude\") ??\n this.headers.get(\"x-vercel-ip-longitude\") ??\n undefined,\n };\n }\n\n /**\n * The build ID of the Next.js application.\n * Delegates to `nextUrl.buildId` to match Next.js API surface.\n * Can be used in middleware to detect deployment skew between client and server.\n */\n get buildId(): string | undefined {\n return this._nextUrl.buildId;\n }\n}\n\n// ---------------------------------------------------------------------------\n// NextResponse\n// ---------------------------------------------------------------------------\n\n/** Valid HTTP redirect status codes, matching Next.js's REDIRECTS set. */\nconst REDIRECT_STATUSES = new Set([301, 302, 303, 307, 308]);\n\nfunction validateURL(url: string | URL | NextURL): string {\n assertSafeNavigationUrl(String(url));\n try {\n return String(new URL(String(url)));\n } catch (error) {\n throw new Error(\n `URL is malformed \"${String(\n url,\n )}\". Please use only absolute URLs - https://nextjs.org/docs/messages/middleware-relative-urls`,\n { cause: error },\n );\n }\n}\n\nexport class NextResponse<_Body = unknown> extends Response {\n private _cookies: ResponseCookies;\n\n constructor(body?: BodyInit | null, init?: ResponseInit) {\n super(body, init);\n this._cookies = new MiddlewareResponseCookies(this.headers);\n }\n\n get cookies(): ResponseCookies {\n return this._cookies;\n }\n\n /**\n * Create a JSON response.\n */\n static json<JsonBody>(body: JsonBody, init?: ResponseInit): NextResponse<JsonBody> {\n const headers = new Headers(init?.headers);\n if (!headers.has(\"content-type\")) {\n headers.set(\"content-type\", \"application/json\");\n }\n return new NextResponse(JSON.stringify(body), {\n ...init,\n headers,\n }) as NextResponse<JsonBody>;\n }\n\n /**\n * Create a redirect response.\n */\n static redirect(url: string | URL | NextURL, init?: number | ResponseInit): NextResponse {\n const status = typeof init === \"number\" ? init : (init?.status ?? 307);\n if (!REDIRECT_STATUSES.has(status)) {\n throw new RangeError(`Failed to execute \"redirect\" on \"response\": Invalid status code`);\n }\n const headers = new Headers(typeof init === \"object\" ? init?.headers : undefined);\n headers.set(\"Location\", validateURL(url));\n return new NextResponse(null, { status, headers });\n }\n\n /**\n * Create a rewrite response (middleware pattern).\n * Sets the x-middleware-rewrite header.\n */\n static rewrite(destination: string | URL | NextURL, init?: MiddlewareResponseInit): NextResponse {\n const headers = new Headers(init?.headers);\n headers.set(MIDDLEWARE_REWRITE_HEADER, validateURL(destination));\n if (init?.request?.headers) {\n encodeMiddlewareRequestHeaders(headers, init.request.headers);\n }\n return new NextResponse(null, { ...init, headers });\n }\n\n /**\n * Continue to the next handler (middleware pattern).\n * Sets the x-middleware-next header.\n */\n static next(init?: MiddlewareResponseInit): NextResponse {\n const headers = new Headers(init?.headers);\n headers.set(MIDDLEWARE_NEXT_HEADER, \"1\");\n if (init?.request?.headers) {\n encodeMiddlewareRequestHeaders(headers, init.request.headers);\n }\n return new NextResponse(null, { ...init, headers });\n }\n}\n\n// ---------------------------------------------------------------------------\n// NextURL — lightweight URL wrapper with pathname helpers\n// ---------------------------------------------------------------------------\n\nexport type NextURLConfig = {\n basePath?: string;\n nextConfig?: {\n i18n?: {\n locales: string[];\n defaultLocale: string;\n };\n /**\n * When true, `href`/`toString()` formats non-root, non-file-like pathnames\n * with a trailing slash. Matches Next.js's `formatNextPathnameInfo` so that\n * `NextResponse.redirect(request.nextUrl)` and `NextResponse.rewrite(url)`\n * honour the user's `trailingSlash` config.\n */\n trailingSlash?: boolean;\n };\n};\n\nexport class NextURL {\n /** Internal URL stores the pathname WITHOUT basePath or locale prefix. */\n private _url: URL;\n private _basePath: string;\n private _trailingSlash: boolean;\n private _locale: string | undefined;\n private _defaultLocale: string | undefined;\n private _locales: string[] | undefined;\n\n constructor(input: string | URL, base?: string | URL, config?: NextURLConfig) {\n this._url = new URL(input.toString(), base);\n this._basePath = config?.basePath ?? \"\";\n this._trailingSlash = config?.nextConfig?.trailingSlash ?? false;\n this._stripBasePath();\n const i18n = config?.nextConfig?.i18n;\n if (i18n) {\n this._locales = [...i18n.locales];\n this._defaultLocale = i18n.defaultLocale;\n this._analyzeLocale(this._locales);\n }\n }\n\n /** Strip basePath prefix from the internal pathname. */\n private _stripBasePath(): void {\n if (!this._basePath) return;\n this._url.pathname = stripBasePath(this._url.pathname, this._basePath);\n }\n\n /** Extract locale from pathname, stripping it from the internal URL. */\n private _analyzeLocale(locales: string[]): void {\n const segments = this._url.pathname.split(\"/\");\n const candidate = segments[1]?.toLowerCase();\n const match = locales.find((l) => l.toLowerCase() === candidate);\n if (match) {\n this._locale = match;\n this._url.pathname = \"/\" + segments.slice(2).join(\"/\");\n } else {\n this._locale = this._defaultLocale;\n }\n }\n\n /**\n * Reconstruct the full pathname with basePath + locale prefix and apply\n * the configured trailingSlash policy.\n * Mirrors Next.js's internal formatNextPathnameInfo().\n */\n private _formatPathname(): string {\n // Build prefix: basePath + locale (skip defaultLocale — Next.js omits it)\n let prefix = this._basePath;\n if (this._locale && this._locale !== this._defaultLocale) {\n prefix += \"/\" + this._locale;\n }\n const inner = this._url.pathname;\n const composed = !prefix ? inner : inner === \"/\" ? prefix : prefix + inner;\n return this._applyTrailingSlash(composed);\n }\n\n /**\n * Apply the configured trailingSlash policy to a composed pathname. Matches\n * Next.js's `formatNextPathnameInfo`: when `trailingSlash` is true, add a\n * trailing slash unless the path is empty/root; when false, strip a trailing\n * slash unless the path is empty/root.\n */\n private _applyTrailingSlash(pathname: string): string {\n // Never strip or add a slash to the root path.\n if (pathname === \"\" || pathname === \"/\") return pathname;\n if (this._trailingSlash) {\n return pathname.endsWith(\"/\") ? pathname : pathname + \"/\";\n }\n return pathname.endsWith(\"/\") ? pathname.slice(0, -1) : pathname;\n }\n\n get href(): string {\n const formatted = this._formatPathname();\n if (formatted === this._url.pathname) return this._url.href;\n // Replace pathname in href via string slicing — avoids URL allocation.\n // URL.href is always <origin+auth><pathname><search><hash>.\n const { href, pathname, search, hash } = this._url;\n const baseEnd = href.length - pathname.length - search.length - hash.length;\n return href.slice(0, baseEnd) + formatted + search + hash;\n }\n set href(value: string) {\n this._url.href = value;\n this._stripBasePath();\n if (this._locales) this._analyzeLocale(this._locales);\n }\n\n get origin(): string {\n return this._url.origin;\n }\n\n get protocol(): string {\n return this._url.protocol;\n }\n set protocol(value: string) {\n this._url.protocol = value;\n }\n\n get username(): string {\n return this._url.username;\n }\n set username(value: string) {\n this._url.username = value;\n }\n\n get password(): string {\n return this._url.password;\n }\n set password(value: string) {\n this._url.password = value;\n }\n\n get host(): string {\n return this._url.host;\n }\n set host(value: string) {\n this._url.host = value;\n }\n\n get hostname(): string {\n return this._url.hostname;\n }\n set hostname(value: string) {\n this._url.hostname = value;\n }\n\n get port(): string {\n return this._url.port;\n }\n set port(value: string) {\n this._url.port = value;\n }\n\n /** Returns the pathname WITHOUT basePath or locale prefix. */\n get pathname(): string {\n return this._url.pathname;\n }\n set pathname(value: string) {\n this._url.pathname = value;\n }\n\n get search(): string {\n return this._url.search;\n }\n set search(value: string) {\n this._url.search = value;\n }\n\n get searchParams(): URLSearchParams {\n return this._url.searchParams;\n }\n\n get hash(): string {\n return this._url.hash;\n }\n set hash(value: string) {\n this._url.hash = value;\n }\n\n get basePath(): string {\n return this._basePath;\n }\n set basePath(value: string) {\n this._basePath = value === \"\" ? \"\" : value.startsWith(\"/\") ? value : \"/\" + value;\n }\n\n get locale(): string {\n return this._locale ?? \"\";\n }\n set locale(value: string | undefined) {\n if (this._locales) {\n if (!value) {\n this._locale = this._defaultLocale;\n return;\n }\n if (!this._locales.includes(value)) {\n throw new TypeError(\n `The locale \"${value}\" is not in the configured locales: ${this._locales.join(\", \")}`,\n );\n }\n }\n this._locale = this._locales ? value : this._locale;\n }\n\n get defaultLocale(): string | undefined {\n return this._defaultLocale;\n }\n\n get locales(): string[] | undefined {\n return this._locales ? [...this._locales] : undefined;\n }\n\n clone(): NextURL {\n const nextConfig: NonNullable<NextURLConfig[\"nextConfig\"]> = {};\n if (this._locales) {\n nextConfig.i18n = { locales: [...this._locales], defaultLocale: this._defaultLocale! };\n }\n if (this._trailingSlash) {\n nextConfig.trailingSlash = true;\n }\n const config: NextURLConfig = {\n basePath: this._basePath,\n nextConfig: Object.keys(nextConfig).length > 0 ? nextConfig : undefined,\n };\n // Pass the full href (with locale/basePath re-added) so the constructor\n // can re-analyze and extract locale correctly.\n return new NextURL(this.href, undefined, config);\n }\n\n toString(): string {\n return this.href;\n }\n\n /**\n * The build ID of the Next.js application.\n * Set from `generateBuildId` in next.config.js, or a random UUID if not configured.\n * Can be used in middleware to detect deployment skew between client and server.\n * Matches the Next.js API: `request.nextUrl.buildId`.\n */\n get buildId(): string | undefined {\n return process.env.__VINEXT_BUILD_ID ?? undefined;\n }\n}\n\n// ---------------------------------------------------------------------------\n// Cookie helpers (minimal implementations)\n// ---------------------------------------------------------------------------\n\ntype CookieEntry = {\n name: string;\n value: string;\n};\n\nexport class RequestCookies {\n private _headers: Headers;\n private _parsed: Map<string, string>;\n\n constructor(headers: Headers) {\n this._headers = headers;\n this._parsed = parseCookieHeader(headers.get(\"cookie\") ?? \"\");\n }\n\n get(name: string): CookieEntry | undefined {\n const value = this._parsed.get(name);\n return value !== undefined ? { name, value } : undefined;\n }\n\n getAll(nameOrOptions?: string | CookieEntry): CookieEntry[] {\n const name = typeof nameOrOptions === \"string\" ? nameOrOptions : nameOrOptions?.name;\n return [...this._parsed.entries()]\n .filter(([cookieName]) => name === undefined || cookieName === name)\n .map(([cookieName, value]) => ({ name: cookieName, value }));\n }\n\n has(name: string): boolean {\n return this._parsed.has(name);\n }\n\n set(nameOrOptions: string | CookieEntry, value?: string): this {\n let cookieName: string;\n let cookieValue: string;\n if (typeof nameOrOptions === \"string\") {\n cookieName = nameOrOptions;\n cookieValue = value ?? \"\";\n } else {\n cookieName = nameOrOptions.name;\n cookieValue = nameOrOptions.value;\n }\n validateCookieName(cookieName);\n this._parsed.set(cookieName, cookieValue);\n this._syncHeader();\n return this;\n }\n\n delete(names: string | string[]): boolean | boolean[] {\n if (Array.isArray(names)) {\n const results = names.map((name) => {\n validateCookieName(name);\n return this._parsed.delete(name);\n });\n this._syncHeader();\n return results;\n }\n validateCookieName(names);\n const result = this._parsed.delete(names);\n this._syncHeader();\n return result;\n }\n\n clear(): this {\n this._parsed.clear();\n this._syncHeader();\n return this;\n }\n\n get size(): number {\n return this._parsed.size;\n }\n\n toString(): string {\n return this._serialize();\n }\n\n private _serialize(): string {\n return [...this._parsed.entries()].map(([n, v]) => `${n}=${encodeURIComponent(v)}`).join(\"; \");\n }\n\n private _syncHeader(): void {\n if (this._parsed.size === 0) {\n this._headers.delete(\"cookie\");\n } else {\n this._headers.set(\"cookie\", this._serialize());\n }\n }\n\n [Symbol.iterator](): IterableIterator<[string, CookieEntry]> {\n const entries = this.getAll().map((c) => [c.name, c] as [string, CookieEntry]);\n return entries[Symbol.iterator]();\n }\n}\n\n// Keep this error message in sync with headers.ts. This adapter backs\n// NextRequest cookies, while headers.ts owns the next/headers cookies object.\nclass ReadonlyRequestCookiesError extends Error {\n constructor() {\n super(\n \"Cookies can only be modified in a Server Action or Route Handler. Read more: https://nextjs.org/docs/app/api-reference/functions/cookies#options\",\n );\n }\n\n static callable(this: void): never {\n throw new ReadonlyRequestCookiesError();\n }\n}\n\nconst REQUEST_HEADERS_MUTATING_METHODS = new Set([\"set\", \"delete\", \"append\"]);\n\n// Keep this error message in sync with headers.ts. This adapter backs\n// NextRequest headers in force-static route handlers, while headers.ts owns the\n// next/headers object.\nclass ReadonlyRequestHeadersError extends Error {\n constructor() {\n super(\n \"Headers cannot be modified. Read more: https://nextjs.org/docs/app/api-reference/functions/headers\",\n );\n }\n\n static callable(this: void): never {\n throw new ReadonlyRequestHeadersError();\n }\n}\n\nexport function sealRequestHeaders(headers: Headers): Headers {\n return new Proxy<Headers>(headers, {\n get(target, prop) {\n if (typeof prop === \"string\" && REQUEST_HEADERS_MUTATING_METHODS.has(prop)) {\n return ReadonlyRequestHeadersError.callable;\n }\n\n const value = Reflect.get(target, prop, target);\n return typeof value === \"function\" ? value.bind(target) : value;\n },\n });\n}\n\nexport function sealRequestCookies(cookies: RequestCookies): RequestCookies {\n return new Proxy<RequestCookies>(cookies, {\n get(target, prop) {\n if (prop === \"set\" || prop === \"delete\" || prop === \"clear\") {\n return ReadonlyRequestCookiesError.callable;\n }\n\n const value = Reflect.get(target, prop, target);\n return typeof value === \"function\" ? value.bind(target) : value;\n },\n });\n}\n\nexport class ResponseCookies {\n private _headers: Headers;\n /** Internal map keyed by cookie name — single source of truth. */\n private _parsed: Map<string, { serialized: string; entry: CookieEntry }> = new Map();\n\n constructor(headers: Headers) {\n this._headers = headers;\n\n // Hydrate internal map from any existing Set-Cookie headers\n for (const header of headers.getSetCookie()) {\n const eq = header.indexOf(\"=\");\n if (eq === -1) continue;\n const cookieName = header.slice(0, eq);\n const semi = header.indexOf(\";\", eq);\n const raw = header.slice(eq + 1, semi === -1 ? undefined : semi);\n let value: string;\n try {\n value = decodeURIComponent(raw);\n } catch {\n value = raw;\n }\n this._parsed.set(cookieName, { serialized: header, entry: { name: cookieName, value } });\n }\n }\n\n set(\n ...args:\n | [name: string, value: string, options?: CookieOptions]\n | [options: CookieOptions & { name: string; value: string }]\n ): this {\n const [name, value, opts] = parseCookieSetArgs(args);\n validateCookieName(name);\n\n const serialized = serializeSetCookie(name, value, opts);\n this._parsed.set(name, { serialized, entry: { name, value } });\n this._syncHeaders();\n return this;\n }\n\n get(...args: [name: string] | [options: { name: string }]): CookieEntry | undefined {\n const key = typeof args[0] === \"string\" ? args[0] : args[0].name;\n return this._parsed.get(key)?.entry;\n }\n\n has(name: string): boolean {\n return this._parsed.has(name);\n }\n\n getAll(...args: [name: string] | [options: { name: string }] | []): CookieEntry[] {\n const all = [...this._parsed.values()].map((v) => v.entry);\n if (args.length === 0) return all;\n const key = typeof args[0] === \"string\" ? args[0] : args[0].name;\n return all.filter((c) => c.name === key);\n }\n\n delete(\n ...args:\n | [name: string]\n | [options: Omit<CookieOptions & { name: string }, \"maxAge\" | \"expires\">]\n ): this {\n const [name, opts] =\n typeof args[0] === \"string\" ? [args[0], undefined] : [args[0].name, args[0]];\n return this.set({\n name,\n value: \"\",\n expires: new Date(0),\n path: opts?.path,\n domain: opts?.domain,\n httpOnly: opts?.httpOnly,\n secure: opts?.secure,\n sameSite: opts?.sameSite,\n });\n }\n\n [Symbol.iterator](): IterableIterator<[string, CookieEntry]> {\n const entries: [string, CookieEntry][] = [...this._parsed.values()].map((v) => [\n v.entry.name,\n v.entry,\n ]);\n return entries[Symbol.iterator]();\n }\n\n /** Delete all Set-Cookie headers and re-append from the internal map. */\n private _syncHeaders(): void {\n this._headers.delete(\"Set-Cookie\");\n for (const { serialized } of this._parsed.values()) {\n this._headers.append(\"Set-Cookie\", serialized);\n }\n }\n}\n\nclass MiddlewareResponseCookies extends ResponseCookies {\n private _responseHeaders: Headers;\n\n constructor(headers: Headers) {\n super(headers);\n this._responseHeaders = headers;\n }\n\n override set(\n ...args:\n | [name: string, value: string, options?: CookieOptions]\n | [options: CookieOptions & { name: string; value: string }]\n ): this {\n super.set(...args);\n this._syncMiddlewareCookieHeader();\n return this;\n }\n\n override delete(\n ...args:\n | [name: string]\n | [options: Omit<CookieOptions & { name: string }, \"maxAge\" | \"expires\">]\n ): this {\n super.delete(...args);\n this._syncMiddlewareCookieHeader();\n return this;\n }\n\n private _syncMiddlewareCookieHeader(): void {\n const cookies = this._responseHeaders.getSetCookie();\n if (cookies.length === 0) {\n this._responseHeaders.delete(MIDDLEWARE_SET_COOKIE_HEADER);\n return;\n }\n\n this._responseHeaders.set(MIDDLEWARE_SET_COOKIE_HEADER, cookies.join(\",\"));\n }\n}\n\ntype CookieOptions = {\n path?: string;\n domain?: string;\n maxAge?: number;\n expires?: Date;\n httpOnly?: boolean;\n secure?: boolean;\n sameSite?: \"Strict\" | \"Lax\" | \"None\";\n};\n\n/**\n * Parse the overloaded arguments for ResponseCookies.set():\n * - (name, value, options?) — positional form\n * - ({ name, value, ...options }) — object form\n */\nfunction parseCookieSetArgs(\n args:\n | [name: string, value: string, options?: CookieOptions]\n | [options: CookieOptions & { name: string; value: string }],\n): [string, string, CookieOptions | undefined] {\n if (typeof args[0] === \"string\") {\n return [args[0], args[1] as string, args[2] as CookieOptions | undefined];\n }\n const { name, value, ...opts } = args[0];\n return [name, value, opts as CookieOptions];\n}\n\n// ---------------------------------------------------------------------------\n// Types\n// ---------------------------------------------------------------------------\n\nexport type MiddlewareResponseInit = {\n request?: {\n headers?: Headers;\n };\n} & ResponseInit;\n\nexport type NextMiddlewareResult = NextResponse | Response | null | undefined | void;\n\nexport type NextMiddleware = (\n request: NextRequest,\n event: NextFetchEvent,\n) => NextMiddlewareResult | Promise<NextMiddlewareResult>;\n\n/**\n * Minimal NextFetchEvent — extends FetchEvent where available,\n * otherwise provides the waitUntil pattern standalone.\n */\nexport class NextFetchEvent {\n sourcePage: string;\n private _waitUntilPromises: Promise<unknown>[] = [];\n\n constructor(params: { page: string }) {\n this.sourcePage = params.page;\n }\n\n waitUntil(promise: Promise<unknown>): void {\n this._waitUntilPromises.push(promise);\n }\n\n get waitUntilPromises(): Promise<unknown>[] {\n return this._waitUntilPromises;\n }\n\n /** Drain all waitUntil promises. Returns a single promise that settles when all are done. */\n drainWaitUntil(): Promise<PromiseSettledResult<unknown>[]> {\n return Promise.allSettled(this._waitUntilPromises);\n }\n}\n\n// ---------------------------------------------------------------------------\n// Utility exports\n// ---------------------------------------------------------------------------\n\n/**\n * Parse user agent string. Minimal implementation — for full UA parsing,\n * apps should use a dedicated library like `ua-parser-js`.\n */\nexport function userAgentFromString(ua: string | undefined): UserAgent {\n const input = ua ?? \"\";\n return {\n isBot: /bot|crawler|spider|crawling/i.test(input),\n ua: input,\n browser: {},\n device: {},\n engine: {},\n os: {},\n cpu: {},\n };\n}\n\nexport function userAgent({ headers }: { headers: Headers }): UserAgent {\n return userAgentFromString(headers.get(\"user-agent\") ?? undefined);\n}\n\nexport type UserAgent = {\n isBot: boolean;\n ua: string;\n browser: { name?: string; version?: string; major?: string };\n device: { model?: string; type?: string; vendor?: string };\n engine: { name?: string; version?: string };\n os: { name?: string; version?: string };\n cpu: { architecture?: string };\n};\n\n/**\n * after() — schedule work after the response is sent.\n *\n * Uses the platform's `waitUntil` (via the per-request ExecutionContext) when\n * available so the task survives past the response on Cloudflare Workers.\n * Falls back to a fire-and-forget microtask on runtimes without an execution\n * context (e.g. Node.js dev server).\n *\n * Throws when called inside a cached scope — request-specific\n * side-effects must not leak into cached results.\n */\nexport function after<T>(task: Promise<T> | (() => T | Promise<T>)): void {\n _throwIfInsideCacheScope(\"after()\");\n\n const promise = typeof task === \"function\" ? Promise.resolve().then(task) : task;\n // NOTE: vinext runs function tasks concurrently with response streaming (next microtask),\n // whereas Next.js queues them to run strictly after the response is sent via onClose.\n // This is a known simplification — function tasks here are not guaranteed to run\n // after the response completes, only after the current synchronous execution.\n //\n // `.catch()` is attached synchronously in the same tick as `promise` is created, so\n // there is no window where a pre-rejected `task` promise could trigger an\n // `unhandledrejection` event before the handler is in place.\n const guarded = promise.catch((err) => {\n console.error(\"[vinext] after() task failed:\", err);\n });\n\n // TODO: Next.js throws when after() is called outside a request context or when\n // waitUntil is unavailable, preventing silent task loss. vinext falls back to\n // fire-and-forget here, which is correct for the Node.js dev server (where\n // getRequestExecutionContext() always returns null). On Workers, a misconfigured\n // entry that omits runWithExecutionContext would silently drop tasks — consider\n // a one-time console.warn on the fallback path, gated to production only (e.g.\n // `process.env.NODE_ENV === 'production'` or `typeof caches !== 'undefined'` for\n // a Workers runtime check) with a module-level `let _warned = false` guard so it\n // fires at most once and doesn't spam the dev-server console.\n getRequestExecutionContext()?.waitUntil(guarded);\n}\n\n/**\n * connection() — signals that the response requires a live connection\n * (not a static/cached response). Opts the page out of ISR caching\n * and sets Cache-Control: no-store on the response.\n */\nexport async function connection(): Promise<void> {\n const { markDynamicUsage, markRenderRequestApiUsage, throwIfInsideCacheScope } =\n await import(\"./headers.js\");\n markRenderRequestApiUsage(\"connection\");\n throwIfInsideCacheScope(\"connection()\");\n markDynamicUsage();\n}\n\n/**\n * URLPattern re-export — used in middleware for route matching.\n * Available natively in Node 20+, Cloudflare Workers, Deno.\n * Falls back to urlpattern-polyfill if the global is not available.\n */\nexport const URLPattern: typeof globalThis.URLPattern =\n globalThis.URLPattern ??\n (() => {\n throw new Error(\n \"URLPattern is not available in this runtime. \" +\n \"Install the `urlpattern-polyfill` package or upgrade to Node 20+.\",\n );\n });\n"],"mappings":"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"}
|
|
@@ -1,6 +1,7 @@
|
|
|
1
1
|
//#region src/shims/url-utils.d.ts
|
|
2
2
|
declare function isAbsoluteUrl(url: string): boolean;
|
|
3
3
|
declare function isAbsoluteOrProtocolRelativeUrl(url: string): boolean;
|
|
4
|
+
declare function getWindowOrigin(): string | null;
|
|
4
5
|
/**
|
|
5
6
|
* If `url` is an absolute same-origin URL, return the local path
|
|
6
7
|
* (pathname + search + hash). Returns null for truly external URLs
|
|
@@ -47,5 +48,5 @@ declare function resolveRelativeHref(href: string, currentUrl?: string, basePath
|
|
|
47
48
|
declare function toBrowserNavigationHref(href: string, currentUrl?: string, basePath?: string): string;
|
|
48
49
|
declare function isHashOnlyBrowserUrlChange(href: string, currentHref: string, basePath?: string): boolean;
|
|
49
50
|
//#endregion
|
|
50
|
-
export { isAbsoluteOrProtocolRelativeUrl, isAbsoluteUrl, isHashOnlyBrowserUrlChange, normalizePathTrailingSlash, resolveRelativeHref, toBrowserNavigationHref, toSameOriginAppPath, toSameOriginPath, withBasePath };
|
|
51
|
+
export { getWindowOrigin, isAbsoluteOrProtocolRelativeUrl, isAbsoluteUrl, isHashOnlyBrowserUrlChange, normalizePathTrailingSlash, resolveRelativeHref, toBrowserNavigationHref, toSameOriginAppPath, toSameOriginPath, withBasePath };
|
|
51
52
|
//# sourceMappingURL=url-utils.d.ts.map
|