@benjavicente/router-core 1.168.9

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 (380) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +5 -0
  3. package/bin/intent.js +25 -0
  4. package/dist/cjs/Matches.cjs +17 -0
  5. package/dist/cjs/Matches.cjs.map +1 -0
  6. package/dist/cjs/Matches.d.cts +139 -0
  7. package/dist/cjs/RouterProvider.d.cts +27 -0
  8. package/dist/cjs/config.cjs +11 -0
  9. package/dist/cjs/config.cjs.map +1 -0
  10. package/dist/cjs/config.d.cts +17 -0
  11. package/dist/cjs/defer.cjs +41 -0
  12. package/dist/cjs/defer.cjs.map +1 -0
  13. package/dist/cjs/defer.d.cts +37 -0
  14. package/dist/cjs/fileRoute.d.cts +24 -0
  15. package/dist/cjs/global.d.cts +7 -0
  16. package/dist/cjs/hash-scroll.cjs +20 -0
  17. package/dist/cjs/hash-scroll.cjs.map +1 -0
  18. package/dist/cjs/hash-scroll.d.cts +7 -0
  19. package/dist/cjs/history.d.cts +8 -0
  20. package/dist/cjs/index.cjs +96 -0
  21. package/dist/cjs/index.d.cts +53 -0
  22. package/dist/cjs/invariant.cjs +8 -0
  23. package/dist/cjs/invariant.cjs.map +1 -0
  24. package/dist/cjs/invariant.d.cts +1 -0
  25. package/dist/cjs/isServer/client.cjs +7 -0
  26. package/dist/cjs/isServer/client.cjs.map +1 -0
  27. package/dist/cjs/isServer/client.d.cts +1 -0
  28. package/dist/cjs/isServer/development.cjs +7 -0
  29. package/dist/cjs/isServer/development.cjs.map +1 -0
  30. package/dist/cjs/isServer/development.d.cts +1 -0
  31. package/dist/cjs/isServer/server.cjs +7 -0
  32. package/dist/cjs/isServer/server.cjs.map +1 -0
  33. package/dist/cjs/isServer/server.d.cts +1 -0
  34. package/dist/cjs/link.cjs +6 -0
  35. package/dist/cjs/link.cjs.map +1 -0
  36. package/dist/cjs/link.d.cts +221 -0
  37. package/dist/cjs/load-matches.cjs +659 -0
  38. package/dist/cjs/load-matches.cjs.map +1 -0
  39. package/dist/cjs/load-matches.d.cts +18 -0
  40. package/dist/cjs/location.d.cts +50 -0
  41. package/dist/cjs/lru-cache.cjs +70 -0
  42. package/dist/cjs/lru-cache.cjs.map +1 -0
  43. package/dist/cjs/lru-cache.d.cts +6 -0
  44. package/dist/cjs/manifest.cjs +18 -0
  45. package/dist/cjs/manifest.cjs.map +1 -0
  46. package/dist/cjs/manifest.d.cts +35 -0
  47. package/dist/cjs/new-process-route-tree.cjs +754 -0
  48. package/dist/cjs/new-process-route-tree.cjs.map +1 -0
  49. package/dist/cjs/new-process-route-tree.d.cts +236 -0
  50. package/dist/cjs/not-found.cjs +26 -0
  51. package/dist/cjs/not-found.cjs.map +1 -0
  52. package/dist/cjs/not-found.d.cts +32 -0
  53. package/dist/cjs/path.cjs +252 -0
  54. package/dist/cjs/path.cjs.map +1 -0
  55. package/dist/cjs/path.d.cts +56 -0
  56. package/dist/cjs/qss.cjs +70 -0
  57. package/dist/cjs/qss.cjs.map +1 -0
  58. package/dist/cjs/qss.d.cts +33 -0
  59. package/dist/cjs/redirect.cjs +56 -0
  60. package/dist/cjs/redirect.cjs.map +1 -0
  61. package/dist/cjs/redirect.d.cts +77 -0
  62. package/dist/cjs/rewrite.cjs +68 -0
  63. package/dist/cjs/rewrite.cjs.map +1 -0
  64. package/dist/cjs/rewrite.d.cts +30 -0
  65. package/dist/cjs/root.cjs +7 -0
  66. package/dist/cjs/root.cjs.map +1 -0
  67. package/dist/cjs/root.d.cts +3 -0
  68. package/dist/cjs/route.cjs +100 -0
  69. package/dist/cjs/route.cjs.map +1 -0
  70. package/dist/cjs/route.d.cts +552 -0
  71. package/dist/cjs/routeInfo.d.cts +54 -0
  72. package/dist/cjs/router.cjs +1173 -0
  73. package/dist/cjs/router.cjs.map +1 -0
  74. package/dist/cjs/router.d.cts +734 -0
  75. package/dist/cjs/scroll-restoration-inline.cjs +6 -0
  76. package/dist/cjs/scroll-restoration-inline.cjs.map +1 -0
  77. package/dist/cjs/scroll-restoration-inline.d.cts +6 -0
  78. package/dist/cjs/scroll-restoration-script/client.cjs +9 -0
  79. package/dist/cjs/scroll-restoration-script/client.cjs.map +1 -0
  80. package/dist/cjs/scroll-restoration-script/client.d.cts +2 -0
  81. package/dist/cjs/scroll-restoration-script/server.cjs +30 -0
  82. package/dist/cjs/scroll-restoration-script/server.cjs.map +1 -0
  83. package/dist/cjs/scroll-restoration-script/server.d.cts +2 -0
  84. package/dist/cjs/scroll-restoration.cjs +191 -0
  85. package/dist/cjs/scroll-restoration.cjs.map +1 -0
  86. package/dist/cjs/scroll-restoration.d.cts +38 -0
  87. package/dist/cjs/searchMiddleware.cjs +55 -0
  88. package/dist/cjs/searchMiddleware.cjs.map +1 -0
  89. package/dist/cjs/searchMiddleware.d.cts +25 -0
  90. package/dist/cjs/searchParams.cjs +65 -0
  91. package/dist/cjs/searchParams.cjs.map +1 -0
  92. package/dist/cjs/searchParams.d.cts +31 -0
  93. package/dist/cjs/ssr/client.cjs +7 -0
  94. package/dist/cjs/ssr/client.d.cts +6 -0
  95. package/dist/cjs/ssr/constants.cjs +8 -0
  96. package/dist/cjs/ssr/constants.cjs.map +1 -0
  97. package/dist/cjs/ssr/constants.d.cts +3 -0
  98. package/dist/cjs/ssr/createRequestHandler.cjs +44 -0
  99. package/dist/cjs/ssr/createRequestHandler.cjs.map +1 -0
  100. package/dist/cjs/ssr/createRequestHandler.d.cts +9 -0
  101. package/dist/cjs/ssr/handlerCallback.cjs +8 -0
  102. package/dist/cjs/ssr/handlerCallback.cjs.map +1 -0
  103. package/dist/cjs/ssr/handlerCallback.d.cts +9 -0
  104. package/dist/cjs/ssr/headers.cjs +21 -0
  105. package/dist/cjs/ssr/headers.cjs.map +1 -0
  106. package/dist/cjs/ssr/headers.d.cts +3 -0
  107. package/dist/cjs/ssr/json.cjs +11 -0
  108. package/dist/cjs/ssr/json.cjs.map +1 -0
  109. package/dist/cjs/ssr/json.d.cts +10 -0
  110. package/dist/cjs/ssr/serializer/RawStream.cjs +287 -0
  111. package/dist/cjs/ssr/serializer/RawStream.cjs.map +1 -0
  112. package/dist/cjs/ssr/serializer/RawStream.d.cts +64 -0
  113. package/dist/cjs/ssr/serializer/ShallowErrorPlugin.cjs +32 -0
  114. package/dist/cjs/ssr/serializer/ShallowErrorPlugin.cjs.map +1 -0
  115. package/dist/cjs/ssr/serializer/ShallowErrorPlugin.d.cts +9 -0
  116. package/dist/cjs/ssr/serializer/seroval-plugins.cjs +13 -0
  117. package/dist/cjs/ssr/serializer/seroval-plugins.cjs.map +1 -0
  118. package/dist/cjs/ssr/serializer/seroval-plugins.d.cts +2 -0
  119. package/dist/cjs/ssr/serializer/transformer.cjs +53 -0
  120. package/dist/cjs/ssr/serializer/transformer.cjs.map +1 -0
  121. package/dist/cjs/ssr/serializer/transformer.d.cts +91 -0
  122. package/dist/cjs/ssr/server.cjs +13 -0
  123. package/dist/cjs/ssr/server.d.cts +6 -0
  124. package/dist/cjs/ssr/ssr-client.cjs +183 -0
  125. package/dist/cjs/ssr/ssr-client.cjs.map +1 -0
  126. package/dist/cjs/ssr/ssr-client.d.cts +10 -0
  127. package/dist/cjs/ssr/ssr-match-id.cjs +12 -0
  128. package/dist/cjs/ssr/ssr-match-id.cjs.map +1 -0
  129. package/dist/cjs/ssr/ssr-match-id.d.cts +2 -0
  130. package/dist/cjs/ssr/ssr-server.cjs +279 -0
  131. package/dist/cjs/ssr/ssr-server.cjs.map +1 -0
  132. package/dist/cjs/ssr/ssr-server.d.cts +42 -0
  133. package/dist/cjs/ssr/transformStreamWithRouter.cjs +327 -0
  134. package/dist/cjs/ssr/transformStreamWithRouter.cjs.map +1 -0
  135. package/dist/cjs/ssr/transformStreamWithRouter.d.cts +11 -0
  136. package/dist/cjs/ssr/tsrScript.cjs +6 -0
  137. package/dist/cjs/ssr/tsrScript.cjs.map +1 -0
  138. package/dist/cjs/ssr/tsrScript.d.cts +1 -0
  139. package/dist/cjs/ssr/types.d.cts +30 -0
  140. package/dist/cjs/stores.cjs +148 -0
  141. package/dist/cjs/stores.cjs.map +1 -0
  142. package/dist/cjs/stores.d.cts +70 -0
  143. package/dist/cjs/structuralSharing.d.cts +4 -0
  144. package/dist/cjs/typePrimitives.d.cts +65 -0
  145. package/dist/cjs/useLoaderData.d.cts +5 -0
  146. package/dist/cjs/useLoaderDeps.d.cts +5 -0
  147. package/dist/cjs/useNavigate.d.cts +3 -0
  148. package/dist/cjs/useParams.d.cts +5 -0
  149. package/dist/cjs/useRouteContext.d.cts +9 -0
  150. package/dist/cjs/useSearch.d.cts +5 -0
  151. package/dist/cjs/utils.cjs +339 -0
  152. package/dist/cjs/utils.cjs.map +1 -0
  153. package/dist/cjs/utils.d.cts +178 -0
  154. package/dist/cjs/validators.d.cts +51 -0
  155. package/dist/esm/Matches.d.ts +139 -0
  156. package/dist/esm/Matches.js +17 -0
  157. package/dist/esm/Matches.js.map +1 -0
  158. package/dist/esm/RouterProvider.d.ts +27 -0
  159. package/dist/esm/config.d.ts +17 -0
  160. package/dist/esm/config.js +11 -0
  161. package/dist/esm/config.js.map +1 -0
  162. package/dist/esm/defer.d.ts +37 -0
  163. package/dist/esm/defer.js +40 -0
  164. package/dist/esm/defer.js.map +1 -0
  165. package/dist/esm/fileRoute.d.ts +24 -0
  166. package/dist/esm/global.d.ts +7 -0
  167. package/dist/esm/hash-scroll.d.ts +7 -0
  168. package/dist/esm/hash-scroll.js +20 -0
  169. package/dist/esm/hash-scroll.js.map +1 -0
  170. package/dist/esm/history.d.ts +8 -0
  171. package/dist/esm/index.d.ts +53 -0
  172. package/dist/esm/index.js +24 -0
  173. package/dist/esm/invariant.d.ts +1 -0
  174. package/dist/esm/invariant.js +8 -0
  175. package/dist/esm/invariant.js.map +1 -0
  176. package/dist/esm/isServer/client.d.ts +1 -0
  177. package/dist/esm/isServer/client.js +6 -0
  178. package/dist/esm/isServer/client.js.map +1 -0
  179. package/dist/esm/isServer/development.d.ts +1 -0
  180. package/dist/esm/isServer/development.js +6 -0
  181. package/dist/esm/isServer/development.js.map +1 -0
  182. package/dist/esm/isServer/server.d.ts +1 -0
  183. package/dist/esm/isServer/server.js +6 -0
  184. package/dist/esm/isServer/server.js.map +1 -0
  185. package/dist/esm/link.d.ts +221 -0
  186. package/dist/esm/link.js +6 -0
  187. package/dist/esm/link.js.map +1 -0
  188. package/dist/esm/load-matches.d.ts +18 -0
  189. package/dist/esm/load-matches.js +657 -0
  190. package/dist/esm/load-matches.js.map +1 -0
  191. package/dist/esm/location.d.ts +50 -0
  192. package/dist/esm/lru-cache.d.ts +6 -0
  193. package/dist/esm/lru-cache.js +70 -0
  194. package/dist/esm/lru-cache.js.map +1 -0
  195. package/dist/esm/manifest.d.ts +35 -0
  196. package/dist/esm/manifest.js +17 -0
  197. package/dist/esm/manifest.js.map +1 -0
  198. package/dist/esm/new-process-route-tree.d.ts +236 -0
  199. package/dist/esm/new-process-route-tree.js +749 -0
  200. package/dist/esm/new-process-route-tree.js.map +1 -0
  201. package/dist/esm/not-found.d.ts +32 -0
  202. package/dist/esm/not-found.js +25 -0
  203. package/dist/esm/not-found.js.map +1 -0
  204. package/dist/esm/path.d.ts +56 -0
  205. package/dist/esm/path.js +243 -0
  206. package/dist/esm/path.js.map +1 -0
  207. package/dist/esm/qss.d.ts +33 -0
  208. package/dist/esm/qss.js +69 -0
  209. package/dist/esm/qss.js.map +1 -0
  210. package/dist/esm/redirect.d.ts +77 -0
  211. package/dist/esm/redirect.js +53 -0
  212. package/dist/esm/redirect.js.map +1 -0
  213. package/dist/esm/rewrite.d.ts +30 -0
  214. package/dist/esm/rewrite.js +65 -0
  215. package/dist/esm/rewrite.js.map +1 -0
  216. package/dist/esm/root.d.ts +3 -0
  217. package/dist/esm/root.js +7 -0
  218. package/dist/esm/root.js.map +1 -0
  219. package/dist/esm/route.d.ts +552 -0
  220. package/dist/esm/route.js +98 -0
  221. package/dist/esm/route.js.map +1 -0
  222. package/dist/esm/routeInfo.d.ts +54 -0
  223. package/dist/esm/router.d.ts +734 -0
  224. package/dist/esm/router.js +1165 -0
  225. package/dist/esm/router.js.map +1 -0
  226. package/dist/esm/scroll-restoration-inline.d.ts +6 -0
  227. package/dist/esm/scroll-restoration-inline.js +6 -0
  228. package/dist/esm/scroll-restoration-inline.js.map +1 -0
  229. package/dist/esm/scroll-restoration-script/client.d.ts +2 -0
  230. package/dist/esm/scroll-restoration-script/client.js +8 -0
  231. package/dist/esm/scroll-restoration-script/client.js.map +1 -0
  232. package/dist/esm/scroll-restoration-script/server.d.ts +2 -0
  233. package/dist/esm/scroll-restoration-script/server.js +29 -0
  234. package/dist/esm/scroll-restoration-script/server.js.map +1 -0
  235. package/dist/esm/scroll-restoration.d.ts +38 -0
  236. package/dist/esm/scroll-restoration.js +187 -0
  237. package/dist/esm/scroll-restoration.js.map +1 -0
  238. package/dist/esm/searchMiddleware.d.ts +25 -0
  239. package/dist/esm/searchMiddleware.js +54 -0
  240. package/dist/esm/searchMiddleware.js.map +1 -0
  241. package/dist/esm/searchParams.d.ts +31 -0
  242. package/dist/esm/searchParams.js +62 -0
  243. package/dist/esm/searchParams.js.map +1 -0
  244. package/dist/esm/ssr/client.d.ts +6 -0
  245. package/dist/esm/ssr/client.js +4 -0
  246. package/dist/esm/ssr/constants.d.ts +3 -0
  247. package/dist/esm/ssr/constants.js +7 -0
  248. package/dist/esm/ssr/constants.js.map +1 -0
  249. package/dist/esm/ssr/createRequestHandler.d.ts +9 -0
  250. package/dist/esm/ssr/createRequestHandler.js +44 -0
  251. package/dist/esm/ssr/createRequestHandler.js.map +1 -0
  252. package/dist/esm/ssr/handlerCallback.d.ts +9 -0
  253. package/dist/esm/ssr/handlerCallback.js +8 -0
  254. package/dist/esm/ssr/handlerCallback.js.map +1 -0
  255. package/dist/esm/ssr/headers.d.ts +3 -0
  256. package/dist/esm/ssr/headers.js +21 -0
  257. package/dist/esm/ssr/headers.js.map +1 -0
  258. package/dist/esm/ssr/json.d.ts +10 -0
  259. package/dist/esm/ssr/json.js +11 -0
  260. package/dist/esm/ssr/json.js.map +1 -0
  261. package/dist/esm/ssr/serializer/RawStream.d.ts +64 -0
  262. package/dist/esm/ssr/serializer/RawStream.js +282 -0
  263. package/dist/esm/ssr/serializer/RawStream.js.map +1 -0
  264. package/dist/esm/ssr/serializer/ShallowErrorPlugin.d.ts +9 -0
  265. package/dist/esm/ssr/serializer/ShallowErrorPlugin.js +33 -0
  266. package/dist/esm/ssr/serializer/ShallowErrorPlugin.js.map +1 -0
  267. package/dist/esm/ssr/serializer/seroval-plugins.d.ts +2 -0
  268. package/dist/esm/ssr/serializer/seroval-plugins.js +13 -0
  269. package/dist/esm/ssr/serializer/seroval-plugins.js.map +1 -0
  270. package/dist/esm/ssr/serializer/transformer.d.ts +91 -0
  271. package/dist/esm/ssr/serializer/transformer.js +51 -0
  272. package/dist/esm/ssr/serializer/transformer.js.map +1 -0
  273. package/dist/esm/ssr/server.d.ts +6 -0
  274. package/dist/esm/ssr/server.js +5 -0
  275. package/dist/esm/ssr/ssr-client.d.ts +10 -0
  276. package/dist/esm/ssr/ssr-client.js +183 -0
  277. package/dist/esm/ssr/ssr-client.js.map +1 -0
  278. package/dist/esm/ssr/ssr-match-id.d.ts +2 -0
  279. package/dist/esm/ssr/ssr-match-id.js +11 -0
  280. package/dist/esm/ssr/ssr-match-id.js.map +1 -0
  281. package/dist/esm/ssr/ssr-server.d.ts +42 -0
  282. package/dist/esm/ssr/ssr-server.js +277 -0
  283. package/dist/esm/ssr/ssr-server.js.map +1 -0
  284. package/dist/esm/ssr/transformStreamWithRouter.d.ts +11 -0
  285. package/dist/esm/ssr/transformStreamWithRouter.js +325 -0
  286. package/dist/esm/ssr/transformStreamWithRouter.js.map +1 -0
  287. package/dist/esm/ssr/tsrScript.d.ts +0 -0
  288. package/dist/esm/ssr/tsrScript.js +6 -0
  289. package/dist/esm/ssr/tsrScript.js.map +1 -0
  290. package/dist/esm/ssr/types.d.ts +30 -0
  291. package/dist/esm/stores.d.ts +70 -0
  292. package/dist/esm/stores.js +146 -0
  293. package/dist/esm/stores.js.map +1 -0
  294. package/dist/esm/structuralSharing.d.ts +4 -0
  295. package/dist/esm/typePrimitives.d.ts +65 -0
  296. package/dist/esm/useLoaderData.d.ts +5 -0
  297. package/dist/esm/useLoaderDeps.d.ts +5 -0
  298. package/dist/esm/useNavigate.d.ts +3 -0
  299. package/dist/esm/useParams.d.ts +5 -0
  300. package/dist/esm/useRouteContext.d.ts +9 -0
  301. package/dist/esm/useSearch.d.ts +5 -0
  302. package/dist/esm/utils.d.ts +178 -0
  303. package/dist/esm/utils.js +322 -0
  304. package/dist/esm/utils.js.map +1 -0
  305. package/dist/esm/validators.d.ts +51 -0
  306. package/package.json +200 -0
  307. package/skills/router-core/SKILL.md +139 -0
  308. package/skills/router-core/auth-and-guards/SKILL.md +458 -0
  309. package/skills/router-core/code-splitting/SKILL.md +322 -0
  310. package/skills/router-core/data-loading/SKILL.md +485 -0
  311. package/skills/router-core/navigation/SKILL.md +448 -0
  312. package/skills/router-core/not-found-and-errors/SKILL.md +435 -0
  313. package/skills/router-core/path-params/SKILL.md +382 -0
  314. package/skills/router-core/search-params/SKILL.md +349 -0
  315. package/skills/router-core/search-params/references/validation-patterns.md +379 -0
  316. package/skills/router-core/ssr/SKILL.md +437 -0
  317. package/skills/router-core/type-safety/SKILL.md +497 -0
  318. package/src/Matches.ts +291 -0
  319. package/src/RouterProvider.ts +47 -0
  320. package/src/config.ts +42 -0
  321. package/src/defer.ts +69 -0
  322. package/src/fileRoute.ts +164 -0
  323. package/src/global.ts +9 -0
  324. package/src/hash-scroll.ts +21 -0
  325. package/src/history.ts +9 -0
  326. package/src/index.ts +471 -0
  327. package/src/invariant.ts +3 -0
  328. package/src/isServer/client.ts +1 -0
  329. package/src/isServer/development.ts +2 -0
  330. package/src/isServer/server.ts +1 -0
  331. package/src/link.ts +704 -0
  332. package/src/load-matches.ts +1281 -0
  333. package/src/location.ts +51 -0
  334. package/src/lru-cache.ts +74 -0
  335. package/src/manifest.ts +68 -0
  336. package/src/new-process-route-tree.ts +1387 -0
  337. package/src/not-found.ts +41 -0
  338. package/src/path.ts +436 -0
  339. package/src/qss.ts +81 -0
  340. package/src/redirect.ts +179 -0
  341. package/src/rewrite.ts +93 -0
  342. package/src/root.ts +3 -0
  343. package/src/route.ts +2235 -0
  344. package/src/routeInfo.ts +235 -0
  345. package/src/router.ts +3207 -0
  346. package/src/scroll-restoration-inline.ts +81 -0
  347. package/src/scroll-restoration-script/client.ts +5 -0
  348. package/src/scroll-restoration-script/server.ts +64 -0
  349. package/src/scroll-restoration.ts +357 -0
  350. package/src/searchMiddleware.ts +76 -0
  351. package/src/searchParams.ts +90 -0
  352. package/src/ssr/client.ts +6 -0
  353. package/src/ssr/constants.ts +3 -0
  354. package/src/ssr/createRequestHandler.ts +98 -0
  355. package/src/ssr/handlerCallback.ts +15 -0
  356. package/src/ssr/headers.ts +40 -0
  357. package/src/ssr/json.ts +16 -0
  358. package/src/ssr/serializer/RawStream.ts +464 -0
  359. package/src/ssr/serializer/ShallowErrorPlugin.ts +43 -0
  360. package/src/ssr/serializer/seroval-plugins.ts +12 -0
  361. package/src/ssr/serializer/transformer.ts +312 -0
  362. package/src/ssr/server.ts +14 -0
  363. package/src/ssr/ssr-client.ts +313 -0
  364. package/src/ssr/ssr-match-id.ts +7 -0
  365. package/src/ssr/ssr-server.ts +425 -0
  366. package/src/ssr/transformStreamWithRouter.ts +493 -0
  367. package/src/ssr/tsrScript.ts +20 -0
  368. package/src/ssr/types.ts +41 -0
  369. package/src/stores.ts +342 -0
  370. package/src/structuralSharing.ts +7 -0
  371. package/src/typePrimitives.ts +181 -0
  372. package/src/useLoaderData.ts +20 -0
  373. package/src/useLoaderDeps.ts +13 -0
  374. package/src/useNavigate.ts +13 -0
  375. package/src/useParams.ts +20 -0
  376. package/src/useRouteContext.ts +39 -0
  377. package/src/useSearch.ts +20 -0
  378. package/src/utils.ts +708 -0
  379. package/src/validators.ts +121 -0
  380. package/src/vite-env.d.ts +4 -0
@@ -0,0 +1,425 @@
1
+ import { crossSerializeStream, getCrossReferenceHeader } from 'seroval'
2
+ import { invariant } from '../invariant'
3
+ import { decodePath } from '../utils'
4
+ import { createLRUCache } from '../lru-cache'
5
+ import minifiedTsrBootStrapScript from './tsrScript?script-string'
6
+ import { GLOBAL_TSR, TSR_SCRIPT_BARRIER_ID } from './constants'
7
+ import { dehydrateSsrMatchId } from './ssr-match-id'
8
+ import { defaultSerovalPlugins } from './serializer/seroval-plugins'
9
+ import { makeSsrSerovalPlugin } from './serializer/transformer'
10
+ import type { LRUCache } from '../lru-cache'
11
+ import type { DehydratedMatch, DehydratedRouter } from './types'
12
+ import type { AnySerializationAdapter } from './serializer/transformer'
13
+ import type { AnyRouter } from '../router'
14
+ import type { AnyRouteMatch } from '../Matches'
15
+ import type { Manifest, RouterManagedTag } from '../manifest'
16
+
17
+ declare module '../router' {
18
+ interface ServerSsr {
19
+ setRenderFinished: () => void
20
+ cleanup: () => void
21
+ }
22
+ interface RouterEvents {
23
+ onInjectedHtml: {
24
+ type: 'onInjectedHtml'
25
+ }
26
+ onSerializationFinished: {
27
+ type: 'onSerializationFinished'
28
+ }
29
+ }
30
+ }
31
+
32
+ const SCOPE_ID = 'tsr'
33
+
34
+ const TSR_PREFIX = GLOBAL_TSR + '.router='
35
+ const P_PREFIX = GLOBAL_TSR + '.p(()=>'
36
+ const P_SUFFIX = ')'
37
+
38
+ export function dehydrateMatch(match: AnyRouteMatch): DehydratedMatch {
39
+ const dehydratedMatch: DehydratedMatch = {
40
+ i: dehydrateSsrMatchId(match.id),
41
+ u: match.updatedAt,
42
+ s: match.status,
43
+ }
44
+
45
+ const properties = [
46
+ ['__beforeLoadContext', 'b'],
47
+ ['loaderData', 'l'],
48
+ ['error', 'e'],
49
+ ['ssr', 'ssr'],
50
+ ] as const
51
+
52
+ for (const [key, shorthand] of properties) {
53
+ if (match[key] !== undefined) {
54
+ dehydratedMatch[shorthand] = match[key]
55
+ }
56
+ }
57
+ if (match.globalNotFound) {
58
+ dehydratedMatch.g = true
59
+ }
60
+ return dehydratedMatch
61
+ }
62
+
63
+ const INITIAL_SCRIPTS = [
64
+ getCrossReferenceHeader(SCOPE_ID),
65
+ minifiedTsrBootStrapScript,
66
+ ]
67
+
68
+ class ScriptBuffer {
69
+ private router: AnyRouter | undefined
70
+ private _queue: Array<string>
71
+ private _scriptBarrierLifted = false
72
+ private _cleanedUp = false
73
+ private _pendingMicrotask = false
74
+
75
+ constructor(router: AnyRouter) {
76
+ this.router = router
77
+ // Copy INITIAL_SCRIPTS to avoid mutating the shared array
78
+ this._queue = INITIAL_SCRIPTS.slice()
79
+ }
80
+
81
+ enqueue(script: string) {
82
+ if (this._cleanedUp) return
83
+ this._queue.push(script)
84
+ // If barrier is lifted, schedule injection (if not already scheduled)
85
+ if (this._scriptBarrierLifted && !this._pendingMicrotask) {
86
+ this._pendingMicrotask = true
87
+ queueMicrotask(() => {
88
+ this._pendingMicrotask = false
89
+ this.injectBufferedScripts()
90
+ })
91
+ }
92
+ }
93
+
94
+ liftBarrier() {
95
+ if (this._scriptBarrierLifted || this._cleanedUp) return
96
+ this._scriptBarrierLifted = true
97
+ if (this._queue.length > 0 && !this._pendingMicrotask) {
98
+ this._pendingMicrotask = true
99
+ queueMicrotask(() => {
100
+ this._pendingMicrotask = false
101
+ this.injectBufferedScripts()
102
+ })
103
+ }
104
+ }
105
+
106
+ /**
107
+ * Flushes any pending scripts synchronously.
108
+ * Call this before emitting onSerializationFinished to ensure all scripts are injected.
109
+ *
110
+ * IMPORTANT: Only injects if the barrier has been lifted. Before the barrier is lifted,
111
+ * scripts should remain in the queue so takeBufferedScripts() can retrieve them
112
+ */
113
+ flush() {
114
+ if (!this._scriptBarrierLifted) return
115
+ if (this._cleanedUp) return
116
+ this._pendingMicrotask = false
117
+ const scriptsToInject = this.takeAll()
118
+ if (scriptsToInject && this.router?.serverSsr) {
119
+ this.router.serverSsr.injectScript(scriptsToInject)
120
+ }
121
+ }
122
+
123
+ takeAll() {
124
+ const bufferedScripts = this._queue
125
+ this._queue = []
126
+ if (bufferedScripts.length === 0) {
127
+ return undefined
128
+ }
129
+ // Optimization: if only one script, avoid join
130
+ if (bufferedScripts.length === 1) {
131
+ return bufferedScripts[0] + ';document.currentScript.remove()'
132
+ }
133
+ // Append cleanup script and join - avoid push() to not mutate then iterate
134
+ return bufferedScripts.join(';') + ';document.currentScript.remove()'
135
+ }
136
+
137
+ injectBufferedScripts() {
138
+ if (this._cleanedUp) return
139
+ // Early return if queue is empty (avoids unnecessary takeAll() call)
140
+ if (this._queue.length === 0) return
141
+ const scriptsToInject = this.takeAll()
142
+ if (scriptsToInject && this.router?.serverSsr) {
143
+ this.router.serverSsr.injectScript(scriptsToInject)
144
+ }
145
+ }
146
+
147
+ cleanup() {
148
+ this._cleanedUp = true
149
+ this._queue = []
150
+ this.router = undefined
151
+ }
152
+ }
153
+
154
+ const isProd = process.env.NODE_ENV === 'production'
155
+
156
+ type FilteredRoutes = Manifest['routes']
157
+
158
+ type ManifestLRU = LRUCache<string, FilteredRoutes>
159
+
160
+ const MANIFEST_CACHE_SIZE = 100
161
+ const manifestCaches = new WeakMap<Manifest, ManifestLRU>()
162
+
163
+ function getManifestCache(manifest: Manifest): ManifestLRU {
164
+ const cache = manifestCaches.get(manifest)
165
+ if (cache) return cache
166
+ const newCache = createLRUCache<string, FilteredRoutes>(MANIFEST_CACHE_SIZE)
167
+ manifestCaches.set(manifest, newCache)
168
+ return newCache
169
+ }
170
+
171
+ export function attachRouterServerSsrUtils({
172
+ router,
173
+ manifest,
174
+ }: {
175
+ router: AnyRouter
176
+ manifest: Manifest | undefined
177
+ }) {
178
+ router.ssr = {
179
+ manifest,
180
+ }
181
+ let _dehydrated = false
182
+ let _serializationFinished = false
183
+ const renderFinishedListeners: Array<() => void> = []
184
+ const serializationFinishedListeners: Array<() => void> = []
185
+ const scriptBuffer = new ScriptBuffer(router)
186
+ let injectedHtmlBuffer = ''
187
+
188
+ router.serverSsr = {
189
+ injectHtml: (html: string) => {
190
+ if (!html) return
191
+ // Buffer the HTML so it can be retrieved via takeBufferedHtml()
192
+ injectedHtmlBuffer += html
193
+ // Emit event to notify subscribers that new HTML is available
194
+ router.emit({
195
+ type: 'onInjectedHtml',
196
+ })
197
+ },
198
+ injectScript: (script: string) => {
199
+ if (!script) return
200
+ const html = `<script${router.options.ssr?.nonce ? ` nonce='${router.options.ssr.nonce}'` : ''}>${script}</script>`
201
+ router.serverSsr!.injectHtml(html)
202
+ },
203
+ dehydrate: async () => {
204
+ if (_dehydrated) {
205
+ if (process.env.NODE_ENV !== 'production') {
206
+ throw new Error('Invariant failed: router is already dehydrated!')
207
+ }
208
+
209
+ invariant()
210
+ }
211
+ let matchesToDehydrate = router.stores.activeMatchesSnapshot.state
212
+ if (router.isShell()) {
213
+ // In SPA mode we only want to dehydrate the root match
214
+ matchesToDehydrate = matchesToDehydrate.slice(0, 1)
215
+ }
216
+ const matches = matchesToDehydrate.map(dehydrateMatch)
217
+
218
+ let manifestToDehydrate: Manifest | undefined = undefined
219
+ // For currently matched routes, send full manifest (preloads + assets)
220
+ // For all other routes, only send assets (no preloads as they are handled via dynamic imports)
221
+ if (manifest) {
222
+ // Prod-only caching; in dev manifests may be replaced/updated (HMR)
223
+ const currentRouteIdsList = matchesToDehydrate.map((m) => m.routeId)
224
+ const manifestCacheKey = currentRouteIdsList.join('\0')
225
+
226
+ let filteredRoutes: FilteredRoutes | undefined
227
+
228
+ if (isProd) {
229
+ filteredRoutes = getManifestCache(manifest).get(manifestCacheKey)
230
+ }
231
+
232
+ if (!filteredRoutes) {
233
+ const currentRouteIds = new Set(currentRouteIdsList)
234
+ const nextFilteredRoutes: FilteredRoutes = {}
235
+
236
+ for (const routeId in manifest.routes) {
237
+ const routeManifest = manifest.routes[routeId]!
238
+ if (currentRouteIds.has(routeId)) {
239
+ nextFilteredRoutes[routeId] = routeManifest
240
+ } else if (
241
+ routeManifest.assets &&
242
+ routeManifest.assets.length > 0
243
+ ) {
244
+ nextFilteredRoutes[routeId] = {
245
+ assets: routeManifest.assets,
246
+ }
247
+ }
248
+ }
249
+
250
+ if (isProd) {
251
+ getManifestCache(manifest).set(manifestCacheKey, nextFilteredRoutes)
252
+ }
253
+
254
+ filteredRoutes = nextFilteredRoutes
255
+ }
256
+
257
+ manifestToDehydrate = {
258
+ routes: filteredRoutes,
259
+ }
260
+ }
261
+ const dehydratedRouter: DehydratedRouter = {
262
+ manifest: manifestToDehydrate,
263
+ matches,
264
+ }
265
+ const lastMatchId = matchesToDehydrate[matchesToDehydrate.length - 1]?.id
266
+ if (lastMatchId) {
267
+ dehydratedRouter.lastMatchId = dehydrateSsrMatchId(lastMatchId)
268
+ }
269
+ const dehydratedData = await router.options.dehydrate?.()
270
+ if (dehydratedData) {
271
+ dehydratedRouter.dehydratedData = dehydratedData
272
+ }
273
+ _dehydrated = true
274
+
275
+ const trackPlugins = { didRun: false }
276
+ const serializationAdapters = router.options.serializationAdapters as
277
+ | Array<AnySerializationAdapter>
278
+ | undefined
279
+ const plugins = serializationAdapters
280
+ ? serializationAdapters
281
+ .map((t) => makeSsrSerovalPlugin(t, trackPlugins))
282
+ .concat(defaultSerovalPlugins)
283
+ : defaultSerovalPlugins
284
+
285
+ const signalSerializationComplete = () => {
286
+ _serializationFinished = true
287
+ try {
288
+ serializationFinishedListeners.forEach((l) => l())
289
+ router.emit({ type: 'onSerializationFinished' })
290
+ } catch (err) {
291
+ console.error('Serialization listener error:', err)
292
+ } finally {
293
+ serializationFinishedListeners.length = 0
294
+ renderFinishedListeners.length = 0
295
+ }
296
+ }
297
+
298
+ crossSerializeStream(dehydratedRouter, {
299
+ refs: new Map(),
300
+ plugins,
301
+ onSerialize: (data, initial) => {
302
+ let serialized = initial ? TSR_PREFIX + data : data
303
+ if (trackPlugins.didRun) {
304
+ serialized = P_PREFIX + serialized + P_SUFFIX
305
+ }
306
+ scriptBuffer.enqueue(serialized)
307
+ },
308
+ scopeId: SCOPE_ID,
309
+ onDone: () => {
310
+ scriptBuffer.enqueue(GLOBAL_TSR + '.e()')
311
+ // Flush all pending scripts synchronously before signaling completion
312
+ // This ensures all scripts are injected before onSerializationFinished is emitted
313
+ scriptBuffer.flush()
314
+ signalSerializationComplete()
315
+ },
316
+ onError: (err) => {
317
+ console.error('Serialization error:', err)
318
+ signalSerializationComplete()
319
+ },
320
+ })
321
+ },
322
+ isDehydrated() {
323
+ return _dehydrated
324
+ },
325
+ isSerializationFinished() {
326
+ return _serializationFinished
327
+ },
328
+ onRenderFinished: (listener) => renderFinishedListeners.push(listener),
329
+ onSerializationFinished: (listener) =>
330
+ serializationFinishedListeners.push(listener),
331
+ setRenderFinished: () => {
332
+ // Wrap in try-catch to ensure scriptBuffer.liftBarrier() is always called
333
+ try {
334
+ renderFinishedListeners.forEach((l) => l())
335
+ } catch (err) {
336
+ console.error('Error in render finished listener:', err)
337
+ } finally {
338
+ // Clear listeners after calling them to prevent memory leaks
339
+ renderFinishedListeners.length = 0
340
+ }
341
+ scriptBuffer.liftBarrier()
342
+ },
343
+ takeBufferedScripts() {
344
+ const scripts = scriptBuffer.takeAll()
345
+ const serverBufferedScript: RouterManagedTag = {
346
+ tag: 'script',
347
+ attrs: {
348
+ nonce: router.options.ssr?.nonce,
349
+ className: '$tsr',
350
+ id: TSR_SCRIPT_BARRIER_ID,
351
+ },
352
+ children: scripts,
353
+ }
354
+ return serverBufferedScript
355
+ },
356
+ liftScriptBarrier() {
357
+ scriptBuffer.liftBarrier()
358
+ },
359
+ takeBufferedHtml() {
360
+ if (!injectedHtmlBuffer) {
361
+ return undefined
362
+ }
363
+ const buffered = injectedHtmlBuffer
364
+ injectedHtmlBuffer = ''
365
+ return buffered
366
+ },
367
+ cleanup() {
368
+ // Guard against multiple cleanup calls
369
+ if (!router.serverSsr) return
370
+ renderFinishedListeners.length = 0
371
+ serializationFinishedListeners.length = 0
372
+ injectedHtmlBuffer = ''
373
+ scriptBuffer.cleanup()
374
+ router.serverSsr = undefined
375
+ },
376
+ }
377
+ }
378
+
379
+ /**
380
+ * Get the origin for the request.
381
+ *
382
+ * SECURITY: We intentionally do NOT trust the Origin header for determining
383
+ * the router's origin. The Origin header can be spoofed by attackers, which
384
+ * could lead to SSRF-like vulnerabilities where redirects are constructed
385
+ * using a malicious origin (CVE-2024-34351).
386
+ *
387
+ * Instead, we derive the origin from request.url, which is typically set by
388
+ * the server infrastructure (not client-controlled headers).
389
+ *
390
+ * For applications behind proxies that need to trust forwarded headers,
391
+ * use the router's `origin` option to explicitly configure a trusted origin.
392
+ */
393
+ export function getOrigin(request: Request) {
394
+ try {
395
+ return new URL(request.url).origin
396
+ } catch {}
397
+ return 'http://localhost'
398
+ }
399
+
400
+ // server and browser can decode/encode characters differently in paths and search params.
401
+ // Server generally strictly follows the WHATWG URL Standard, while browsers may differ for legacy reasons.
402
+ // for example, in paths "|" is not encoded on the server but is encoded on chromium (and not on firefox) while "대" is encoded on both sides.
403
+ // Another anomaly is that in Node new URLSearchParams and new URL also decode/encode characters differently.
404
+ // new URLSearchParams() encodes "|" while new URL() does not, and in this instance
405
+ // chromium treats search params differently than paths, i.e. "|" is not encoded in search params.
406
+ export function getNormalizedURL(url: string | URL, base?: string | URL) {
407
+ // ensure backslashes are encoded correctly in the URL
408
+ if (typeof url === 'string') url = url.replace('\\', '%5C')
409
+
410
+ const rawUrl = new URL(url, base)
411
+ const { path: decodedPathname, handledProtocolRelativeURL } = decodePath(
412
+ rawUrl.pathname,
413
+ )
414
+ const searchParams = new URLSearchParams(rawUrl.search)
415
+ const normalizedHref =
416
+ decodedPathname +
417
+ (searchParams.size > 0 ? '?' : '') +
418
+ searchParams.toString() +
419
+ rawUrl.hash
420
+
421
+ return {
422
+ url: new URL(normalizedHref, rawUrl.origin),
423
+ handledProtocolRelativeURL,
424
+ }
425
+ }