@rangojs/router 0.0.0-experimental.3424742f → 0.0.0-experimental.349e5728

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (186) hide show
  1. package/README.md +188 -35
  2. package/dist/bin/rango.js +138 -50
  3. package/dist/vite/index.js +2018 -732
  4. package/dist/vite/index.js.bak +5448 -0
  5. package/dist/vite/plugins/cloudflare-protocol-loader-hook.mjs +76 -0
  6. package/package.json +7 -5
  7. package/skills/breadcrumbs/SKILL.md +3 -1
  8. package/skills/cache-guide/SKILL.md +32 -0
  9. package/skills/caching/SKILL.md +45 -4
  10. package/skills/handler-use/SKILL.md +364 -0
  11. package/skills/hooks/SKILL.md +33 -20
  12. package/skills/i18n/SKILL.md +276 -0
  13. package/skills/intercept/SKILL.md +45 -0
  14. package/skills/layout/SKILL.md +24 -0
  15. package/skills/links/SKILL.md +93 -17
  16. package/skills/loader/SKILL.md +123 -46
  17. package/skills/middleware/SKILL.md +36 -3
  18. package/skills/migrate-nextjs/SKILL.md +562 -0
  19. package/skills/migrate-react-router/SKILL.md +769 -0
  20. package/skills/parallel/SKILL.md +194 -0
  21. package/skills/prerender/SKILL.md +110 -68
  22. package/skills/rango/SKILL.md +26 -22
  23. package/skills/response-routes/SKILL.md +8 -0
  24. package/skills/route/SKILL.md +79 -0
  25. package/skills/router-setup/SKILL.md +87 -2
  26. package/skills/server-actions/SKILL.md +739 -0
  27. package/skills/streams-and-websockets/SKILL.md +283 -0
  28. package/skills/typesafety/SKILL.md +19 -1
  29. package/skills/view-transitions/SKILL.md +212 -0
  30. package/src/__internal.ts +1 -1
  31. package/src/browser/app-shell.ts +52 -0
  32. package/src/browser/app-version.ts +14 -0
  33. package/src/browser/event-controller.ts +49 -4
  34. package/src/browser/navigation-bridge.ts +150 -15
  35. package/src/browser/navigation-client.ts +167 -59
  36. package/src/browser/navigation-store.ts +68 -9
  37. package/src/browser/navigation-transaction.ts +11 -9
  38. package/src/browser/partial-update.ts +133 -24
  39. package/src/browser/prefetch/cache.ts +175 -15
  40. package/src/browser/prefetch/fetch.ts +180 -33
  41. package/src/browser/prefetch/queue.ts +123 -20
  42. package/src/browser/prefetch/resource-ready.ts +77 -0
  43. package/src/browser/rango-state.ts +53 -13
  44. package/src/browser/react/Link.tsx +81 -9
  45. package/src/browser/react/NavigationProvider.tsx +109 -21
  46. package/src/browser/react/context.ts +7 -2
  47. package/src/browser/react/filter-segment-order.ts +51 -7
  48. package/src/browser/react/use-handle.ts +9 -58
  49. package/src/browser/react/use-navigation.ts +22 -2
  50. package/src/browser/react/use-params.ts +17 -4
  51. package/src/browser/react/use-router.ts +29 -9
  52. package/src/browser/react/use-segments.ts +11 -8
  53. package/src/browser/rsc-router.tsx +168 -65
  54. package/src/browser/scroll-restoration.ts +34 -26
  55. package/src/browser/segment-reconciler.ts +36 -9
  56. package/src/browser/server-action-bridge.ts +8 -6
  57. package/src/browser/types.ts +55 -5
  58. package/src/build/generate-manifest.ts +6 -6
  59. package/src/build/generate-route-types.ts +3 -0
  60. package/src/build/route-trie.ts +52 -25
  61. package/src/build/route-types/include-resolution.ts +8 -1
  62. package/src/build/route-types/router-processing.ts +223 -74
  63. package/src/build/route-types/scan-filter.ts +8 -1
  64. package/src/cache/cache-runtime.ts +15 -11
  65. package/src/cache/cache-scope.ts +48 -7
  66. package/src/cache/cf/cf-cache-store.ts +455 -15
  67. package/src/cache/cf/index.ts +5 -1
  68. package/src/cache/document-cache.ts +17 -7
  69. package/src/cache/index.ts +1 -0
  70. package/src/cache/taint.ts +55 -0
  71. package/src/client.tsx +84 -230
  72. package/src/context-var.ts +72 -2
  73. package/src/debug.ts +2 -2
  74. package/src/handle.ts +40 -0
  75. package/src/href-client.ts +4 -1
  76. package/src/index.rsc.ts +6 -1
  77. package/src/index.ts +49 -6
  78. package/src/outlet-context.ts +1 -1
  79. package/src/prerender/store.ts +5 -4
  80. package/src/prerender.ts +138 -77
  81. package/src/response-utils.ts +28 -0
  82. package/src/reverse.ts +28 -2
  83. package/src/route-definition/dsl-helpers.ts +240 -40
  84. package/src/route-definition/helpers-types.ts +73 -20
  85. package/src/route-definition/index.ts +3 -0
  86. package/src/route-definition/redirect.ts +11 -3
  87. package/src/route-definition/resolve-handler-use.ts +155 -0
  88. package/src/route-types.ts +18 -0
  89. package/src/router/content-negotiation.ts +100 -1
  90. package/src/router/handler-context.ts +102 -25
  91. package/src/router/intercept-resolution.ts +9 -4
  92. package/src/router/lazy-includes.ts +8 -7
  93. package/src/router/loader-resolution.ts +159 -21
  94. package/src/router/logging.ts +1 -1
  95. package/src/router/manifest.ts +28 -15
  96. package/src/router/match-api.ts +128 -192
  97. package/src/router/match-handlers.ts +1 -0
  98. package/src/router/match-middleware/background-revalidation.ts +30 -2
  99. package/src/router/match-middleware/cache-lookup.ts +94 -17
  100. package/src/router/match-middleware/cache-store.ts +53 -10
  101. package/src/router/match-middleware/intercept-resolution.ts +9 -7
  102. package/src/router/match-middleware/segment-resolution.ts +60 -5
  103. package/src/router/match-result.ts +123 -10
  104. package/src/router/metrics.ts +6 -1
  105. package/src/router/middleware-types.ts +20 -33
  106. package/src/router/middleware.ts +58 -13
  107. package/src/router/navigation-snapshot.ts +182 -0
  108. package/src/router/pattern-matching.ts +101 -17
  109. package/src/router/prerender-match.ts +110 -10
  110. package/src/router/preview-match.ts +30 -102
  111. package/src/router/request-classification.ts +310 -0
  112. package/src/router/revalidation.ts +15 -1
  113. package/src/router/route-snapshot.ts +245 -0
  114. package/src/router/router-context.ts +1 -0
  115. package/src/router/router-interfaces.ts +36 -4
  116. package/src/router/router-options.ts +37 -11
  117. package/src/router/segment-resolution/fresh.ts +206 -20
  118. package/src/router/segment-resolution/helpers.ts +29 -24
  119. package/src/router/segment-resolution/loader-cache.ts +1 -0
  120. package/src/router/segment-resolution/revalidation.ts +447 -282
  121. package/src/router/trie-matching.ts +18 -13
  122. package/src/router/types.ts +1 -0
  123. package/src/router/url-params.ts +49 -0
  124. package/src/router.ts +56 -8
  125. package/src/rsc/handler.ts +478 -374
  126. package/src/rsc/helpers.ts +69 -41
  127. package/src/rsc/loader-fetch.ts +23 -3
  128. package/src/rsc/manifest-init.ts +5 -1
  129. package/src/rsc/progressive-enhancement.ts +18 -2
  130. package/src/rsc/response-route-handler.ts +14 -1
  131. package/src/rsc/rsc-rendering.ts +20 -1
  132. package/src/rsc/server-action.ts +12 -0
  133. package/src/rsc/ssr-setup.ts +2 -2
  134. package/src/rsc/types.ts +15 -1
  135. package/src/segment-content-promise.ts +67 -0
  136. package/src/segment-loader-promise.ts +122 -0
  137. package/src/segment-system.tsx +169 -32
  138. package/src/server/context.ts +166 -17
  139. package/src/server/handle-store.ts +19 -0
  140. package/src/server/loader-registry.ts +9 -8
  141. package/src/server/request-context.ts +194 -60
  142. package/src/ssr/index.tsx +9 -1
  143. package/src/static-handler.ts +18 -6
  144. package/src/types/cache-types.ts +4 -4
  145. package/src/types/handler-context.ts +145 -68
  146. package/src/types/loader-types.ts +41 -15
  147. package/src/types/request-scope.ts +126 -0
  148. package/src/types/route-entry.ts +12 -1
  149. package/src/types/segments.ts +19 -0
  150. package/src/urls/include-helper.ts +24 -14
  151. package/src/urls/path-helper-types.ts +39 -6
  152. package/src/urls/path-helper.ts +48 -13
  153. package/src/urls/pattern-types.ts +12 -0
  154. package/src/urls/response-types.ts +18 -16
  155. package/src/use-loader.tsx +77 -5
  156. package/src/vite/debug.ts +184 -0
  157. package/src/vite/discovery/bundle-postprocess.ts +30 -33
  158. package/src/vite/discovery/discover-routers.ts +36 -4
  159. package/src/vite/discovery/gate-state.ts +171 -0
  160. package/src/vite/discovery/prerender-collection.ts +175 -74
  161. package/src/vite/discovery/self-gen-tracking.ts +27 -1
  162. package/src/vite/discovery/state.ts +13 -6
  163. package/src/vite/index.ts +4 -0
  164. package/src/vite/plugin-types.ts +51 -79
  165. package/src/vite/plugins/cjs-to-esm.ts +5 -0
  166. package/src/vite/plugins/client-ref-dedup.ts +16 -0
  167. package/src/vite/plugins/client-ref-hashing.ts +16 -4
  168. package/src/vite/plugins/cloudflare-protocol-loader-hook.d.mts +23 -0
  169. package/src/vite/plugins/cloudflare-protocol-loader-hook.mjs +76 -0
  170. package/src/vite/plugins/cloudflare-protocol-stub.ts +214 -0
  171. package/src/vite/plugins/expose-action-id.ts +53 -31
  172. package/src/vite/plugins/expose-id-utils.ts +12 -0
  173. package/src/vite/plugins/expose-ids/handler-transform.ts +30 -0
  174. package/src/vite/plugins/expose-ids/router-transform.ts +20 -3
  175. package/src/vite/plugins/expose-internal-ids.ts +563 -316
  176. package/src/vite/plugins/performance-tracks.ts +96 -0
  177. package/src/vite/plugins/refresh-cmd.ts +88 -26
  178. package/src/vite/plugins/use-cache-transform.ts +56 -43
  179. package/src/vite/plugins/version-injector.ts +37 -11
  180. package/src/vite/plugins/version-plugin.ts +13 -1
  181. package/src/vite/rango.ts +204 -217
  182. package/src/vite/router-discovery.ts +732 -94
  183. package/src/vite/utils/banner.ts +4 -4
  184. package/src/vite/utils/package-resolution.ts +41 -1
  185. package/src/vite/utils/prerender-utils.ts +38 -5
  186. package/src/vite/utils/shared-utils.ts +3 -2
package/README.md CHANGED
@@ -10,6 +10,7 @@ Named-route RSC router with structural composability and type-safe partial rende
10
10
  - **Structural composability** — Attach routes, loaders, middleware, handles, caching, prerendering, and static generation without hiding the route tree
11
11
  - **Composable URL patterns** — Django-style `urls()` DSL with `path`, `layout`, `include`
12
12
  - **Data loaders** — `createLoader()` with automatic streaming and Suspense integration
13
+ - **Server actions** — `"use server"` mutations with `useActionState`, `useOptimistic`, and per-segment + per-loader `revalidate()` rules
13
14
  - **Live data layer** — Pre-render or cache the UI shell while loaders stay live by default at request time
14
15
  - **Layouts & nesting** — Nested layouts with `<Outlet />` and parallel routes
15
16
  - **Segment-level caching** — `cache()` DSL with TTL/SWR and pluggable cache stores
@@ -91,24 +92,29 @@ This file is a server/RSC module and should import router construction APIs from
91
92
 
92
93
  ```tsx
93
94
  // src/router.tsx
94
- import { createRouter, urls } from "@rangojs/router";
95
- import { Document } from "./document";
95
+ import { createRouter } from "@rangojs/router";
96
96
 
97
- const blogPatterns = urls(({ path }) => [
98
- path("/", BlogIndexPage, { name: "index" }),
99
- path("/:slug", BlogPostPage, { name: "post" }),
97
+ export const router = createRouter().routes(({ path }) => [
98
+ path("/", HomePage, { name: "home" }),
99
+ path("/about", AboutPage, { name: "about" }),
100
100
  ]);
101
101
 
102
+ export const reverse = router.reverse;
103
+ // reverse("home") -> "/"
104
+ ```
105
+
106
+ For larger apps, extract route modules with `urls()` and compose with `include()`:
107
+
108
+ ```tsx
109
+ import { createRouter, urls } from "@rangojs/router";
110
+ import { blogPatterns } from "./urls/blog";
111
+
102
112
  const urlpatterns = urls(({ path, include }) => [
103
113
  path("/", HomePage, { name: "home" }),
104
114
  include("/blog", blogPatterns, { name: "blog" }),
105
115
  ]);
106
116
 
107
- export const router = createRouter({ document: Document }).routes(urlpatterns);
108
-
109
- // Export typed reverse function for URL generation by route name
110
- export const reverse = router.reverse;
111
-
117
+ export const router = createRouter().routes(urlpatterns);
112
118
  // reverse("blog.post", { slug: "hello-world" }) -> "/blog/hello-world"
113
119
  ```
114
120
 
@@ -156,13 +162,18 @@ const urlpatterns = urls(({ path }) => [
156
162
  ]);
157
163
  ```
158
164
 
159
- Use `reverse()` as the default way to link to routes:
165
+ Use `ctx.reverse()` from handler context as the default way to link to routes from server code:
160
166
 
161
167
  ```tsx
162
- router.reverse("product", { slug: "widget" }); // "/product/widget"
163
- router.reverse("search", undefined, { q: "rsc" }); // "/search?q=rsc"
168
+ const ProductPage: Handler<"product"> = (ctx) => {
169
+ const url = ctx.reverse("product", { slug: "widget" }); // "/product/widget"
170
+ const searchUrl = ctx.reverse("search", undefined, { q: "rsc" }); // "/search?q=rsc"
171
+ return <Link to={url}>Widget</Link>;
172
+ };
164
173
  ```
165
174
 
175
+ `router.reverse()` (exported from the router module) is the same function without a handler context, useful in scripts or tests. In request code, prefer `ctx.reverse()` — it auto-fills mount params from the current match.
176
+
166
177
  ### Composable URL Modules
167
178
 
168
179
  Local route names compose cleanly with `include(..., { name })`:
@@ -472,41 +483,130 @@ const urlpatterns = urls(({ path, loader }) => [
472
483
  ]);
473
484
  ```
474
485
 
475
- ## Navigation & Links
486
+ ## Server Actions
487
+
488
+ Server actions are React's RSC mutation primitive. Define them with the
489
+ `"use server"` directive — Rango uses standard React 19 hooks
490
+ (`useActionState`, `useFormStatus`, `useOptimistic`) with no framework wrapper.
476
491
 
477
- ### Named Routes with `reverse()` (Server Components)
492
+ ```tsx
493
+ // app/actions/cart.ts
494
+ "use server";
495
+
496
+ import { getRequestContext } from "@rangojs/router";
478
497
 
479
- In server components, use `reverse()` to generate URLs by route name:
498
+ export async function addToCart(productId: string): Promise<void> {
499
+ const ctx = getRequestContext();
500
+ const userId = ctx.get("user").id;
501
+ await db.cart.insert({ userId, productId });
502
+ }
503
+ ```
480
504
 
481
505
  ```tsx
482
- import { Link } from "@rangojs/router/client";
483
- import { reverse } from "./router";
506
+ // Client form with progressive enhancement + pending state
507
+ "use client";
508
+ import { useActionState } from "react";
509
+ import { saveProfile } from "../actions/profile";
484
510
 
485
- function BlogIndex() {
511
+ export function ProfileForm() {
512
+ const [state, action, pending] = useActionState(saveProfile, null);
486
513
  return (
487
- <nav>
488
- <Link to={reverse("home")}>Home</Link>
489
- <Link to={reverse("blogPost", { slug: "my-post" })}>My Post</Link>
490
- <Link to={reverse("about")}>About</Link>
491
- </nav>
514
+ <form action={action}>
515
+ <input name="name" defaultValue={state?.values?.name} />
516
+ {state?.errors?.name && <p role="alert">{state.errors.name}</p>}
517
+ <button disabled={pending}>{pending ? "Saving…" : "Save"}</button>
518
+ </form>
492
519
  );
493
520
  }
494
521
  ```
495
522
 
496
- `reverse()` is type-safe route names and required params are checked at compile time. Included routes use dotted names: `reverse("api.health")`.
523
+ After an action runs, matched route segments (path/layout/parallel/intercept)
524
+ and loaders can re-render/re-resolve so the UI reflects the new state.
525
+ Attach a `revalidate(({ actionId }) => ...)` rule on any segment or loader
526
+ that owns data the action touched:
527
+
528
+ ```tsx
529
+ urls(({ path, loader, revalidate }) => [
530
+ // Segment-level: re-render the cart page handler after cart actions.
531
+ // Nest loaders that belong to this route inside the same path() so the
532
+ // segment owns its data dependencies.
533
+ path("/cart", CartPage, { name: "cart" }, () => [
534
+ revalidate(
535
+ ({ actionId }) => actionId?.startsWith("src/actions/cart.ts#") ?? false,
536
+ ),
537
+ loader(CartLoader, () => [
538
+ revalidate(
539
+ ({ actionId }) => actionId?.startsWith("src/actions/cart.ts#") ?? false,
540
+ ),
541
+ ]),
542
+ ]),
543
+ ]);
544
+ ```
545
+
546
+ For the full guide — validation with Zod, error handling, file uploads,
547
+ `useOptimistic`, redirects, and progressive enhancement — see the
548
+ `/server-actions` skill.
549
+
550
+ ## Navigation & Links
551
+
552
+ ### Named Routes with `ctx.reverse()` (Server)
497
553
 
498
- Handlers also have `ctx.reverse()` directly on the context:
554
+ In server components and handlers, use `ctx.reverse()` to generate URLs by route name. This is the default — it is typed, auto-fills mount params from the current match, and resolves both local (`.name`) and absolute (`name.sub`) names:
499
555
 
500
556
  ```tsx
557
+ import { Link } from "@rangojs/router/client";
558
+ import type { Handler } from "@rangojs/router";
559
+
501
560
  const BlogPostPage: Handler<"blogPost"> = (ctx) => {
502
561
  const backUrl = ctx.reverse("blog");
503
562
  return <Link to={backUrl}>Back to blog</Link>;
504
563
  };
505
564
  ```
506
565
 
566
+ `reverse()` is type-safe — route names and required params are checked at compile time. Included routes use dotted names: `ctx.reverse("api.health")`.
567
+
568
+ For scripts, tests, or other code without a handler context, import the router-level `reverse`:
569
+
570
+ ```tsx
571
+ import { reverse } from "./router";
572
+ reverse("blogPost", { slug: "my-post" });
573
+ ```
574
+
575
+ ### Client Components
576
+
577
+ **`reverse()` is server-only.** It depends on the route manifest and handler context — neither is available in the browser bundle. Client components receive URLs as props, loader data, or server-action return values:
578
+
579
+ ```tsx
580
+ // server
581
+ function BlogIndex(ctx: HandlerContext) {
582
+ return (
583
+ <Nav
584
+ home={ctx.reverse("home")}
585
+ post={ctx.reverse("blogPost", { slug: "my-post" })}
586
+ />
587
+ );
588
+ }
589
+ ```
590
+
591
+ ```tsx
592
+ "use client";
593
+ import { Link } from "@rangojs/router/client";
594
+
595
+ export function Nav({ home, post }: { home: string; post: string }) {
596
+ return (
597
+ <nav>
598
+ <Link to={home}>Home</Link>
599
+ <Link to={post}>My Post</Link>
600
+ </nav>
601
+ );
602
+ }
603
+ ```
604
+
605
+ For client-side navigation to static paths (no named-route lookup), use `href()` — see below. For URLs tied to named routes, always generate on the server and pass the string in.
606
+
507
607
  ### `href()` for Path Validation (Client Components)
508
608
 
509
- In client components, use `href()` for compile-time path validation:
609
+ In client components, use `href()` for compile-time path validation on static path strings:
510
610
 
511
611
  ```tsx
512
612
  "use client";
@@ -711,10 +811,12 @@ export const BlogPost = Prerender(
711
811
 
712
812
  ### Passthrough for Unknown Params
713
813
 
814
+ Wrap a `Prerender` definition with `Passthrough()` to add a live handler for unknown params at runtime. The build handler runs at build time, the live handler runs at request time for params not in the prerender cache.
815
+
714
816
  ```tsx
715
- import { Prerender } from "@rangojs/router";
817
+ import { Prerender, Passthrough } from "@rangojs/router";
716
818
 
717
- export const ProductPage = Prerender(
819
+ export const ProductPageDef = Prerender(
718
820
  async () => {
719
821
  const featured = await db.getFeaturedProducts();
720
822
  return featured.map((p) => ({ id: p.id }));
@@ -723,16 +825,22 @@ export const ProductPage = Prerender(
723
825
  const product = await db.getProduct(ctx.params.id);
724
826
  return <Product data={product} />;
725
827
  },
726
- { passthrough: true },
727
828
  );
728
- ```
729
829
 
730
- With `passthrough: true`, known params are served from the build-time cache and unknown params fall through to live rendering.
830
+ // In route definition:
831
+ path(
832
+ "/products/:id",
833
+ Passthrough(ProductPageDef, async (ctx) => {
834
+ const product = await ctx.env.DB.getProduct(ctx.params.id);
835
+ return <Product data={product} />;
836
+ }),
837
+ );
838
+ ```
731
839
 
732
- Handlers can also skip individual param sets with `ctx.passthrough()`, deferring them to the live handler at runtime:
840
+ Build handlers can also skip individual param sets with `ctx.passthrough()`, deferring them to the live handler:
733
841
 
734
842
  ```tsx
735
- export const ProductPage = Prerender(
843
+ export const ProductPageDef = Prerender(
736
844
  async () => {
737
845
  const all = await db.getAllProducts();
738
846
  return all.map((p) => ({ id: p.id }));
@@ -742,10 +850,55 @@ export const ProductPage = Prerender(
742
850
  if (!product.published) return ctx.passthrough();
743
851
  return <Product data={product} />;
744
852
  },
745
- { passthrough: true },
746
853
  );
747
854
  ```
748
855
 
856
+ ### Build-Time Environment Bindings
857
+
858
+ Prerender handlers can access platform bindings (KV, D1, R2) at build time when `buildEnv` is configured in the Vite plugin:
859
+
860
+ ```ts
861
+ // vite.config.ts
862
+ import { rango } from "@rangojs/router/vite";
863
+
864
+ rango({ preset: "cloudflare", buildEnv: "auto" });
865
+ ```
866
+
867
+ With `buildEnv: "auto"`, the plugin calls `wrangler.getPlatformProxy()` to provide local bindings. Handlers then access `ctx.env` during build:
868
+
869
+ ```tsx
870
+ export const BlogPosts = Prerender<{ slug: string }>(
871
+ async (ctx) => {
872
+ const rows = await ctx.env.DB.prepare("SELECT slug FROM posts").all();
873
+ return rows.map((r) => ({ slug: r.slug }));
874
+ },
875
+ async (ctx) => {
876
+ const post = await ctx.env.DB.prepare("SELECT * FROM posts WHERE slug = ?")
877
+ .bind(ctx.params.slug)
878
+ .first();
879
+ return <BlogPost post={post} />;
880
+ },
881
+ );
882
+ ```
883
+
884
+ `buildEnv` also accepts a factory function or plain object:
885
+
886
+ ```ts
887
+ // Custom factory
888
+ rango({
889
+ buildEnv: async (ctx) => {
890
+ const { getPlatformProxy } = await import("wrangler");
891
+ const proxy = await getPlatformProxy();
892
+ return { env: proxy.env, dispose: proxy.dispose };
893
+ },
894
+ });
895
+
896
+ // Plain object (Node.js)
897
+ rango({ buildEnv: { DATABASE_URL: process.env.DATABASE_URL } });
898
+ ```
899
+
900
+ Build-time env applies to both production builds and dev on-demand prerender. Without `buildEnv`, accessing `ctx.env` in a Prerender handler throws with a clear error.
901
+
749
902
  ## Theme
750
903
 
751
904
  ### Router Configuration
package/dist/bin/rango.js CHANGED
@@ -218,7 +218,8 @@ function findTsFiles(dir, filter) {
218
218
  for (const entry of entries) {
219
219
  const fullPath = join(dir, entry.name);
220
220
  if (entry.isDirectory()) {
221
- if (entry.name === "node_modules" || entry.name.startsWith(".")) continue;
221
+ if (entry.name === "node_modules" || entry.name.startsWith(".") || entry.name === "dist" || entry.name === "build" || entry.name === "coverage")
222
+ continue;
222
223
  results.push(...findTsFiles(fullPath, filter));
223
224
  } else if ((entry.name.endsWith(".ts") || entry.name.endsWith(".tsx") || entry.name.endsWith(".js") || entry.name.endsWith(".jsx")) && !entry.name.includes(".gen.")) {
224
225
  if (filter && !filter(fullPath)) continue;
@@ -450,7 +451,7 @@ function buildRouteMapFromBlock(block, fullSource, filePath, visited, searchSche
450
451
  }
451
452
  return routeMap;
452
453
  }
453
- function buildCombinedRouteMapWithSearch(filePath, variableName, visited, diagnosticsOut) {
454
+ function buildCombinedRouteMapWithSearch(filePath, variableName, visited, diagnosticsOut, inlineBlock) {
454
455
  visited = visited ?? /* @__PURE__ */ new Set();
455
456
  const realPath = resolve(filePath);
456
457
  const key = variableName ? `${realPath}:${variableName}` : realPath;
@@ -466,7 +467,9 @@ function buildCombinedRouteMapWithSearch(filePath, variableName, visited, diagno
466
467
  return { routes: {}, searchSchemas: {} };
467
468
  }
468
469
  let block;
469
- if (variableName) {
470
+ if (inlineBlock) {
471
+ block = inlineBlock;
472
+ } else if (variableName) {
470
473
  const extracted = extractUrlsBlockForVariable(source, variableName);
471
474
  if (!extracted) return { routes: {}, searchSchemas: {} };
472
475
  block = extracted;
@@ -601,7 +604,7 @@ function countPublicRouteEntries(source) {
601
604
  return count;
602
605
  }
603
606
  function isRoutableSourceFile(name) {
604
- return (name.endsWith(".ts") || name.endsWith(".tsx") || name.endsWith(".js") || name.endsWith(".jsx")) && !name.includes(".gen.");
607
+ return (name.endsWith(".ts") || name.endsWith(".tsx") || name.endsWith(".js") || name.endsWith(".jsx")) && !name.includes(".gen.") && !name.includes(".test.") && !name.includes(".spec.");
605
608
  }
606
609
  function findRouterFilesRecursive(dir, filter, results) {
607
610
  let entries;
@@ -618,7 +621,8 @@ function findRouterFilesRecursive(dir, filter, results) {
618
621
  for (const entry of entries) {
619
622
  const fullPath = join2(dir, entry.name);
620
623
  if (entry.isDirectory()) {
621
- if (entry.name === "node_modules" || entry.name.startsWith(".")) continue;
624
+ if (entry.name === "node_modules" || entry.name === "dist" || entry.name === "coverage" || entry.name === "__tests__" || entry.name === "__mocks__" || entry.name.startsWith("."))
625
+ continue;
622
626
  childDirs.push(fullPath);
623
627
  continue;
624
628
  }
@@ -670,7 +674,7 @@ Router root: ${conflict.ancestor}
670
674
  Nested router: ${conflict.nested}
671
675
  Move the nested router into a sibling directory or configure it as a separate app root.`;
672
676
  }
673
- function extractUrlsVariableFromRouter(code) {
677
+ function extractUrlsFromRouter(code) {
674
678
  const sourceFile = ts5.createSourceFile(
675
679
  "router.tsx",
676
680
  code,
@@ -684,24 +688,70 @@ function extractUrlsVariableFromRouter(code) {
684
688
  const callee = node.expression;
685
689
  return ts5.isIdentifier(callee) && callee.text === "createRouter";
686
690
  }
691
+ function isInlineBuilder(node) {
692
+ return ts5.isArrowFunction(node) || ts5.isFunctionExpression(node);
693
+ }
694
+ function isRoutesOnCreateRouter(node) {
695
+ if (!ts5.isPropertyAccessExpression(node.expression) || node.expression.name.text !== "routes")
696
+ return false;
697
+ let inner = node.expression.expression;
698
+ while (ts5.isCallExpression(inner) && ts5.isPropertyAccessExpression(inner.expression)) {
699
+ inner = inner.expression.expression;
700
+ }
701
+ return isCreateRouterCall(inner);
702
+ }
687
703
  function visit(node) {
688
704
  if (result) return;
689
- if (ts5.isCallExpression(node) && ts5.isPropertyAccessExpression(node.expression) && node.expression.name.text === "routes" && node.arguments.length >= 1 && ts5.isIdentifier(node.arguments[0])) {
690
- let inner = node.expression.expression;
691
- while (ts5.isCallExpression(inner) && ts5.isPropertyAccessExpression(inner.expression)) {
692
- inner = inner.expression.expression;
693
- }
694
- if (isCreateRouterCall(inner)) {
695
- result = node.arguments[0].text;
696
- return;
705
+ if (ts5.isCallExpression(node) && node.arguments.length >= 1 && isRoutesOnCreateRouter(node)) {
706
+ const arg = node.arguments[0];
707
+ if (ts5.isIdentifier(arg)) {
708
+ result = { kind: "variable", name: arg.text };
709
+ } else if (isInlineBuilder(arg)) {
710
+ result = { kind: "inline", block: arg.getText(sourceFile) };
697
711
  }
712
+ return;
698
713
  }
699
714
  if (isCreateRouterCall(node)) {
700
715
  const callExpr = node;
701
- for (const arg of callExpr.arguments) {
716
+ for (const callArg of callExpr.arguments) {
717
+ if (ts5.isObjectLiteralExpression(callArg)) {
718
+ for (const prop of callArg.properties) {
719
+ if (ts5.isPropertyAssignment(prop) && ts5.isIdentifier(prop.name) && prop.name.text === "urls") {
720
+ if (ts5.isIdentifier(prop.initializer)) {
721
+ result = { kind: "variable", name: prop.initializer.text };
722
+ } else if (isInlineBuilder(prop.initializer)) {
723
+ result = {
724
+ kind: "inline",
725
+ block: prop.initializer.getText(sourceFile)
726
+ };
727
+ }
728
+ return;
729
+ }
730
+ }
731
+ }
732
+ }
733
+ }
734
+ ts5.forEachChild(node, visit);
735
+ }
736
+ visit(sourceFile);
737
+ return result;
738
+ }
739
+ function extractBasenameFromRouter(code) {
740
+ const sourceFile = ts5.createSourceFile(
741
+ "router.tsx",
742
+ code,
743
+ ts5.ScriptTarget.Latest,
744
+ true,
745
+ ts5.ScriptKind.TSX
746
+ );
747
+ let result;
748
+ function visit(node) {
749
+ if (result !== void 0) return;
750
+ if (ts5.isCallExpression(node) && ts5.isIdentifier(node.expression) && node.expression.text === "createRouter") {
751
+ for (const arg of node.arguments) {
702
752
  if (ts5.isObjectLiteralExpression(arg)) {
703
753
  for (const prop of arg.properties) {
704
- if (ts5.isPropertyAssignment(prop) && ts5.isIdentifier(prop.name) && prop.name.text === "urls" && ts5.isIdentifier(prop.initializer)) {
754
+ if (ts5.isPropertyAssignment(prop) && ts5.isIdentifier(prop.name) && prop.name.text === "basename" && ts5.isStringLiteral(prop.initializer)) {
705
755
  result = prop.initializer.text;
706
756
  return;
707
757
  }
@@ -714,6 +764,19 @@ function extractUrlsVariableFromRouter(code) {
714
764
  visit(sourceFile);
715
765
  return result;
716
766
  }
767
+ function applyBasenameToRoutes(result, basename2) {
768
+ const prefixed = {};
769
+ for (const [name, pattern] of Object.entries(result.routes)) {
770
+ if (pattern === "/") {
771
+ prefixed[name] = basename2;
772
+ } else if (basename2.endsWith("/") && pattern.startsWith("/")) {
773
+ prefixed[name] = basename2 + pattern.slice(1);
774
+ } else {
775
+ prefixed[name] = basename2 + pattern;
776
+ }
777
+ }
778
+ return { routes: prefixed, searchSchemas: result.searchSchemas };
779
+ }
717
780
  function buildCombinedRouteMapForRouterFile(routerFilePath) {
718
781
  let routerSource;
719
782
  try {
@@ -721,19 +784,40 @@ function buildCombinedRouteMapForRouterFile(routerFilePath) {
721
784
  } catch {
722
785
  return { routes: {}, searchSchemas: {} };
723
786
  }
724
- const urlsVarName = extractUrlsVariableFromRouter(routerSource);
725
- if (!urlsVarName) {
787
+ const extraction = extractUrlsFromRouter(routerSource);
788
+ if (!extraction) {
726
789
  return { routes: {}, searchSchemas: {} };
727
790
  }
728
- const imported = resolveImportedVariable(routerSource, urlsVarName);
729
- if (imported) {
730
- const targetFile = resolveImportPath(imported.specifier, routerFilePath);
731
- if (!targetFile) {
732
- return { routes: {}, searchSchemas: {} };
791
+ const rawBasename = extractBasenameFromRouter(routerSource);
792
+ const basename2 = rawBasename ? ("/" + rawBasename.replace(/^\/+|\/+$/g, "")).replace(/^\/$/, "") : void 0;
793
+ let result;
794
+ if (extraction.kind === "inline") {
795
+ result = buildCombinedRouteMapWithSearch(
796
+ routerFilePath,
797
+ void 0,
798
+ void 0,
799
+ void 0,
800
+ extraction.block
801
+ );
802
+ } else {
803
+ const imported = resolveImportedVariable(routerSource, extraction.name);
804
+ if (imported) {
805
+ const targetFile = resolveImportPath(imported.specifier, routerFilePath);
806
+ if (!targetFile) {
807
+ return { routes: {}, searchSchemas: {} };
808
+ }
809
+ result = buildCombinedRouteMapWithSearch(
810
+ targetFile,
811
+ imported.exportedName
812
+ );
813
+ } else {
814
+ result = buildCombinedRouteMapWithSearch(routerFilePath, extraction.name);
733
815
  }
734
- return buildCombinedRouteMapWithSearch(targetFile, imported.exportedName);
735
816
  }
736
- return buildCombinedRouteMapWithSearch(routerFilePath, urlsVarName);
817
+ if (basename2) {
818
+ result = applyBasenameToRoutes(result, basename2);
819
+ }
820
+ return result;
737
821
  }
738
822
  function detectUnresolvableIncludes(routerFilePath) {
739
823
  const realPath = resolve2(routerFilePath);
@@ -743,9 +827,20 @@ function detectUnresolvableIncludes(routerFilePath) {
743
827
  } catch {
744
828
  return [];
745
829
  }
746
- const urlsVarName = extractUrlsVariableFromRouter(source);
747
- if (!urlsVarName) return [];
748
- const imported = resolveImportedVariable(source, urlsVarName);
830
+ const extraction = extractUrlsFromRouter(source);
831
+ if (!extraction) return [];
832
+ const diagnostics = [];
833
+ if (extraction.kind === "inline") {
834
+ buildCombinedRouteMapWithSearch(
835
+ realPath,
836
+ void 0,
837
+ /* @__PURE__ */ new Set(),
838
+ diagnostics,
839
+ extraction.block
840
+ );
841
+ return diagnostics;
842
+ }
843
+ const imported = resolveImportedVariable(source, extraction.name);
749
844
  let targetFile;
750
845
  let exportedName;
751
846
  if (imported) {
@@ -765,9 +860,8 @@ function detectUnresolvableIncludes(routerFilePath) {
765
860
  exportedName = imported.exportedName;
766
861
  } else {
767
862
  targetFile = realPath;
768
- exportedName = urlsVarName;
863
+ exportedName = extraction.name;
769
864
  }
770
- const diagnostics = [];
771
865
  buildCombinedRouteMapWithSearch(
772
866
  targetFile,
773
867
  exportedName,
@@ -815,25 +909,15 @@ function writeCombinedRouteTypes(root, knownRouterFiles, opts) {
815
909
  throw new Error(formatNestedRouterConflictError(nestedRouterConflict));
816
910
  }
817
911
  for (const routerFilePath of routerFilePaths) {
818
- let routerSource;
819
- try {
820
- routerSource = readFileSync3(routerFilePath, "utf-8");
821
- } catch {
822
- continue;
823
- }
824
- const urlsVarName = extractUrlsVariableFromRouter(routerSource);
825
- if (!urlsVarName) continue;
826
- let result;
827
- const imported = resolveImportedVariable(routerSource, urlsVarName);
828
- if (imported) {
829
- const targetFile = resolveImportPath(imported.specifier, routerFilePath);
830
- if (!targetFile) continue;
831
- result = buildCombinedRouteMapWithSearch(
832
- targetFile,
833
- imported.exportedName
834
- );
835
- } else {
836
- result = buildCombinedRouteMapWithSearch(routerFilePath, urlsVarName);
912
+ const result = buildCombinedRouteMapForRouterFile(routerFilePath);
913
+ if (Object.keys(result.routes).length === 0 && Object.keys(result.searchSchemas).length === 0) {
914
+ let routerSource;
915
+ try {
916
+ routerSource = readFileSync3(routerFilePath, "utf-8");
917
+ } catch {
918
+ continue;
919
+ }
920
+ if (!extractUrlsFromRouter(routerSource)) continue;
837
921
  }
838
922
  const routerBasename = pathBasename(routerFilePath).replace(
839
923
  /\.(tsx?|jsx?)$/,
@@ -1062,8 +1146,9 @@ function createVersionPlugin() {
1062
1146
  let isDev = false;
1063
1147
  let server = null;
1064
1148
  const clientModuleSignatures = /* @__PURE__ */ new Map();
1149
+ let versionCounter = 0;
1065
1150
  const bumpVersion = (reason) => {
1066
- currentVersion = Date.now().toString(16);
1151
+ currentVersion = Date.now().toString(16) + String(++versionCounter);
1067
1152
  console.log(`[rsc-router] ${reason}, version updated: ${currentVersion}`);
1068
1153
  const rscEnv = server?.environments?.rsc;
1069
1154
  const versionMod = rscEnv?.moduleGraph?.getModuleById(
@@ -1119,6 +1204,9 @@ function createVersionPlugin() {
1119
1204
  if (!isDev) return;
1120
1205
  const isRscModule = this.environment?.name === "rsc";
1121
1206
  if (!isRscModule) return;
1207
+ if (ctx.modules.length === 1 && ctx.modules[0].id === "\0" + VIRTUAL_IDS.version) {
1208
+ return;
1209
+ }
1122
1210
  if (isCodeModule(ctx.file)) {
1123
1211
  const filePath = normalizeModuleId(ctx.file);
1124
1212
  const previousSignature = clientModuleSignatures.get(filePath);