@timber-js/app 0.2.0-alpha.7 → 0.2.0-alpha.71
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/LICENSE +8 -0
- package/dist/_chunks/{als-registry-B7DbZ2hS.js → als-registry-BJARkOcu.js} +1 -1
- package/dist/_chunks/als-registry-BJARkOcu.js.map +1 -0
- package/dist/_chunks/chunk-DYhsFzuS.js +33 -0
- package/dist/_chunks/{debug-gwlJkDuf.js → debug-ECi_61pb.js} +2 -2
- package/dist/_chunks/debug-ECi_61pb.js.map +1 -0
- package/dist/_chunks/define-CGuYoRHU.js +199 -0
- package/dist/_chunks/define-CGuYoRHU.js.map +1 -0
- package/dist/_chunks/define-Dz1bqwaS.js +106 -0
- package/dist/_chunks/define-Dz1bqwaS.js.map +1 -0
- package/dist/_chunks/define-cookie-B5mewxwM.js +93 -0
- package/dist/_chunks/define-cookie-B5mewxwM.js.map +1 -0
- package/dist/_chunks/error-boundary-D9hzsveV.js +216 -0
- package/dist/_chunks/error-boundary-D9hzsveV.js.map +1 -0
- package/dist/_chunks/{format-DviM89f0.js → format-Rn922VH2.js} +3 -20
- package/dist/_chunks/format-Rn922VH2.js.map +1 -0
- package/dist/_chunks/{tracing-Cwn7697K.js → handler-store-BVePM7hp.js} +68 -3
- package/dist/_chunks/handler-store-BVePM7hp.js.map +1 -0
- package/dist/_chunks/{interception-BOoWmLUA.js → interception-CEdHHviP.js} +171 -97
- package/dist/_chunks/interception-CEdHHviP.js.map +1 -0
- package/dist/_chunks/{metadata-routes-Cjmvi3rQ.js → metadata-routes-DS3eKNmf.js} +1 -1
- package/dist/_chunks/{metadata-routes-Cjmvi3rQ.js.map → metadata-routes-DS3eKNmf.js.map} +1 -1
- package/dist/_chunks/{request-context-DIkVh_jG.js → request-context-CywiO4jV.js} +181 -69
- package/dist/_chunks/request-context-CywiO4jV.js.map +1 -0
- package/dist/_chunks/schema-bridge-C4SwjCQD.js +86 -0
- package/dist/_chunks/schema-bridge-C4SwjCQD.js.map +1 -0
- package/dist/_chunks/segment-classify-BDNn6EzD.js +65 -0
- package/dist/_chunks/segment-classify-BDNn6EzD.js.map +1 -0
- package/dist/_chunks/segment-context-hzuJ048X.js +72 -0
- package/dist/_chunks/segment-context-hzuJ048X.js.map +1 -0
- package/dist/_chunks/stale-reload-BLUC_Pl_.js +64 -0
- package/dist/_chunks/stale-reload-BLUC_Pl_.js.map +1 -0
- package/dist/_chunks/{use-query-states-D5KaffOK.js → use-query-states-DAhgj8Gx.js} +1 -1
- package/dist/_chunks/use-query-states-DAhgj8Gx.js.map +1 -0
- package/dist/_chunks/wrappers-LZbghvn0.js +63 -0
- package/dist/_chunks/wrappers-LZbghvn0.js.map +1 -0
- package/dist/adapters/cloudflare-dev.d.ts +109 -0
- package/dist/adapters/cloudflare-dev.d.ts.map +1 -0
- package/dist/adapters/cloudflare-dev.js +73 -0
- package/dist/adapters/cloudflare-dev.js.map +1 -0
- package/dist/adapters/cloudflare.d.ts +148 -12
- package/dist/adapters/cloudflare.d.ts.map +1 -1
- package/dist/adapters/cloudflare.js +135 -11
- package/dist/adapters/cloudflare.js.map +1 -1
- package/dist/adapters/compress-module.d.ts.map +1 -1
- package/dist/adapters/nitro.d.ts +17 -1
- package/dist/adapters/nitro.d.ts.map +1 -1
- package/dist/adapters/nitro.js +56 -13
- package/dist/adapters/nitro.js.map +1 -1
- package/dist/cache/cache-api.d.ts +24 -0
- package/dist/cache/cache-api.d.ts.map +1 -0
- package/dist/cache/fast-hash.d.ts +22 -0
- package/dist/cache/fast-hash.d.ts.map +1 -0
- package/dist/cache/handler-store.d.ts +31 -0
- package/dist/cache/handler-store.d.ts.map +1 -0
- package/dist/cache/index.d.ts +7 -5
- package/dist/cache/index.d.ts.map +1 -1
- package/dist/cache/index.js +111 -73
- package/dist/cache/index.js.map +1 -1
- package/dist/cache/singleflight.d.ts +18 -1
- package/dist/cache/singleflight.d.ts.map +1 -1
- package/dist/cache/timber-cache.d.ts +1 -1
- package/dist/cache/timber-cache.d.ts.map +1 -1
- package/dist/client/error-boundary.d.ts +12 -5
- package/dist/client/error-boundary.d.ts.map +1 -1
- package/dist/client/error-boundary.js +1 -125
- package/dist/client/error-reconstituter.d.ts +54 -0
- package/dist/client/error-reconstituter.d.ts.map +1 -0
- package/dist/client/form.d.ts +2 -2
- package/dist/client/form.d.ts.map +1 -1
- package/dist/client/history.d.ts +19 -4
- package/dist/client/history.d.ts.map +1 -1
- package/dist/client/index.d.ts +6 -5
- package/dist/client/index.d.ts.map +1 -1
- package/dist/client/index.js +537 -166
- package/dist/client/index.js.map +1 -1
- package/dist/client/link-pending-store.d.ts +78 -0
- package/dist/client/link-pending-store.d.ts.map +1 -0
- package/dist/client/link.d.ts +90 -32
- package/dist/client/link.d.ts.map +1 -1
- package/dist/client/nav-link-store.d.ts +36 -0
- package/dist/client/nav-link-store.d.ts.map +1 -0
- package/dist/client/navigation-api-types.d.ts +90 -0
- package/dist/client/navigation-api-types.d.ts.map +1 -0
- package/dist/client/navigation-api.d.ts +115 -0
- package/dist/client/navigation-api.d.ts.map +1 -0
- package/dist/client/navigation-context.d.ts +13 -2
- package/dist/client/navigation-context.d.ts.map +1 -1
- package/dist/client/{transition-root.d.ts → navigation-root.d.ts} +42 -8
- package/dist/client/navigation-root.d.ts.map +1 -0
- package/dist/client/nuqs-adapter.d.ts.map +1 -1
- package/dist/client/router.d.ts +70 -4
- package/dist/client/router.d.ts.map +1 -1
- package/dist/client/rsc-fetch.d.ts +38 -3
- package/dist/client/rsc-fetch.d.ts.map +1 -1
- package/dist/client/segment-cache.d.ts +1 -1
- package/dist/client/segment-cache.d.ts.map +1 -1
- package/dist/client/segment-context.d.ts +1 -1
- package/dist/client/segment-context.d.ts.map +1 -1
- package/dist/client/segment-merger.d.ts.map +1 -1
- package/dist/client/segment-outlet.d.ts +63 -0
- package/dist/client/segment-outlet.d.ts.map +1 -0
- package/dist/client/ssr-data.d.ts +13 -4
- package/dist/client/ssr-data.d.ts.map +1 -1
- package/dist/client/stale-reload.d.ts +15 -0
- package/dist/client/stale-reload.d.ts.map +1 -1
- package/dist/client/top-loader.d.ts +3 -3
- package/dist/client/top-loader.d.ts.map +1 -1
- package/dist/client/use-params.d.ts +6 -4
- package/dist/client/use-params.d.ts.map +1 -1
- package/dist/client/use-query-states.d.ts +1 -1
- package/dist/client/use-query-states.d.ts.map +1 -1
- package/dist/codec.d.ts +23 -0
- package/dist/codec.d.ts.map +1 -0
- package/dist/codec.js +2 -0
- package/dist/cookies/define-cookie.d.ts +35 -14
- package/dist/cookies/define-cookie.d.ts.map +1 -1
- package/dist/cookies/index.d.ts +2 -0
- package/dist/cookies/index.d.ts.map +1 -1
- package/dist/cookies/index.js +3 -84
- package/dist/fonts/css.d.ts +1 -0
- package/dist/fonts/css.d.ts.map +1 -1
- package/dist/index.d.ts +154 -38
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +12092 -11916
- package/dist/index.js.map +1 -1
- package/dist/plugins/adapter-build.d.ts +1 -1
- package/dist/plugins/adapter-build.d.ts.map +1 -1
- package/dist/plugins/build-manifest.d.ts +2 -2
- package/dist/plugins/build-manifest.d.ts.map +1 -1
- package/dist/plugins/build-report.d.ts +3 -3
- package/dist/plugins/build-report.d.ts.map +1 -1
- package/dist/plugins/client-chunks.d.ts +32 -0
- package/dist/plugins/client-chunks.d.ts.map +1 -0
- package/dist/plugins/content.d.ts +1 -1
- package/dist/plugins/content.d.ts.map +1 -1
- package/dist/plugins/dev-browser-logs.d.ts +84 -0
- package/dist/plugins/dev-browser-logs.d.ts.map +1 -0
- package/dist/plugins/dev-error-overlay.d.ts +26 -1
- package/dist/plugins/dev-error-overlay.d.ts.map +1 -1
- package/dist/plugins/dev-logs.d.ts +1 -1
- package/dist/plugins/dev-logs.d.ts.map +1 -1
- package/dist/plugins/dev-server.d.ts +1 -1
- package/dist/plugins/dev-server.d.ts.map +1 -1
- package/dist/plugins/entries.d.ts +1 -1
- package/dist/plugins/entries.d.ts.map +1 -1
- package/dist/plugins/fonts.d.ts +19 -5
- package/dist/plugins/fonts.d.ts.map +1 -1
- package/dist/plugins/mdx.d.ts +1 -1
- package/dist/plugins/mdx.d.ts.map +1 -1
- package/dist/plugins/routing.d.ts +1 -1
- package/dist/plugins/routing.d.ts.map +1 -1
- package/dist/plugins/server-bundle.d.ts.map +1 -1
- package/dist/plugins/shims.d.ts +6 -5
- package/dist/plugins/shims.d.ts.map +1 -1
- package/dist/plugins/static-build.d.ts +1 -1
- package/dist/plugins/static-build.d.ts.map +1 -1
- package/dist/routing/codegen.d.ts +2 -2
- package/dist/routing/codegen.d.ts.map +1 -1
- package/dist/routing/index.d.ts +2 -0
- package/dist/routing/index.d.ts.map +1 -1
- package/dist/routing/index.js +3 -2
- package/dist/routing/scanner.d.ts.map +1 -1
- package/dist/routing/segment-classify.d.ts +46 -0
- package/dist/routing/segment-classify.d.ts.map +1 -0
- package/dist/routing/status-file-lint.d.ts +2 -1
- package/dist/routing/status-file-lint.d.ts.map +1 -1
- package/dist/routing/types.d.ts +16 -4
- package/dist/routing/types.d.ts.map +1 -1
- package/dist/rsc-runtime/rsc.d.ts +1 -1
- package/dist/rsc-runtime/rsc.d.ts.map +1 -1
- package/dist/rsc-runtime/ssr.d.ts +12 -0
- package/dist/rsc-runtime/ssr.d.ts.map +1 -1
- package/dist/schema-bridge.d.ts +76 -0
- package/dist/schema-bridge.d.ts.map +1 -0
- package/dist/search-params/define.d.ts +139 -0
- package/dist/search-params/define.d.ts.map +1 -0
- package/dist/search-params/index.d.ts +4 -6
- package/dist/search-params/index.d.ts.map +1 -1
- package/dist/search-params/index.js +4 -474
- package/dist/search-params/registry.d.ts +1 -1
- package/dist/search-params/wrappers.d.ts +53 -0
- package/dist/search-params/wrappers.d.ts.map +1 -0
- package/dist/segment-params/define.d.ts +78 -0
- package/dist/segment-params/define.d.ts.map +1 -0
- package/dist/segment-params/index.d.ts +7 -0
- package/dist/segment-params/index.d.ts.map +1 -0
- package/dist/segment-params/index.js +4 -0
- package/dist/server/access-gate.d.ts +4 -0
- package/dist/server/access-gate.d.ts.map +1 -1
- package/dist/server/action-client.d.ts +12 -1
- package/dist/server/action-client.d.ts.map +1 -1
- package/dist/server/action-encryption.d.ts +76 -0
- package/dist/server/action-encryption.d.ts.map +1 -0
- package/dist/server/action-handler.d.ts.map +1 -1
- package/dist/server/actions.d.ts +3 -6
- package/dist/server/actions.d.ts.map +1 -1
- package/dist/server/als-registry.d.ts +32 -4
- package/dist/server/als-registry.d.ts.map +1 -1
- package/dist/server/build-manifest.d.ts +2 -2
- package/dist/server/build-manifest.d.ts.map +1 -1
- package/dist/server/debug.d.ts +1 -1
- package/dist/server/default-logger.d.ts +22 -0
- package/dist/server/default-logger.d.ts.map +1 -0
- package/dist/server/deny-page-resolver.d.ts +52 -0
- package/dist/server/deny-page-resolver.d.ts.map +1 -0
- package/dist/server/deny-renderer.d.ts.map +1 -1
- package/dist/server/dev-warnings.d.ts +0 -14
- package/dist/server/dev-warnings.d.ts.map +1 -1
- package/dist/server/early-hints.d.ts +13 -5
- package/dist/server/early-hints.d.ts.map +1 -1
- package/dist/server/error-boundary-wrapper.d.ts +7 -1
- package/dist/server/error-boundary-wrapper.d.ts.map +1 -1
- package/dist/server/fallback-error.d.ts +4 -3
- package/dist/server/fallback-error.d.ts.map +1 -1
- package/dist/server/flight-injection-state.d.ts +66 -0
- package/dist/server/flight-injection-state.d.ts.map +1 -0
- package/dist/server/flight-scripts.d.ts +42 -0
- package/dist/server/flight-scripts.d.ts.map +1 -0
- package/dist/server/flush.d.ts.map +1 -1
- package/dist/server/form-data.d.ts +29 -0
- package/dist/server/form-data.d.ts.map +1 -1
- package/dist/server/html-injectors.d.ts +51 -11
- package/dist/server/html-injectors.d.ts.map +1 -1
- package/dist/server/index.d.ts +5 -3
- package/dist/server/index.d.ts.map +1 -1
- package/dist/server/index.js +2176 -1663
- package/dist/server/index.js.map +1 -1
- package/dist/server/logger.d.ts +25 -7
- package/dist/server/logger.d.ts.map +1 -1
- package/dist/server/middleware-runner.d.ts +19 -4
- package/dist/server/middleware-runner.d.ts.map +1 -1
- package/dist/server/node-stream-transforms.d.ts +113 -0
- package/dist/server/node-stream-transforms.d.ts.map +1 -0
- package/dist/server/page-deny-boundary.d.ts +31 -0
- package/dist/server/page-deny-boundary.d.ts.map +1 -0
- package/dist/server/pipeline-interception.d.ts +1 -1
- package/dist/server/pipeline-interception.d.ts.map +1 -1
- package/dist/server/pipeline-metadata.d.ts +6 -0
- package/dist/server/pipeline-metadata.d.ts.map +1 -1
- package/dist/server/pipeline.d.ts +32 -10
- package/dist/server/pipeline.d.ts.map +1 -1
- package/dist/server/primitives.d.ts +30 -3
- package/dist/server/primitives.d.ts.map +1 -1
- package/dist/server/render-timeout.d.ts +51 -0
- package/dist/server/render-timeout.d.ts.map +1 -0
- package/dist/server/request-context.d.ts +76 -37
- package/dist/server/request-context.d.ts.map +1 -1
- package/dist/server/route-element-builder.d.ts +27 -1
- package/dist/server/route-element-builder.d.ts.map +1 -1
- package/dist/server/route-handler.d.ts.map +1 -1
- package/dist/server/route-matcher.d.ts +9 -2
- package/dist/server/route-matcher.d.ts.map +1 -1
- package/dist/server/rsc-entry/api-handler.d.ts +2 -2
- package/dist/server/rsc-entry/api-handler.d.ts.map +1 -1
- package/dist/server/rsc-entry/error-renderer.d.ts +26 -13
- package/dist/server/rsc-entry/error-renderer.d.ts.map +1 -1
- package/dist/server/rsc-entry/helpers.d.ts +48 -5
- package/dist/server/rsc-entry/helpers.d.ts.map +1 -1
- package/dist/server/rsc-entry/index.d.ts +8 -3
- package/dist/server/rsc-entry/index.d.ts.map +1 -1
- package/dist/server/rsc-entry/rsc-payload.d.ts +3 -3
- package/dist/server/rsc-entry/rsc-payload.d.ts.map +1 -1
- package/dist/server/rsc-entry/rsc-stream.d.ts +10 -1
- package/dist/server/rsc-entry/rsc-stream.d.ts.map +1 -1
- package/dist/server/rsc-entry/ssr-bridge.d.ts +1 -1
- package/dist/server/rsc-entry/ssr-bridge.d.ts.map +1 -1
- package/dist/server/rsc-entry/ssr-renderer.d.ts +19 -4
- package/dist/server/rsc-entry/ssr-renderer.d.ts.map +1 -1
- package/dist/server/safe-load.d.ts +46 -0
- package/dist/server/safe-load.d.ts.map +1 -0
- package/dist/server/sitemap-generator.d.ts +129 -0
- package/dist/server/sitemap-generator.d.ts.map +1 -0
- package/dist/server/sitemap-handler.d.ts +22 -0
- package/dist/server/sitemap-handler.d.ts.map +1 -0
- package/dist/server/slot-resolver.d.ts +1 -1
- package/dist/server/slot-resolver.d.ts.map +1 -1
- package/dist/server/ssr-entry.d.ts +22 -0
- package/dist/server/ssr-entry.d.ts.map +1 -1
- package/dist/server/ssr-render.d.ts +39 -21
- package/dist/server/ssr-render.d.ts.map +1 -1
- package/dist/server/ssr-wrappers.d.ts +50 -0
- package/dist/server/ssr-wrappers.d.ts.map +1 -0
- package/dist/server/status-code-resolver.d.ts +1 -1
- package/dist/server/status-code-resolver.d.ts.map +1 -1
- package/dist/server/stream-utils.d.ts +36 -0
- package/dist/server/stream-utils.d.ts.map +1 -0
- package/dist/server/tracing.d.ts +10 -0
- package/dist/server/tracing.d.ts.map +1 -1
- package/dist/server/tree-builder.d.ts +22 -19
- package/dist/server/tree-builder.d.ts.map +1 -1
- package/dist/server/types.d.ts +1 -4
- package/dist/server/types.d.ts.map +1 -1
- package/dist/server/version-skew.d.ts +61 -0
- package/dist/server/version-skew.d.ts.map +1 -0
- package/dist/server/waituntil-bridge.d.ts.map +1 -1
- package/dist/shared/merge-search-params.d.ts +22 -0
- package/dist/shared/merge-search-params.d.ts.map +1 -0
- package/dist/shims/font-google.d.ts +1 -1
- package/dist/shims/font-google.d.ts.map +1 -1
- package/dist/shims/font-google.js +42 -0
- package/dist/shims/font-google.js.map +1 -0
- package/dist/shims/font-local.d.ts +26 -0
- package/dist/shims/font-local.d.ts.map +1 -0
- package/dist/shims/font-local.js +20 -0
- package/dist/shims/font-local.js.map +1 -0
- package/dist/shims/navigation-client.d.ts +1 -1
- package/dist/shims/navigation-client.d.ts.map +1 -1
- package/dist/shims/navigation.d.ts +1 -1
- package/dist/shims/navigation.d.ts.map +1 -1
- package/dist/utils/directive-parser.d.ts +5 -2
- package/dist/utils/directive-parser.d.ts.map +1 -1
- package/dist/utils/state-machine.d.ts +80 -0
- package/dist/utils/state-machine.d.ts.map +1 -0
- package/package.json +37 -17
- package/src/adapters/cloudflare-dev.ts +177 -0
- package/src/adapters/cloudflare.ts +342 -28
- package/src/adapters/compress-module.ts +24 -4
- package/src/adapters/nitro.ts +58 -9
- package/src/adapters/wrangler.d.ts +7 -0
- package/src/cache/cache-api.ts +38 -0
- package/src/cache/fast-hash.ts +34 -0
- package/src/cache/handler-store.ts +68 -0
- package/src/cache/index.ts +9 -5
- package/src/cache/singleflight.ts +62 -4
- package/src/cache/timber-cache.ts +40 -29
- package/src/cli.ts +0 -0
- package/src/client/browser-entry.ts +314 -142
- package/src/client/error-boundary.tsx +48 -16
- package/src/client/error-reconstituter.tsx +65 -0
- package/src/client/form.tsx +2 -2
- package/src/client/history.ts +26 -4
- package/src/client/index.ts +13 -4
- package/src/client/link-pending-store.ts +136 -0
- package/src/client/link.tsx +346 -105
- package/src/client/nav-link-store.ts +47 -0
- package/src/client/navigation-api-types.ts +112 -0
- package/src/client/navigation-api.ts +332 -0
- package/src/client/navigation-context.ts +27 -6
- package/src/client/navigation-root.tsx +346 -0
- package/src/client/nuqs-adapter.tsx +16 -3
- package/src/client/router.ts +302 -77
- package/src/client/rsc-fetch.ts +93 -5
- package/src/client/segment-cache.ts +1 -1
- package/src/client/segment-context.ts +6 -1
- package/src/client/segment-merger.ts +2 -8
- package/src/client/segment-outlet.tsx +86 -0
- package/src/client/ssr-data.ts +13 -5
- package/src/client/stale-reload.ts +73 -6
- package/src/client/top-loader.tsx +22 -13
- package/src/client/use-navigation-pending.ts +1 -1
- package/src/client/use-params.ts +7 -5
- package/src/client/use-query-states.ts +2 -2
- package/src/codec.ts +34 -0
- package/src/cookies/define-cookie.ts +72 -21
- package/src/cookies/index.ts +7 -0
- package/src/fonts/css.ts +2 -1
- package/src/index.ts +328 -92
- package/src/plugins/adapter-build.ts +8 -2
- package/src/plugins/build-manifest.ts +13 -2
- package/src/plugins/build-report.ts +3 -3
- package/src/plugins/client-chunks.ts +65 -0
- package/src/plugins/content.ts +1 -1
- package/src/plugins/dev-browser-logs.ts +288 -0
- package/src/plugins/dev-error-overlay.ts +70 -1
- package/src/plugins/dev-logs.ts +1 -1
- package/src/plugins/dev-server.ts +55 -9
- package/src/plugins/entries.ts +70 -9
- package/src/plugins/fonts.ts +167 -61
- package/src/plugins/mdx.ts +1 -1
- package/src/plugins/routing.ts +57 -17
- package/src/plugins/server-action-exports.ts +1 -1
- package/src/plugins/server-bundle.ts +32 -1
- package/src/plugins/shims.ts +76 -33
- package/src/plugins/static-build.ts +10 -6
- package/src/routing/codegen.ts +165 -105
- package/src/routing/index.ts +2 -0
- package/src/routing/scanner.ts +93 -23
- package/src/routing/segment-classify.ts +89 -0
- package/src/routing/status-file-lint.ts +3 -2
- package/src/routing/types.ts +17 -4
- package/src/rsc-runtime/rsc.ts +2 -0
- package/src/rsc-runtime/ssr.ts +50 -0
- package/src/rsc-runtime/vendor-types.d.ts +7 -0
- package/src/{search-params/codecs.ts → schema-bridge.ts} +57 -20
- package/src/search-params/define.ts +482 -0
- package/src/search-params/index.ts +13 -19
- package/src/search-params/registry.ts +1 -1
- package/src/search-params/wrappers.ts +85 -0
- package/src/segment-params/define.ts +279 -0
- package/src/segment-params/index.ts +28 -0
- package/src/server/access-gate.tsx +70 -29
- package/src/server/action-client.ts +28 -3
- package/src/server/action-encryption.ts +144 -0
- package/src/server/action-handler.ts +20 -3
- package/src/server/actions.ts +10 -9
- package/src/server/als-registry.ts +32 -4
- package/src/server/build-manifest.ts +10 -4
- package/src/server/compress.ts +25 -7
- package/src/server/debug.ts +1 -1
- package/src/server/default-logger.ts +99 -0
- package/src/server/deny-page-resolver.ts +154 -0
- package/src/server/deny-renderer.ts +24 -38
- package/src/server/dev-warnings.ts +2 -28
- package/src/server/early-hints.ts +36 -15
- package/src/server/error-boundary-wrapper.ts +74 -22
- package/src/server/fallback-error.ts +31 -15
- package/src/server/flight-injection-state.ts +113 -0
- package/src/server/flight-scripts.ts +62 -0
- package/src/server/flush.ts +2 -1
- package/src/server/form-data.ts +76 -0
- package/src/server/html-injectors.ts +277 -117
- package/src/server/index.ts +9 -5
- package/src/server/logger.ts +44 -36
- package/src/server/middleware-runner.ts +31 -4
- package/src/server/node-stream-transforms.ts +509 -0
- package/src/server/page-deny-boundary.tsx +56 -0
- package/src/server/pipeline-interception.ts +17 -16
- package/src/server/pipeline-metadata.ts +13 -0
- package/src/server/pipeline.ts +195 -51
- package/src/server/primitives.ts +47 -5
- package/src/server/render-timeout.ts +108 -0
- package/src/server/request-context.ts +240 -117
- package/src/server/route-element-builder.ts +284 -197
- package/src/server/route-handler.ts +24 -4
- package/src/server/route-matcher.ts +24 -20
- package/src/server/rsc-entry/api-handler.ts +15 -16
- package/src/server/rsc-entry/error-renderer.ts +300 -89
- package/src/server/rsc-entry/helpers.ts +134 -5
- package/src/server/rsc-entry/index.ts +202 -113
- package/src/server/rsc-entry/rsc-payload.ts +100 -21
- package/src/server/rsc-entry/rsc-stream.ts +74 -18
- package/src/server/rsc-entry/ssr-bridge.ts +14 -5
- package/src/server/rsc-entry/ssr-renderer.ts +173 -40
- package/src/server/safe-load.ts +60 -0
- package/src/server/sitemap-generator.ts +338 -0
- package/src/server/sitemap-handler.ts +126 -0
- package/src/server/slot-resolver.ts +243 -228
- package/src/server/ssr-entry.ts +211 -32
- package/src/server/ssr-render.ts +289 -67
- package/src/server/ssr-wrappers.tsx +139 -0
- package/src/server/status-code-resolver.ts +1 -1
- package/src/server/stream-utils.ts +213 -0
- package/src/server/tracing.ts +37 -3
- package/src/server/tree-builder.ts +92 -58
- package/src/server/types.ts +3 -6
- package/src/server/version-skew.ts +104 -0
- package/src/server/waituntil-bridge.ts +4 -1
- package/src/shared/merge-search-params.ts +55 -0
- package/src/shims/font-google.ts +1 -1
- package/src/shims/font-local.ts +34 -0
- package/src/shims/navigation-client.ts +1 -1
- package/src/shims/navigation.ts +2 -1
- package/src/utils/directive-parser.ts +5 -2
- package/src/utils/state-machine.ts +111 -0
- package/dist/_chunks/als-registry-B7DbZ2hS.js.map +0 -1
- package/dist/_chunks/debug-gwlJkDuf.js.map +0 -1
- package/dist/_chunks/format-DviM89f0.js.map +0 -1
- package/dist/_chunks/interception-BOoWmLUA.js.map +0 -1
- package/dist/_chunks/request-context-DIkVh_jG.js.map +0 -1
- package/dist/_chunks/ssr-data-MjmprTmO.js +0 -88
- package/dist/_chunks/ssr-data-MjmprTmO.js.map +0 -1
- package/dist/_chunks/tracing-Cwn7697K.js.map +0 -1
- package/dist/_chunks/use-cookie-DX-l1_5E.js +0 -91
- package/dist/_chunks/use-cookie-DX-l1_5E.js.map +0 -1
- package/dist/_chunks/use-query-states-D5KaffOK.js.map +0 -1
- package/dist/cache/register-cached-function.d.ts +0 -17
- package/dist/cache/register-cached-function.d.ts.map +0 -1
- package/dist/client/error-boundary.js.map +0 -1
- package/dist/client/link-status-provider.d.ts +0 -11
- package/dist/client/link-status-provider.d.ts.map +0 -1
- package/dist/client/transition-root.d.ts.map +0 -1
- package/dist/cookies/index.js.map +0 -1
- package/dist/plugins/cache-transform.d.ts +0 -36
- package/dist/plugins/cache-transform.d.ts.map +0 -1
- package/dist/plugins/dynamic-transform.d.ts +0 -72
- package/dist/plugins/dynamic-transform.d.ts.map +0 -1
- package/dist/search-params/analyze.d.ts +0 -54
- package/dist/search-params/analyze.d.ts.map +0 -1
- package/dist/search-params/builtin-codecs.d.ts +0 -105
- package/dist/search-params/builtin-codecs.d.ts.map +0 -1
- package/dist/search-params/codecs.d.ts +0 -53
- package/dist/search-params/codecs.d.ts.map +0 -1
- package/dist/search-params/create.d.ts +0 -106
- package/dist/search-params/create.d.ts.map +0 -1
- package/dist/search-params/index.js.map +0 -1
- package/dist/server/prerender.d.ts +0 -77
- package/dist/server/prerender.d.ts.map +0 -1
- package/dist/server/response-cache.d.ts +0 -53
- package/dist/server/response-cache.d.ts.map +0 -1
- package/src/cache/register-cached-function.ts +0 -99
- package/src/client/link-status-provider.tsx +0 -30
- package/src/client/transition-root.tsx +0 -160
- package/src/plugins/cache-transform.ts +0 -199
- package/src/plugins/dynamic-transform.ts +0 -161
- package/src/search-params/analyze.ts +0 -192
- package/src/search-params/builtin-codecs.ts +0 -228
- package/src/search-params/create.ts +0 -321
- package/src/server/prerender.ts +0 -139
- package/src/server/response-cache.ts +0 -277
|
@@ -1,199 +0,0 @@
|
|
|
1
|
-
import type { Plugin } from 'vite';
|
|
2
|
-
import { findFunctionsWithDirective, containsDirective } from '#/utils/directive-parser.js';
|
|
3
|
-
|
|
4
|
-
/**
|
|
5
|
-
* Parse a cacheLife duration string to seconds.
|
|
6
|
-
* Supports: '30s', '5m', '1h', '2d', '1w', or a plain number (seconds).
|
|
7
|
-
*/
|
|
8
|
-
export function parseCacheLife(value: string | number): number {
|
|
9
|
-
if (typeof value === 'number') return value;
|
|
10
|
-
|
|
11
|
-
const match = value.match(/^(\d+)(s|m|h|d|w)$/);
|
|
12
|
-
if (!match) {
|
|
13
|
-
throw new Error(
|
|
14
|
-
`Invalid cacheLife value: "${value}". Expected format: "30s", "5m", "1h", "2d", "1w", or a number.`
|
|
15
|
-
);
|
|
16
|
-
}
|
|
17
|
-
|
|
18
|
-
const amount = parseInt(match[1], 10);
|
|
19
|
-
const unit = match[2];
|
|
20
|
-
const multipliers: Record<string, number> = { s: 1, m: 60, h: 3600, d: 86400, w: 604800 };
|
|
21
|
-
return amount * multipliers[unit];
|
|
22
|
-
}
|
|
23
|
-
|
|
24
|
-
// Default TTL when no cacheLife() is specified (Infinity means cache until explicit invalidation).
|
|
25
|
-
const DEFAULT_TTL = Infinity;
|
|
26
|
-
|
|
27
|
-
export interface CacheTransformWarning {
|
|
28
|
-
message: string;
|
|
29
|
-
functionName: string;
|
|
30
|
-
}
|
|
31
|
-
|
|
32
|
-
interface TransformResult {
|
|
33
|
-
code: string;
|
|
34
|
-
map?: null;
|
|
35
|
-
warnings?: CacheTransformWarning[];
|
|
36
|
-
}
|
|
37
|
-
|
|
38
|
-
/**
|
|
39
|
-
* Match cacheLife() calls: cacheLife('1h'), cacheLife("5m"), cacheLife(300)
|
|
40
|
-
*/
|
|
41
|
-
const CACHE_LIFE_PATTERN = /cacheLife\(\s*(?:'([^']+)'|"([^"]+)"|(\d+))\s*\)/;
|
|
42
|
-
|
|
43
|
-
/**
|
|
44
|
-
* Strip the 'use cache' directive and cacheLife() call from a function body.
|
|
45
|
-
* Returns the cleaned body and the extracted TTL.
|
|
46
|
-
*/
|
|
47
|
-
function extractCacheDirectives(body: string): { cleanBody: string; ttl: number } {
|
|
48
|
-
let ttl = DEFAULT_TTL;
|
|
49
|
-
|
|
50
|
-
// Remove 'use cache' / "use cache" directive (including optional semicolon and newline)
|
|
51
|
-
let cleanBody = body.replace(/\s*['"]use cache['"];?\s*\n?/, '\n');
|
|
52
|
-
|
|
53
|
-
// Extract and remove cacheLife() calls
|
|
54
|
-
const lifeMatch = cleanBody.match(CACHE_LIFE_PATTERN);
|
|
55
|
-
if (lifeMatch) {
|
|
56
|
-
const value = lifeMatch[1] || lifeMatch[2] || parseInt(lifeMatch[3], 10);
|
|
57
|
-
ttl = parseCacheLife(value);
|
|
58
|
-
cleanBody = cleanBody.replace(/\s*cacheLife\([^)]*\);?\s*\n?/, '\n');
|
|
59
|
-
}
|
|
60
|
-
|
|
61
|
-
return { cleanBody, ttl };
|
|
62
|
-
}
|
|
63
|
-
|
|
64
|
-
/**
|
|
65
|
-
* Determine if a function name is a React component (PascalCase).
|
|
66
|
-
*/
|
|
67
|
-
function isComponentName(name: string): boolean {
|
|
68
|
-
return /^[A-Z]/.test(name);
|
|
69
|
-
}
|
|
70
|
-
|
|
71
|
-
/**
|
|
72
|
-
* Pattern matching page/layout file conventions in a dynamic route segment.
|
|
73
|
-
* Matches paths like: app/[slug]/page.tsx, app/[id]/layout.ts, etc.
|
|
74
|
-
*/
|
|
75
|
-
const DYNAMIC_ROUTE_PAGE_PATTERN = /\/\[[^\]]+\].*\/(page|layout)\.[jt]sx?$/;
|
|
76
|
-
|
|
77
|
-
/**
|
|
78
|
-
* Detect if a function declaration has Promise-typed parameters.
|
|
79
|
-
*
|
|
80
|
-
* Checks for common patterns:
|
|
81
|
-
* - `params: Promise<...>`
|
|
82
|
-
* - `{ params }: { params: Promise<...> }`
|
|
83
|
-
* - `props: { params: Promise<...> }`
|
|
84
|
-
*/
|
|
85
|
-
const PROMISE_PARAMS_PATTERN = /params\s*(?::|.*?:)\s*Promise\s*</;
|
|
86
|
-
|
|
87
|
-
/**
|
|
88
|
-
* Check if a 'use cache' function in a dynamic route page/layout receives
|
|
89
|
-
* Promise-typed params, which are not serializable as cache keys.
|
|
90
|
-
*/
|
|
91
|
-
export function detectPromiseParamsWarning(
|
|
92
|
-
declaration: string,
|
|
93
|
-
functionName: string,
|
|
94
|
-
fileId: string
|
|
95
|
-
): CacheTransformWarning | null {
|
|
96
|
-
if (!DYNAMIC_ROUTE_PAGE_PATTERN.test(fileId)) return null;
|
|
97
|
-
if (!PROMISE_PARAMS_PATTERN.test(declaration)) return null;
|
|
98
|
-
|
|
99
|
-
return {
|
|
100
|
-
message:
|
|
101
|
-
`'use cache' on "${functionName}" in "${fileId}" receives Promise params. ` +
|
|
102
|
-
`Promise is not serializable as a cache key and will cause runtime errors. ` +
|
|
103
|
-
`Either remove 'use cache' or await the params before using them in a separate cached function.`,
|
|
104
|
-
functionName,
|
|
105
|
-
};
|
|
106
|
-
}
|
|
107
|
-
|
|
108
|
-
/**
|
|
109
|
-
* Transform source code containing 'use cache' directives into
|
|
110
|
-
* registerCachedFunction() calls.
|
|
111
|
-
*
|
|
112
|
-
* Returns null if no transformations were made.
|
|
113
|
-
*/
|
|
114
|
-
export function transformUseCache(code: string, fileId: string): TransformResult | null {
|
|
115
|
-
if (!containsDirective(code, 'use cache')) return null;
|
|
116
|
-
|
|
117
|
-
const functions = findFunctionsWithDirective(code, 'use cache');
|
|
118
|
-
if (functions.length === 0) return null;
|
|
119
|
-
|
|
120
|
-
let result = code;
|
|
121
|
-
let needsImport = false;
|
|
122
|
-
const warnings: CacheTransformWarning[] = [];
|
|
123
|
-
|
|
124
|
-
// Process functions from end to start (sorted descending by start position)
|
|
125
|
-
for (const fn of functions) {
|
|
126
|
-
// Warn if function receives Promise params in a dynamic route page/layout
|
|
127
|
-
const promiseWarning = detectPromiseParamsWarning(fn.declaration, fn.name, fileId);
|
|
128
|
-
if (promiseWarning) warnings.push(promiseWarning);
|
|
129
|
-
const { cleanBody, ttl } = extractCacheDirectives(fn.bodyContent);
|
|
130
|
-
const stableId = `${fileId}#${fn.name}`;
|
|
131
|
-
const isComponent = isComponentName(fn.name);
|
|
132
|
-
|
|
133
|
-
// Build the options object
|
|
134
|
-
const optsParts = [`ttl: ${ttl === Infinity ? 'Infinity' : ttl}`];
|
|
135
|
-
optsParts.push(`id: '${stableId}'`);
|
|
136
|
-
if (isComponent) {
|
|
137
|
-
optsParts.push('isComponent: true');
|
|
138
|
-
}
|
|
139
|
-
const optsStr = `{ ${optsParts.join(', ')} }`;
|
|
140
|
-
|
|
141
|
-
// Build the replacement
|
|
142
|
-
let replacement: string;
|
|
143
|
-
if (fn.isArrow) {
|
|
144
|
-
// const Name = async (...) => { body } → const Name = registerCachedFunction(async (...) => { body }, opts)
|
|
145
|
-
const arrowSig = fn.declaration.replace(/^(?:const|let|var)\s+\w+\s*=\s*/, '');
|
|
146
|
-
replacement = `const ${fn.name} = registerCachedFunction(${arrowSig} {${cleanBody}}, ${optsStr})`;
|
|
147
|
-
} else {
|
|
148
|
-
// async function Name(...) { body } → const Name = registerCachedFunction(async function Name(...) { body }, opts)
|
|
149
|
-
const fnDecl = fn.declaration.replace(/^(?:export\s+default\s+|export\s+)?/, '');
|
|
150
|
-
const exportPrefix = fn.prefix.includes('default')
|
|
151
|
-
? 'export default '
|
|
152
|
-
: fn.prefix.includes('export')
|
|
153
|
-
? 'export '
|
|
154
|
-
: '';
|
|
155
|
-
|
|
156
|
-
replacement = `${exportPrefix}const ${fn.name} = registerCachedFunction(${fnDecl} {${cleanBody}}, ${optsStr})`;
|
|
157
|
-
}
|
|
158
|
-
|
|
159
|
-
result = result.slice(0, fn.start) + replacement + result.slice(fn.end);
|
|
160
|
-
needsImport = true;
|
|
161
|
-
}
|
|
162
|
-
|
|
163
|
-
if (needsImport) {
|
|
164
|
-
// Add the import at the top of the file
|
|
165
|
-
result = `import { registerCachedFunction } from '@timber-js/app/cache';\n` + result;
|
|
166
|
-
}
|
|
167
|
-
|
|
168
|
-
return { code: result, map: null, warnings: warnings.length > 0 ? warnings : undefined };
|
|
169
|
-
}
|
|
170
|
-
|
|
171
|
-
/**
|
|
172
|
-
* Vite plugin: timber-cache
|
|
173
|
-
*
|
|
174
|
-
* Transforms 'use cache' directives into registerCachedFunction() calls.
|
|
175
|
-
* Only runs in the RSC environment.
|
|
176
|
-
*/
|
|
177
|
-
export function cacheTransformPlugin(): Plugin {
|
|
178
|
-
return {
|
|
179
|
-
name: 'timber-cache',
|
|
180
|
-
|
|
181
|
-
transform(code, id) {
|
|
182
|
-
// Only transform in RSC environment
|
|
183
|
-
// Skip node_modules and non-JS/TS files
|
|
184
|
-
if (id.includes('node_modules')) return null;
|
|
185
|
-
if (!/\.[jt]sx?$/.test(id)) return null;
|
|
186
|
-
|
|
187
|
-
// Quick bail-out: no 'use cache' directive in this file
|
|
188
|
-
if (!containsDirective(code, 'use cache')) return null;
|
|
189
|
-
|
|
190
|
-
const result = transformUseCache(code, id);
|
|
191
|
-
if (result?.warnings) {
|
|
192
|
-
for (const w of result.warnings) {
|
|
193
|
-
this.warn(`[timber] ${w.message}`);
|
|
194
|
-
}
|
|
195
|
-
}
|
|
196
|
-
return result;
|
|
197
|
-
},
|
|
198
|
-
};
|
|
199
|
-
}
|
|
@@ -1,161 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* timber-dynamic-transform — Vite sub-plugin for 'use dynamic' directive.
|
|
3
|
-
*
|
|
4
|
-
* Detects `'use dynamic'` directives in server component function bodies
|
|
5
|
-
* and transforms them into `markDynamic()` runtime calls. The directive
|
|
6
|
-
* declares a dynamic boundary — the component and its subtree opt out of
|
|
7
|
-
* the pre-rendered shell and render per-request.
|
|
8
|
-
*
|
|
9
|
-
* - In `output: 'static'` mode, `'use dynamic'` is a build error.
|
|
10
|
-
* - In standard SSR routes (no prerender.ts), the directive is a no-op
|
|
11
|
-
* (everything is already per-request), but the transform still runs
|
|
12
|
-
* so the runtime can skip unnecessary work.
|
|
13
|
-
*
|
|
14
|
-
* Design doc: design/15-future-prerendering.md §"'use dynamic'"
|
|
15
|
-
*/
|
|
16
|
-
|
|
17
|
-
import type { Plugin } from 'vite';
|
|
18
|
-
import type { PluginContext } from '#/index.js';
|
|
19
|
-
import { findFunctionsWithDirective, containsDirective } from '#/utils/directive-parser.js';
|
|
20
|
-
|
|
21
|
-
// ---------------------------------------------------------------------------
|
|
22
|
-
// Detection
|
|
23
|
-
// ---------------------------------------------------------------------------
|
|
24
|
-
|
|
25
|
-
/**
|
|
26
|
-
* Quick check: does this source file contain 'use dynamic' anywhere?
|
|
27
|
-
* Used as a fast bail-out before doing expensive AST parsing.
|
|
28
|
-
*/
|
|
29
|
-
export function containsUseDynamic(code: string): boolean {
|
|
30
|
-
return containsDirective(code, 'use dynamic');
|
|
31
|
-
}
|
|
32
|
-
|
|
33
|
-
// ---------------------------------------------------------------------------
|
|
34
|
-
// Transform
|
|
35
|
-
// ---------------------------------------------------------------------------
|
|
36
|
-
|
|
37
|
-
interface TransformResult {
|
|
38
|
-
code: string;
|
|
39
|
-
map?: null;
|
|
40
|
-
}
|
|
41
|
-
|
|
42
|
-
/**
|
|
43
|
-
* Find function declarations/expressions containing 'use dynamic' and
|
|
44
|
-
* transform them into markDynamic() calls.
|
|
45
|
-
*
|
|
46
|
-
* Input:
|
|
47
|
-
* ```tsx
|
|
48
|
-
* export default async function AddToCartButton({ productId }) {
|
|
49
|
-
* 'use dynamic'
|
|
50
|
-
* const user = await getUser()
|
|
51
|
-
* return <button>Add to cart</button>
|
|
52
|
-
* }
|
|
53
|
-
* ```
|
|
54
|
-
*
|
|
55
|
-
* Output:
|
|
56
|
-
* ```tsx
|
|
57
|
-
* import { markDynamic as __markDynamic } from '@timber-js/app/runtime';
|
|
58
|
-
* export default async function AddToCartButton({ productId }) {
|
|
59
|
-
* __markDynamic();
|
|
60
|
-
* const user = await getUser()
|
|
61
|
-
* return <button>Add to cart</button>
|
|
62
|
-
* }
|
|
63
|
-
* ```
|
|
64
|
-
*
|
|
65
|
-
* The markDynamic() call registers the component boundary as dynamic
|
|
66
|
-
* at render time. The pre-render pass uses this to know which subtrees
|
|
67
|
-
* to skip and leave as holes for per-request rendering.
|
|
68
|
-
*/
|
|
69
|
-
export function transformUseDynamic(code: string): TransformResult | null {
|
|
70
|
-
if (!containsUseDynamic(code)) return null;
|
|
71
|
-
|
|
72
|
-
const functions = findFunctionsWithDirective(code, 'use dynamic');
|
|
73
|
-
if (functions.length === 0) return null;
|
|
74
|
-
|
|
75
|
-
// Replace directive strings with __markDynamic() calls, processing
|
|
76
|
-
// from end to start to preserve source offsets
|
|
77
|
-
let result = code;
|
|
78
|
-
for (const fn of functions) {
|
|
79
|
-
// Replace the directive in the body content
|
|
80
|
-
const cleanBody = fn.bodyContent.replace(/['"]use dynamic['"];?/, '__markDynamic();');
|
|
81
|
-
// Reconstruct: replace the body content between braces
|
|
82
|
-
result = result.slice(0, fn.bodyStart) + cleanBody + result.slice(fn.bodyEnd);
|
|
83
|
-
}
|
|
84
|
-
|
|
85
|
-
// Add the import at the top
|
|
86
|
-
result = `import { markDynamic as __markDynamic } from '@timber-js/app/runtime';\n` + result;
|
|
87
|
-
|
|
88
|
-
return { code: result, map: null };
|
|
89
|
-
}
|
|
90
|
-
|
|
91
|
-
// ---------------------------------------------------------------------------
|
|
92
|
-
// Static mode validation
|
|
93
|
-
// ---------------------------------------------------------------------------
|
|
94
|
-
|
|
95
|
-
/**
|
|
96
|
-
* In `output: 'static'` mode, `'use dynamic'` is a build error.
|
|
97
|
-
* Static mode renders everything at build time — there is no per-request
|
|
98
|
-
* rendering to opt into.
|
|
99
|
-
*/
|
|
100
|
-
export function validateNoDynamicInStaticMode(
|
|
101
|
-
code: string
|
|
102
|
-
): { message: string; line?: number } | null {
|
|
103
|
-
if (!containsUseDynamic(code)) return null;
|
|
104
|
-
|
|
105
|
-
const functions = findFunctionsWithDirective(code, 'use dynamic');
|
|
106
|
-
if (functions.length === 0) return null;
|
|
107
|
-
|
|
108
|
-
return {
|
|
109
|
-
message:
|
|
110
|
-
`'use dynamic' cannot be used in static mode (output: 'static'). ` +
|
|
111
|
-
`Static mode renders all content at build time — there is no per-request rendering. ` +
|
|
112
|
-
`Remove the directive or switch to output: 'server'.`,
|
|
113
|
-
line: functions[functions.length - 1].directiveLine, // First occurrence (sorted descending)
|
|
114
|
-
};
|
|
115
|
-
}
|
|
116
|
-
|
|
117
|
-
// ---------------------------------------------------------------------------
|
|
118
|
-
// Vite Plugin
|
|
119
|
-
// ---------------------------------------------------------------------------
|
|
120
|
-
|
|
121
|
-
/**
|
|
122
|
-
* Create the timber-dynamic-transform Vite plugin.
|
|
123
|
-
*
|
|
124
|
-
* In server mode: transforms 'use dynamic' into markDynamic() calls.
|
|
125
|
-
* In static mode: rejects 'use dynamic' as a build error.
|
|
126
|
-
*/
|
|
127
|
-
export function timberDynamicTransform(ctx: PluginContext): Plugin {
|
|
128
|
-
const isStatic = ctx.config.output === 'static';
|
|
129
|
-
|
|
130
|
-
return {
|
|
131
|
-
name: 'timber-dynamic-transform',
|
|
132
|
-
|
|
133
|
-
transform(code: string, id: string) {
|
|
134
|
-
// Skip node_modules
|
|
135
|
-
if (id.includes('node_modules')) return null;
|
|
136
|
-
|
|
137
|
-
// Only check files in the app directory
|
|
138
|
-
if (!id.includes('/app/') && !id.startsWith('app/')) return null;
|
|
139
|
-
|
|
140
|
-
// Only check JS/TS files
|
|
141
|
-
if (!/\.[jt]sx?$/.test(id)) return null;
|
|
142
|
-
|
|
143
|
-
// Quick bail-out
|
|
144
|
-
if (!containsUseDynamic(code)) return null;
|
|
145
|
-
|
|
146
|
-
// In static mode, 'use dynamic' is a build error
|
|
147
|
-
if (isStatic) {
|
|
148
|
-
const error = validateNoDynamicInStaticMode(code);
|
|
149
|
-
if (error) {
|
|
150
|
-
this.error(
|
|
151
|
-
`[timber] Static mode error in ${id}${error.line ? `:${error.line}` : ''}: ${error.message}`
|
|
152
|
-
);
|
|
153
|
-
}
|
|
154
|
-
return null;
|
|
155
|
-
}
|
|
156
|
-
|
|
157
|
-
// In server mode, transform the directive
|
|
158
|
-
return transformUseDynamic(code);
|
|
159
|
-
},
|
|
160
|
-
};
|
|
161
|
-
}
|
|
@@ -1,192 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Static analyzability checker for search-params.ts files.
|
|
3
|
-
*
|
|
4
|
-
* Validates that a search-params.ts file's default export is statically
|
|
5
|
-
* analyzable — a createSearchParams() call or a chain of .extend()/.pick()
|
|
6
|
-
* calls on a SearchParamsDefinition.
|
|
7
|
-
*
|
|
8
|
-
* Non-analyzable files produce a hard build error with a diagnostic.
|
|
9
|
-
*
|
|
10
|
-
* Design doc: design/09-typescript.md §"Static Analyzability"
|
|
11
|
-
*/
|
|
12
|
-
|
|
13
|
-
// ---------------------------------------------------------------------------
|
|
14
|
-
// Types
|
|
15
|
-
// ---------------------------------------------------------------------------
|
|
16
|
-
|
|
17
|
-
/** Result of analyzing a search-params.ts file. */
|
|
18
|
-
export interface AnalyzeResult {
|
|
19
|
-
/** Whether the file is statically analyzable. */
|
|
20
|
-
valid: boolean;
|
|
21
|
-
/** Error details when valid is false. */
|
|
22
|
-
error?: AnalyzeError;
|
|
23
|
-
}
|
|
24
|
-
|
|
25
|
-
/** Diagnostic error for non-analyzable search-params.ts. */
|
|
26
|
-
export interface AnalyzeError {
|
|
27
|
-
/** Absolute file path. */
|
|
28
|
-
filePath: string;
|
|
29
|
-
/** Description of the non-analyzable expression. */
|
|
30
|
-
expression: string;
|
|
31
|
-
/** Suggested fix. */
|
|
32
|
-
suggestion: string;
|
|
33
|
-
}
|
|
34
|
-
|
|
35
|
-
// ---------------------------------------------------------------------------
|
|
36
|
-
// AST-free source analysis
|
|
37
|
-
//
|
|
38
|
-
// We use a lightweight regex-based approach to validate the structure of the
|
|
39
|
-
// default export. This avoids requiring a TypeScript compiler instance at
|
|
40
|
-
// build time for the initial validation pass. The full type extraction
|
|
41
|
-
// (reading T from SearchParamsDefinition<T>) still happens via the TypeScript
|
|
42
|
-
// compiler in the codegen step — this module just validates the *shape*.
|
|
43
|
-
// ---------------------------------------------------------------------------
|
|
44
|
-
|
|
45
|
-
/**
|
|
46
|
-
* Patterns that indicate a valid default export:
|
|
47
|
-
*
|
|
48
|
-
* 1. `export default createSearchParams(...)`
|
|
49
|
-
* 2. `export default someVar.extend(...)`
|
|
50
|
-
* 3. `export default someVar.pick(...)`
|
|
51
|
-
* 4. `export default someVar.extend(...).extend(...)` (chained)
|
|
52
|
-
* 5. `export default someVar.extend(...).pick(...)` (chained)
|
|
53
|
-
* 6. `export default createSearchParams(...).extend(...)`
|
|
54
|
-
*
|
|
55
|
-
* Invalid patterns:
|
|
56
|
-
* - `export default someFunction(...)` (arbitrary factory)
|
|
57
|
-
* - `export default condition ? a : b` (runtime conditional)
|
|
58
|
-
* - `export default variable` (opaque reference without call)
|
|
59
|
-
*/
|
|
60
|
-
|
|
61
|
-
/**
|
|
62
|
-
* Analyze a search-params.ts file source for static analyzability.
|
|
63
|
-
*
|
|
64
|
-
* @param source - The file content as a string
|
|
65
|
-
* @param filePath - Absolute path to the file (for diagnostics)
|
|
66
|
-
*/
|
|
67
|
-
export function analyzeSearchParams(source: string, filePath: string): AnalyzeResult {
|
|
68
|
-
// Strip comments to avoid false matches
|
|
69
|
-
const stripped = stripComments(source);
|
|
70
|
-
|
|
71
|
-
// Find the default export
|
|
72
|
-
const defaultExport = extractDefaultExport(stripped);
|
|
73
|
-
|
|
74
|
-
if (!defaultExport) {
|
|
75
|
-
return {
|
|
76
|
-
valid: false,
|
|
77
|
-
error: {
|
|
78
|
-
filePath,
|
|
79
|
-
expression: '(no default export found)',
|
|
80
|
-
suggestion:
|
|
81
|
-
'search-params.ts must have a default export. Use: export default createSearchParams({ ... })',
|
|
82
|
-
},
|
|
83
|
-
};
|
|
84
|
-
}
|
|
85
|
-
|
|
86
|
-
// Validate the expression
|
|
87
|
-
if (isValidExpression(defaultExport.trim())) {
|
|
88
|
-
return { valid: true };
|
|
89
|
-
}
|
|
90
|
-
|
|
91
|
-
return {
|
|
92
|
-
valid: false,
|
|
93
|
-
error: {
|
|
94
|
-
filePath,
|
|
95
|
-
expression: defaultExport.trim(),
|
|
96
|
-
suggestion:
|
|
97
|
-
'The default export must be a createSearchParams() call, or a chain of ' +
|
|
98
|
-
'.extend() / .pick() calls on a SearchParamsDefinition. Arbitrary factory ' +
|
|
99
|
-
'functions and runtime conditionals are not supported.',
|
|
100
|
-
},
|
|
101
|
-
};
|
|
102
|
-
}
|
|
103
|
-
|
|
104
|
-
// ---------------------------------------------------------------------------
|
|
105
|
-
// Internal helpers
|
|
106
|
-
// ---------------------------------------------------------------------------
|
|
107
|
-
|
|
108
|
-
/** Strip single-line and multi-line comments from source. */
|
|
109
|
-
function stripComments(source: string): string {
|
|
110
|
-
// Remove multi-line comments
|
|
111
|
-
let result = source.replace(/\/\*[\s\S]*?\*\//g, '');
|
|
112
|
-
// Remove single-line comments
|
|
113
|
-
result = result.replace(/\/\/.*$/gm, '');
|
|
114
|
-
return result;
|
|
115
|
-
}
|
|
116
|
-
|
|
117
|
-
/**
|
|
118
|
-
* Extract the expression from `export default <expr>`.
|
|
119
|
-
*
|
|
120
|
-
* Handles both:
|
|
121
|
-
* export default createSearchParams(...)
|
|
122
|
-
* export default expr\n (terminated by newline or semicolon before next statement)
|
|
123
|
-
*/
|
|
124
|
-
function extractDefaultExport(source: string): string | undefined {
|
|
125
|
-
// Match `export default` followed by the expression
|
|
126
|
-
const match = source.match(
|
|
127
|
-
/export\s+default\s+([\s\S]+?)(?:;|\n(?=export|import|const|let|var|function|class|type|interface|declare))/
|
|
128
|
-
);
|
|
129
|
-
if (match) {
|
|
130
|
-
return match[1];
|
|
131
|
-
}
|
|
132
|
-
|
|
133
|
-
// Fallback: match everything after `export default` to end of file
|
|
134
|
-
const fallback = source.match(/export\s+default\s+([\s\S]+)$/);
|
|
135
|
-
if (fallback) {
|
|
136
|
-
return fallback[1].replace(/;\s*$/, '');
|
|
137
|
-
}
|
|
138
|
-
|
|
139
|
-
return undefined;
|
|
140
|
-
}
|
|
141
|
-
|
|
142
|
-
/**
|
|
143
|
-
* Check if an expression is a valid statically-analyzable pattern.
|
|
144
|
-
*
|
|
145
|
-
* Valid patterns:
|
|
146
|
-
* - Starts with `createSearchParams(`
|
|
147
|
-
* - Contains `.extend(` or `.pick(` chains (possibly starting with createSearchParams or a variable)
|
|
148
|
-
* - A variable identifier followed by chaining
|
|
149
|
-
*/
|
|
150
|
-
function isValidExpression(expr: string): boolean {
|
|
151
|
-
// Normalize whitespace
|
|
152
|
-
const normalized = expr.replace(/\s+/g, ' ').trim();
|
|
153
|
-
|
|
154
|
-
// Pattern 1: starts with createSearchParams(
|
|
155
|
-
if (normalized.startsWith('createSearchParams(')) {
|
|
156
|
-
return true;
|
|
157
|
-
}
|
|
158
|
-
|
|
159
|
-
// Pattern 2: chain ending with .extend(...) or .pick(...)
|
|
160
|
-
// This covers: someVar.extend(...), createSearchParams(...).extend(...).pick(...), etc.
|
|
161
|
-
if (/\.(extend|pick)\s*\(/.test(normalized)) {
|
|
162
|
-
// Reject ternaries and other conditional patterns
|
|
163
|
-
if (/\?/.test(normalized) && /:/.test(normalized)) {
|
|
164
|
-
return false;
|
|
165
|
-
}
|
|
166
|
-
// Reject function declarations/expressions
|
|
167
|
-
if (/^\s*(function|=>|\()/.test(normalized)) {
|
|
168
|
-
return false;
|
|
169
|
-
}
|
|
170
|
-
return true;
|
|
171
|
-
}
|
|
172
|
-
|
|
173
|
-
return false;
|
|
174
|
-
}
|
|
175
|
-
|
|
176
|
-
/**
|
|
177
|
-
* Format an AnalyzeError into a human-readable build error message.
|
|
178
|
-
*/
|
|
179
|
-
export function formatAnalyzeError(error: AnalyzeError): string {
|
|
180
|
-
return [
|
|
181
|
-
`[timber] Non-analyzable search-params.ts`,
|
|
182
|
-
``,
|
|
183
|
-
` File: ${error.filePath}`,
|
|
184
|
-
` Expression: ${error.expression}`,
|
|
185
|
-
``,
|
|
186
|
-
` ${error.suggestion}`,
|
|
187
|
-
``,
|
|
188
|
-
` The framework must be able to statically extract the type from your`,
|
|
189
|
-
` search-params.ts at build time. Dynamic values, conditionals, and`,
|
|
190
|
-
` arbitrary factory functions prevent this analysis.`,
|
|
191
|
-
].join('\n');
|
|
192
|
-
}
|