@rangojs/router 0.0.0-experimental.ea6d5eec → 0.0.0-experimental.ede38110
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 +76 -18
- package/dist/bin/rango.js +130 -47
- package/dist/vite/index.js +719 -240
- package/package.json +3 -3
- package/skills/cache-guide/SKILL.md +32 -0
- package/skills/caching/SKILL.md +8 -0
- package/skills/handler-use/SKILL.md +362 -0
- package/skills/intercept/SKILL.md +20 -0
- package/skills/layout/SKILL.md +22 -0
- package/skills/links/SKILL.md +3 -1
- package/skills/loader/SKILL.md +53 -43
- package/skills/middleware/SKILL.md +34 -3
- package/skills/migrate-nextjs/SKILL.md +560 -0
- package/skills/migrate-react-router/SKILL.md +764 -0
- package/skills/parallel/SKILL.md +185 -0
- package/skills/prerender/SKILL.md +110 -68
- package/skills/rango/SKILL.md +24 -22
- package/skills/route/SKILL.md +55 -0
- package/skills/router-setup/SKILL.md +87 -2
- package/skills/typesafety/SKILL.md +10 -0
- package/src/__internal.ts +1 -1
- package/src/browser/app-version.ts +14 -0
- package/src/browser/event-controller.ts +5 -0
- package/src/browser/navigation-bridge.ts +37 -5
- package/src/browser/navigation-client.ts +107 -75
- package/src/browser/navigation-store.ts +43 -8
- package/src/browser/partial-update.ts +51 -6
- package/src/browser/prefetch/cache.ts +22 -12
- package/src/browser/prefetch/fetch.ts +81 -20
- package/src/browser/prefetch/queue.ts +61 -29
- package/src/browser/prefetch/resource-ready.ts +77 -0
- package/src/browser/react/Link.tsx +67 -8
- package/src/browser/react/NavigationProvider.tsx +13 -4
- package/src/browser/react/context.ts +7 -2
- package/src/browser/react/use-handle.ts +9 -58
- package/src/browser/react/use-navigation.ts +11 -10
- package/src/browser/react/use-router.ts +21 -8
- package/src/browser/rsc-router.tsx +45 -3
- package/src/browser/scroll-restoration.ts +10 -8
- package/src/browser/segment-reconciler.ts +36 -9
- package/src/browser/server-action-bridge.ts +8 -6
- package/src/browser/types.ts +27 -5
- package/src/build/generate-manifest.ts +6 -6
- package/src/build/generate-route-types.ts +3 -0
- package/src/build/route-trie.ts +50 -24
- package/src/build/route-types/include-resolution.ts +8 -1
- package/src/build/route-types/router-processing.ts +211 -72
- package/src/build/route-types/scan-filter.ts +8 -1
- package/src/cache/cache-runtime.ts +15 -11
- package/src/cache/cache-scope.ts +46 -5
- package/src/cache/document-cache.ts +17 -7
- package/src/cache/taint.ts +55 -0
- package/src/client.tsx +84 -230
- package/src/context-var.ts +72 -2
- package/src/debug.ts +2 -2
- package/src/handle.ts +40 -0
- package/src/index.rsc.ts +3 -1
- package/src/index.ts +46 -6
- package/src/prerender/store.ts +5 -4
- package/src/prerender.ts +138 -77
- package/src/reverse.ts +25 -1
- package/src/route-definition/dsl-helpers.ts +224 -37
- package/src/route-definition/helpers-types.ts +67 -19
- package/src/route-definition/index.ts +3 -0
- package/src/route-definition/redirect.ts +9 -1
- package/src/route-definition/resolve-handler-use.ts +149 -0
- package/src/route-types.ts +18 -0
- package/src/router/content-negotiation.ts +100 -1
- package/src/router/handler-context.ts +82 -23
- package/src/router/intercept-resolution.ts +9 -4
- package/src/router/lazy-includes.ts +7 -6
- package/src/router/loader-resolution.ts +156 -21
- package/src/router/logging.ts +1 -1
- package/src/router/manifest.ts +28 -15
- package/src/router/match-api.ts +124 -189
- package/src/router/match-middleware/background-revalidation.ts +30 -2
- package/src/router/match-middleware/cache-lookup.ts +94 -17
- package/src/router/match-middleware/cache-store.ts +53 -10
- package/src/router/match-middleware/intercept-resolution.ts +9 -7
- package/src/router/match-middleware/segment-resolution.ts +60 -5
- package/src/router/match-result.ts +104 -10
- package/src/router/metrics.ts +6 -1
- package/src/router/middleware-types.ts +6 -8
- package/src/router/middleware.ts +2 -5
- package/src/router/navigation-snapshot.ts +182 -0
- package/src/router/prerender-match.ts +110 -10
- package/src/router/preview-match.ts +30 -102
- package/src/router/request-classification.ts +310 -0
- package/src/router/route-snapshot.ts +245 -0
- package/src/router/router-context.ts +1 -0
- package/src/router/router-interfaces.ts +36 -4
- package/src/router/router-options.ts +37 -11
- package/src/router/segment-resolution/fresh.ts +198 -20
- package/src/router/segment-resolution/helpers.ts +29 -24
- package/src/router/segment-resolution/loader-cache.ts +1 -0
- package/src/router/segment-resolution/revalidation.ts +433 -296
- package/src/router/types.ts +1 -0
- package/src/router.ts +55 -6
- package/src/rsc/handler.ts +472 -372
- package/src/rsc/loader-fetch.ts +23 -3
- package/src/rsc/manifest-init.ts +5 -1
- package/src/rsc/progressive-enhancement.ts +14 -2
- package/src/rsc/rsc-rendering.ts +10 -1
- package/src/rsc/server-action.ts +8 -0
- package/src/rsc/ssr-setup.ts +2 -2
- package/src/rsc/types.ts +9 -1
- package/src/segment-content-promise.ts +67 -0
- package/src/segment-loader-promise.ts +122 -0
- package/src/segment-system.tsx +109 -23
- package/src/server/context.ts +166 -17
- package/src/server/handle-store.ts +19 -0
- package/src/server/loader-registry.ts +9 -8
- package/src/server/request-context.ts +175 -15
- package/src/ssr/index.tsx +4 -0
- package/src/static-handler.ts +18 -6
- package/src/types/cache-types.ts +4 -4
- package/src/types/handler-context.ts +137 -33
- package/src/types/loader-types.ts +36 -9
- package/src/types/route-entry.ts +12 -1
- package/src/types/segments.ts +2 -0
- package/src/urls/include-helper.ts +24 -14
- package/src/urls/path-helper-types.ts +39 -6
- package/src/urls/path-helper.ts +48 -13
- package/src/urls/pattern-types.ts +12 -0
- package/src/urls/response-types.ts +16 -6
- package/src/use-loader.tsx +77 -5
- package/src/vite/discovery/bundle-postprocess.ts +30 -33
- package/src/vite/discovery/discover-routers.ts +5 -1
- package/src/vite/discovery/prerender-collection.ts +128 -74
- package/src/vite/discovery/state.ts +13 -4
- package/src/vite/index.ts +4 -0
- package/src/vite/plugin-types.ts +60 -5
- package/src/vite/plugins/expose-id-utils.ts +12 -0
- package/src/vite/plugins/expose-ids/handler-transform.ts +30 -0
- package/src/vite/plugins/expose-internal-ids.ts +257 -40
- package/src/vite/plugins/performance-tracks.ts +88 -0
- package/src/vite/plugins/refresh-cmd.ts +88 -26
- package/src/vite/rango.ts +19 -2
- package/src/vite/router-discovery.ts +178 -37
- package/src/vite/utils/banner.ts +3 -3
- package/src/vite/utils/prerender-utils.ts +37 -5
- package/src/vite/utils/shared-utils.ts +3 -2
|
@@ -7,6 +7,7 @@
|
|
|
7
7
|
import type { ReactNode } from "react";
|
|
8
8
|
import { track } from "../server/context";
|
|
9
9
|
import type { EntryData } from "../server/context";
|
|
10
|
+
import { contextGet } from "../context-var.js";
|
|
10
11
|
import type {
|
|
11
12
|
ResolvedSegment,
|
|
12
13
|
HandlerContext,
|
|
@@ -19,10 +20,11 @@ import type {
|
|
|
19
20
|
ErrorInfo,
|
|
20
21
|
} from "../types";
|
|
21
22
|
import type { LoaderRevalidationResult, ActionContext } from "./types";
|
|
22
|
-
import { isHandle, type Handle } from "../handle.js";
|
|
23
|
-
import
|
|
23
|
+
import { isHandle, collectHandleData, type Handle } from "../handle.js";
|
|
24
|
+
import { buildHandleSnapshot } from "../server/handle-store.js";
|
|
24
25
|
import { getFetchableLoader } from "../server/fetchable-loader-store.js";
|
|
25
26
|
import { _getRequestContext } from "../server/request-context.js";
|
|
27
|
+
import { isInsideLoaderScope } from "../server/context.js";
|
|
26
28
|
import { debugLog } from "./logging.js";
|
|
27
29
|
|
|
28
30
|
/**
|
|
@@ -241,6 +243,21 @@ function createLoaderExecutor<TEnv>(
|
|
|
241
243
|
pendingLoaders.add(loader.$$id);
|
|
242
244
|
|
|
243
245
|
const currentLoaderId = loader.$$id;
|
|
246
|
+
const variables = (ctx as InternalHandlerContext<any, TEnv>)._variables;
|
|
247
|
+
|
|
248
|
+
// Capture whether this loader is being started from a DSL loader scope
|
|
249
|
+
// (runInsideLoaderScope in fresh.ts). Handler-invoked loaders are NOT
|
|
250
|
+
// inside loader scope. This determines whether rendered() is allowed.
|
|
251
|
+
const isDslLoader = isInsideLoaderScope();
|
|
252
|
+
|
|
253
|
+
let renderedResolved = false;
|
|
254
|
+
let renderedPromise: Promise<void> | null = null;
|
|
255
|
+
|
|
256
|
+
// Loader functions are always fresh (never cached), so they get an
|
|
257
|
+
// unguarded get that bypasses non-cacheable read guards. This applies
|
|
258
|
+
// to ALL loaders — DSL and handler-called — because the loader
|
|
259
|
+
// function itself always re-executes. Also handles nested deps
|
|
260
|
+
// (loaderA → use(loaderB)) since all share this unguarded get.
|
|
244
261
|
const loaderCtx: LoaderContext<Record<string, string | undefined>, TEnv> = {
|
|
245
262
|
params: ctx.params,
|
|
246
263
|
routeParams: (ctx.params ?? {}) as Record<string, string>,
|
|
@@ -250,16 +267,86 @@ function createLoaderExecutor<TEnv>(
|
|
|
250
267
|
pathname: ctx.pathname,
|
|
251
268
|
url: ctx.url,
|
|
252
269
|
env: ctx.env,
|
|
253
|
-
|
|
254
|
-
|
|
255
|
-
use: <
|
|
256
|
-
|
|
257
|
-
|
|
258
|
-
|
|
259
|
-
|
|
270
|
+
get: ((keyOrVar: any) =>
|
|
271
|
+
contextGet(variables, keyOrVar)) as typeof ctx.get,
|
|
272
|
+
use: ((item: LoaderDefinition<any, any> | Handle<any, any>) => {
|
|
273
|
+
if (isHandle(item)) {
|
|
274
|
+
if (!renderedResolved) {
|
|
275
|
+
throw new Error(
|
|
276
|
+
`ctx.use(handle) in a loader requires "await ctx.rendered()" first. ` +
|
|
277
|
+
`Handle "${item.$$id}" cannot be read until the render tree has settled.`,
|
|
278
|
+
);
|
|
279
|
+
}
|
|
280
|
+
const reqCtx = reqCtxRef ?? _getRequestContext();
|
|
281
|
+
if (!reqCtx) {
|
|
282
|
+
throw new Error(
|
|
283
|
+
`ctx.use(handle) failed: request context not available.`,
|
|
284
|
+
);
|
|
285
|
+
}
|
|
286
|
+
const segmentOrder = reqCtx._renderBarrierSegmentOrder ?? [];
|
|
287
|
+
const snapshot =
|
|
288
|
+
reqCtx._renderBarrierHandleSnapshot ??
|
|
289
|
+
buildHandleSnapshot(reqCtx._handleStore, segmentOrder);
|
|
290
|
+
return collectHandleData(item, snapshot, segmentOrder);
|
|
291
|
+
}
|
|
292
|
+
|
|
293
|
+
// Loader case
|
|
294
|
+
return useLoader(item as LoaderDefinition<any, any>, currentLoaderId);
|
|
295
|
+
}) as LoaderContext["use"],
|
|
260
296
|
method: "GET",
|
|
261
297
|
body: undefined,
|
|
262
298
|
reverse: ctx.reverse as LoaderContext["reverse"],
|
|
299
|
+
rendered: (): Promise<void> => {
|
|
300
|
+
// Guard: only DSL loaders may use rendered()
|
|
301
|
+
if (!isDslLoader) {
|
|
302
|
+
throw new Error(
|
|
303
|
+
`ctx.rendered() is only available in DSL loaders (registered via loader() in urls()). ` +
|
|
304
|
+
`Handler-invoked loaders (ctx.use(Loader) inside a handler) cannot use rendered().`,
|
|
305
|
+
);
|
|
306
|
+
}
|
|
307
|
+
|
|
308
|
+
// Guard: reject streaming trees
|
|
309
|
+
const reqCtx = reqCtxRef ?? _getRequestContext();
|
|
310
|
+
if (reqCtx?._treeHasStreaming) {
|
|
311
|
+
throw new Error(
|
|
312
|
+
`ctx.rendered() is not supported when the matched route tree uses loading(). ` +
|
|
313
|
+
`Streaming handlers may not have settled when rendered() resolves. ` +
|
|
314
|
+
`Remove loading() from the route tree or restructure to avoid rendered().`,
|
|
315
|
+
);
|
|
316
|
+
}
|
|
317
|
+
|
|
318
|
+
if (renderedPromise) return renderedPromise;
|
|
319
|
+
|
|
320
|
+
if (!reqCtx) {
|
|
321
|
+
throw new Error(
|
|
322
|
+
`ctx.rendered() failed: request context not available.`,
|
|
323
|
+
);
|
|
324
|
+
}
|
|
325
|
+
|
|
326
|
+
// Bidirectional deadlock check: if a handler already started
|
|
327
|
+
// awaiting this loader, calling rendered() would deadlock.
|
|
328
|
+
if (reqCtx._handlerLoaderDeps?.has(currentLoaderId)) {
|
|
329
|
+
throw new Error(
|
|
330
|
+
`Deadlock: loader "${currentLoaderId}" called ctx.rendered() but a handler ` +
|
|
331
|
+
`is already awaiting this loader via ctx.use(). The handler blocks ` +
|
|
332
|
+
`segment resolution, which blocks the barrier, which blocks this loader. ` +
|
|
333
|
+
`Move the data dependency to a loader-to-loader pattern instead.`,
|
|
334
|
+
);
|
|
335
|
+
}
|
|
336
|
+
|
|
337
|
+
// Register this loader as waiting for the barrier so that
|
|
338
|
+
// setupLoaderAccess can detect deadlocks when a handler
|
|
339
|
+
// tries to await the same loader via ctx.use().
|
|
340
|
+
if (!reqCtx._renderBarrierWaiters) {
|
|
341
|
+
reqCtx._renderBarrierWaiters = new Set();
|
|
342
|
+
}
|
|
343
|
+
reqCtx._renderBarrierWaiters.add(currentLoaderId);
|
|
344
|
+
|
|
345
|
+
renderedPromise = reqCtx._renderBarrier.then(() => {
|
|
346
|
+
renderedResolved = true;
|
|
347
|
+
});
|
|
348
|
+
return renderedPromise;
|
|
349
|
+
},
|
|
263
350
|
};
|
|
264
351
|
|
|
265
352
|
const doneLoader = track(`loader:${loader.$$id}`, 2);
|
|
@@ -290,15 +377,22 @@ export function setupLoaderAccess<TEnv>(
|
|
|
290
377
|
ctx: HandlerContext<any, TEnv>,
|
|
291
378
|
loaderPromises: Map<string, Promise<any>>,
|
|
292
379
|
): void {
|
|
293
|
-
// Eagerly capture the HandleStore at setup time
|
|
294
|
-
// In workerd/Cloudflare, dynamic imports and
|
|
295
|
-
// can disrupt AsyncLocalStorage, causing
|
|
296
|
-
// undefined when handlers later call
|
|
297
|
-
//
|
|
298
|
-
const
|
|
380
|
+
// Eagerly capture the request context and HandleStore at setup time
|
|
381
|
+
// (before pipeline async ops). In workerd/Cloudflare, dynamic imports and
|
|
382
|
+
// fetch() in the match pipeline can disrupt AsyncLocalStorage, causing
|
|
383
|
+
// getRequestContext() to return undefined when handlers later call
|
|
384
|
+
// ctx.use(handle). Capturing early ensures references survive ALS disruption.
|
|
385
|
+
const reqCtxRef = _getRequestContext();
|
|
386
|
+
const handleStoreRef = reqCtxRef?._handleStore;
|
|
299
387
|
|
|
300
388
|
const useLoader = createLoaderExecutor(ctx, loaderPromises);
|
|
301
389
|
|
|
390
|
+
// Track whether we're inside a handle push callback. Loaders started
|
|
391
|
+
// from push callbacks (e.g. push(async () => ctx.use(Loader))) do NOT
|
|
392
|
+
// block segment resolution, so they must not be registered as handler
|
|
393
|
+
// dependencies for deadlock detection.
|
|
394
|
+
let insideHandlePush = false;
|
|
395
|
+
|
|
302
396
|
ctx.use = ((item: LoaderDefinition<any, any> | Handle<any, any>) => {
|
|
303
397
|
if (isHandle(item)) {
|
|
304
398
|
const handle = item;
|
|
@@ -318,16 +412,57 @@ export function setupLoaderAccess<TEnv>(
|
|
|
318
412
|
) => {
|
|
319
413
|
if (!store) return;
|
|
320
414
|
|
|
321
|
-
|
|
322
|
-
|
|
323
|
-
|
|
324
|
-
|
|
415
|
+
if (typeof dataOrFn === "function") {
|
|
416
|
+
// Mark scope so ctx.use(loader) calls inside the callback
|
|
417
|
+
// are not registered as handler-to-loader deps.
|
|
418
|
+
insideHandlePush = true;
|
|
419
|
+
try {
|
|
420
|
+
const result = (dataOrFn as () => Promise<unknown>)();
|
|
421
|
+
store.push(handle.$$id, segmentId, result);
|
|
422
|
+
} finally {
|
|
423
|
+
insideHandlePush = false;
|
|
424
|
+
}
|
|
425
|
+
return;
|
|
426
|
+
}
|
|
325
427
|
|
|
326
|
-
store.push(handle.$$id, segmentId,
|
|
428
|
+
store.push(handle.$$id, segmentId, dataOrFn);
|
|
327
429
|
};
|
|
328
430
|
}
|
|
329
431
|
|
|
330
|
-
|
|
432
|
+
// Deadlock guard and handler-to-loader dependency tracking.
|
|
433
|
+
// Skip when inside a DSL loader scope (resolveLoaderData also calls
|
|
434
|
+
// ctx.use() but that's DSL-to-DSL, not handler-to-loader) or when
|
|
435
|
+
// inside a handle push callback (push callbacks don't block segment
|
|
436
|
+
// resolution so they can't cause rendered() deadlocks).
|
|
437
|
+
const loader = item as LoaderDefinition<any, any>;
|
|
438
|
+
if (!isInsideLoaderScope() && !insideHandlePush) {
|
|
439
|
+
const reqCtx = reqCtxRef ?? _getRequestContext();
|
|
440
|
+
if (reqCtx) {
|
|
441
|
+
// Direction 1: handler awaits loader that already called rendered()
|
|
442
|
+
if (
|
|
443
|
+
loaderPromises.has(loader.$$id) &&
|
|
444
|
+
reqCtx._renderBarrierWaiters?.has(loader.$$id)
|
|
445
|
+
) {
|
|
446
|
+
throw new Error(
|
|
447
|
+
`Deadlock: handler is awaiting loader "${loader.$$id}" which called ctx.rendered(). ` +
|
|
448
|
+
`The loader is waiting for segment resolution, but the handler blocks resolution. ` +
|
|
449
|
+
`Move the data dependency to a loader-to-loader pattern instead.`,
|
|
450
|
+
);
|
|
451
|
+
}
|
|
452
|
+
// Direction 2: track dep so rendered() can detect the deadlock
|
|
453
|
+
// if the loader calls it later. Skip when the barrier has already
|
|
454
|
+
// resolved — no deadlock is possible (rendered() resolves immediately).
|
|
455
|
+
// _renderBarrierSegmentOrder is undefined before resolution, string[]
|
|
456
|
+
// after. This also prevents false positives from handle push callbacks
|
|
457
|
+
// that resume after their first await (post-barrier-resolution).
|
|
458
|
+
if (reqCtx._renderBarrierSegmentOrder === undefined) {
|
|
459
|
+
if (!reqCtx._handlerLoaderDeps) reqCtx._handlerLoaderDeps = new Set();
|
|
460
|
+
reqCtx._handlerLoaderDeps.add(loader.$$id);
|
|
461
|
+
}
|
|
462
|
+
}
|
|
463
|
+
}
|
|
464
|
+
|
|
465
|
+
return useLoader(loader, null);
|
|
331
466
|
}) as typeof ctx.use;
|
|
332
467
|
}
|
|
333
468
|
|
package/src/router/logging.ts
CHANGED
|
@@ -74,7 +74,7 @@ function getHeaderRequestId(request: Request): string | null {
|
|
|
74
74
|
return trimmed.length > 0 ? trimmed : null;
|
|
75
75
|
}
|
|
76
76
|
|
|
77
|
-
function getOrCreateRequestId(request: Request): string {
|
|
77
|
+
export function getOrCreateRequestId(request: Request): string {
|
|
78
78
|
const existing = requestIds.get(request);
|
|
79
79
|
if (existing) return existing;
|
|
80
80
|
|
package/src/router/manifest.ts
CHANGED
|
@@ -9,6 +9,7 @@ import { createRouteHelpers } from "../route-definition";
|
|
|
9
9
|
import {
|
|
10
10
|
getContext,
|
|
11
11
|
runWithPrefixes,
|
|
12
|
+
getIsolatedLazyParent,
|
|
12
13
|
type EntryData,
|
|
13
14
|
type MetricsStore,
|
|
14
15
|
} from "../server/context";
|
|
@@ -114,36 +115,48 @@ export async function loadManifest(
|
|
|
114
115
|
// This ensures routes are registered under the correct layout hierarchy
|
|
115
116
|
const lazyContext =
|
|
116
117
|
entry.lazy && entry.lazyPatterns ? entry.lazyContext : null;
|
|
117
|
-
const parentForContext =
|
|
118
|
-
(
|
|
118
|
+
const parentForContext = lazyContext
|
|
119
|
+
? getIsolatedLazyParent(
|
|
120
|
+
(lazyContext.parent as EntryData | null) ?? Store.parent,
|
|
121
|
+
)
|
|
122
|
+
: Store.parent;
|
|
119
123
|
|
|
120
124
|
// For lazy entries, merge captured counters from include() so the
|
|
121
125
|
// handler's entries get shortCode indices after sibling entries that
|
|
122
126
|
// were created during pattern extraction. This prevents shortCode
|
|
123
127
|
// collisions between lazy and non-lazy entries under the same parent
|
|
124
128
|
// (e.g., ArticlesLayout and BlogLayout both under NavLayout).
|
|
125
|
-
if (lazyContext
|
|
126
|
-
const
|
|
127
|
-
for (const [key, value] of Object.entries(captured)) {
|
|
129
|
+
if (lazyContext?.counters) {
|
|
130
|
+
for (const [key, value] of Object.entries(lazyContext.counters)) {
|
|
128
131
|
Store.counters[key] = Math.max(Store.counters[key] ?? 0, value);
|
|
129
132
|
}
|
|
130
133
|
}
|
|
131
134
|
|
|
132
135
|
// Propagate cache profiles for DSL-time cache("profileName") resolution.
|
|
133
136
|
// Non-lazy entries carry profiles directly; lazy entries carry them
|
|
134
|
-
// in the captured lazyContext from include() time.
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
|
|
139
|
-
|
|
137
|
+
// in the captured lazyContext from include() time. Always write
|
|
138
|
+
// (including clearing to undefined) so a prior lazy build's profile
|
|
139
|
+
// map cannot leak into a later non-lazy build on the same ALS-backed
|
|
140
|
+
// Store — which would otherwise let cache("name") resolve a profile
|
|
141
|
+
// from an unrelated entry.
|
|
142
|
+
Store.cacheProfiles = entry.cacheProfiles ?? lazyContext?.cacheProfiles;
|
|
140
143
|
|
|
141
144
|
// Propagate rootScoped from lazyContext so that routes inside
|
|
142
145
|
// nested { name: "sub" } under { name: "" } keep inherited root scope
|
|
143
|
-
// when the manifest is rebuilt on each request.
|
|
144
|
-
|
|
145
|
-
|
|
146
|
-
|
|
146
|
+
// when the manifest is rebuilt on each request. Always write
|
|
147
|
+
// (including clearing to undefined, which makes getRootScoped()
|
|
148
|
+
// return its true default) so a prior lazy build's scope cannot leak
|
|
149
|
+
// into a later non-lazy build on the same ALS-backed Store — which
|
|
150
|
+
// would otherwise mis-register plain routes as non-root-scoped and
|
|
151
|
+
// break dot-local reverse resolution.
|
|
152
|
+
Store.rootScoped = lazyContext?.rootScoped;
|
|
153
|
+
|
|
154
|
+
// Propagate includeScope from lazyContext so that direct-descendant
|
|
155
|
+
// shortCodes of this include use the correct scoped counter namespace
|
|
156
|
+
// on every manifest rebuild. Always write (including clearing to
|
|
157
|
+
// undefined) so a prior lazy build's scope cannot leak into a later
|
|
158
|
+
// non-lazy build on the same ALS-backed Store.
|
|
159
|
+
Store.includeScope = lazyContext?.includeScope;
|
|
147
160
|
|
|
148
161
|
const handlerExecStart = performance.now();
|
|
149
162
|
const useItems = await getContext().runWithStore(
|