@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
|
@@ -0,0 +1,177 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Cloudflare dev-mode bindings via wrangler's getPlatformProxy().
|
|
3
|
+
*
|
|
4
|
+
* This Vite plugin starts a local workerd process (via Miniflare) that
|
|
5
|
+
* emulates Cloudflare bindings (KV, D1, R2, Queues, Durable Objects, etc.)
|
|
6
|
+
* during development. The proxy env is injected into the request ALS so
|
|
7
|
+
* `getCloudflareBindings()` works identically in dev and production.
|
|
8
|
+
*
|
|
9
|
+
* Design doc: design/35-cloudflare-primitives.md §"Dev Experience"
|
|
10
|
+
*
|
|
11
|
+
* Usage:
|
|
12
|
+
* ```ts
|
|
13
|
+
* // vite.config.ts
|
|
14
|
+
* import { timber } from '@timber-js/app'
|
|
15
|
+
* import { cloudflare } from '@timber-js/app/adapters/cloudflare'
|
|
16
|
+
* import { cloudflareDevBindings } from '@timber-js/app/adapters/cloudflare/dev'
|
|
17
|
+
*
|
|
18
|
+
* export default defineConfig({
|
|
19
|
+
* plugins: [
|
|
20
|
+
* timber({ adapter: cloudflare() }),
|
|
21
|
+
* cloudflareDevBindings(),
|
|
22
|
+
* ],
|
|
23
|
+
* })
|
|
24
|
+
* ```
|
|
25
|
+
*
|
|
26
|
+
* Requires `wrangler` as a dev dependency. Install it:
|
|
27
|
+
* ```bash
|
|
28
|
+
* pnpm add -D wrangler
|
|
29
|
+
* ```
|
|
30
|
+
*/
|
|
31
|
+
|
|
32
|
+
import type { Plugin, ViteDevServer } from 'vite';
|
|
33
|
+
import { runWithBindings } from './cloudflare.js';
|
|
34
|
+
|
|
35
|
+
/**
|
|
36
|
+
* Dynamically import wrangler. Extracted as a named export so tests
|
|
37
|
+
* can mock it without needing wrangler installed.
|
|
38
|
+
* @internal
|
|
39
|
+
*/
|
|
40
|
+
export async function loadWrangler(): Promise<{ getPlatformProxy: (opts: any) => Promise<any> }> {
|
|
41
|
+
try {
|
|
42
|
+
return await import('wrangler');
|
|
43
|
+
} catch {
|
|
44
|
+
throw new Error(
|
|
45
|
+
'[timber] Could not import wrangler. ' +
|
|
46
|
+
'cloudflareDevBindings() requires wrangler as a dev dependency.\n' +
|
|
47
|
+
'Install it: pnpm add -D wrangler'
|
|
48
|
+
);
|
|
49
|
+
}
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
// ─── Contract ─────────────────────────────────────────────────────────────
|
|
53
|
+
//
|
|
54
|
+
// The dev request wrapper is stored on the ViteDevServer instance using a
|
|
55
|
+
// well-known Symbol. This allows cross-plugin communication without import
|
|
56
|
+
// dependencies — the dev-server plugin checks for this Symbol per-request
|
|
57
|
+
// and wraps the handler call when present.
|
|
58
|
+
//
|
|
59
|
+
// The wrapper type is: <T>(fn: () => T) => T
|
|
60
|
+
// It runs `fn` inside the Cloudflare bindings ALS so getCloudflareBindings()
|
|
61
|
+
// works throughout the request lifecycle.
|
|
62
|
+
|
|
63
|
+
/**
|
|
64
|
+
* Symbol key used to store the dev request wrapper on ViteDevServer.
|
|
65
|
+
*
|
|
66
|
+
* @internal Exported for testing. The dev-server plugin uses
|
|
67
|
+
* `Symbol.for('timber:dev-request-wrapper')` directly to avoid
|
|
68
|
+
* importing from this module.
|
|
69
|
+
*/
|
|
70
|
+
export const DEV_REQUEST_WRAPPER_KEY = Symbol.for('timber:dev-request-wrapper');
|
|
71
|
+
|
|
72
|
+
/** Options for the Cloudflare dev bindings plugin. */
|
|
73
|
+
export interface CloudflareDevBindingsOptions {
|
|
74
|
+
/**
|
|
75
|
+
* Path to the wrangler configuration file.
|
|
76
|
+
* @default Auto-detected by wrangler (wrangler.jsonc, wrangler.json, wrangler.toml)
|
|
77
|
+
*/
|
|
78
|
+
configPath?: string;
|
|
79
|
+
|
|
80
|
+
/**
|
|
81
|
+
* Whether to persist binding data (KV, D1, R2) between dev server restarts.
|
|
82
|
+
*
|
|
83
|
+
* - `true` — persist in wrangler's default location (`.wrangler/state/`)
|
|
84
|
+
* - `{ path: string }` — persist in a custom directory
|
|
85
|
+
* - `false` — start fresh on every restart
|
|
86
|
+
*
|
|
87
|
+
* @default true
|
|
88
|
+
*/
|
|
89
|
+
persist?: boolean | { path: string };
|
|
90
|
+
|
|
91
|
+
/**
|
|
92
|
+
* Wrangler environment to use (for multi-environment configs).
|
|
93
|
+
* Maps to `[env.<name>]` sections in wrangler.toml / wrangler.jsonc.
|
|
94
|
+
*/
|
|
95
|
+
environment?: string;
|
|
96
|
+
|
|
97
|
+
/**
|
|
98
|
+
* @internal Override the wrangler loader for testing.
|
|
99
|
+
* Not part of the public API — may change without notice.
|
|
100
|
+
*/
|
|
101
|
+
_loadWrangler?: () => Promise<{ getPlatformProxy: (opts: any) => Promise<any> }>;
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
/**
|
|
105
|
+
* Vite plugin that provides Cloudflare bindings in dev mode via
|
|
106
|
+
* wrangler's `getPlatformProxy()`.
|
|
107
|
+
*
|
|
108
|
+
* Starts a local workerd process on dev server startup and injects
|
|
109
|
+
* the proxy env into the per-request ALS. This makes
|
|
110
|
+
* `getCloudflareBindings()` return real (emulated) bindings during
|
|
111
|
+
* development — KV, D1, R2, Queues, Durable Objects, and all other
|
|
112
|
+
* bindings declared in wrangler.jsonc.
|
|
113
|
+
*
|
|
114
|
+
* The proxy is automatically disposed when the dev server closes.
|
|
115
|
+
*
|
|
116
|
+
* **Requirements:**
|
|
117
|
+
* - `wrangler` installed as a dev dependency
|
|
118
|
+
* - A `wrangler.jsonc` (or `wrangler.toml`) with binding declarations
|
|
119
|
+
*
|
|
120
|
+
* @example
|
|
121
|
+
* ```ts
|
|
122
|
+
* import { cloudflareDevBindings } from '@timber-js/app/adapters/cloudflare/dev'
|
|
123
|
+
*
|
|
124
|
+
* export default defineConfig({
|
|
125
|
+
* plugins: [
|
|
126
|
+
* timber({ adapter: cloudflare() }),
|
|
127
|
+
* cloudflareDevBindings(),
|
|
128
|
+
* ],
|
|
129
|
+
* })
|
|
130
|
+
* ```
|
|
131
|
+
*/
|
|
132
|
+
export function cloudflareDevBindings(options: CloudflareDevBindingsOptions = {}): Plugin {
|
|
133
|
+
return {
|
|
134
|
+
name: 'timber-cloudflare-dev-bindings',
|
|
135
|
+
|
|
136
|
+
// Dev mode only — never active during production builds.
|
|
137
|
+
// See design/35-cloudflare-primitives.md §"Security Considerations":
|
|
138
|
+
// "Dev bindings are dev-only. apply: 'serve' ensures it never runs
|
|
139
|
+
// in production builds."
|
|
140
|
+
apply: 'serve',
|
|
141
|
+
|
|
142
|
+
async configureServer(server: ViteDevServer) {
|
|
143
|
+
// Dynamic import — wrangler is an optional peer dependency.
|
|
144
|
+
// Users must install it themselves: `pnpm add -D wrangler`
|
|
145
|
+
const { getPlatformProxy } = await (options._loadWrangler ?? loadWrangler)();
|
|
146
|
+
|
|
147
|
+
// Build getPlatformProxy options from our public API.
|
|
148
|
+
const proxyOptions: Record<string, unknown> = {
|
|
149
|
+
persist: options.persist ?? true,
|
|
150
|
+
};
|
|
151
|
+
if (options.configPath) {
|
|
152
|
+
proxyOptions.configPath = options.configPath;
|
|
153
|
+
}
|
|
154
|
+
if (options.environment) {
|
|
155
|
+
proxyOptions.environment = options.environment;
|
|
156
|
+
}
|
|
157
|
+
|
|
158
|
+
const proxy = await getPlatformProxy(proxyOptions);
|
|
159
|
+
|
|
160
|
+
// Store a request wrapper on the server instance. The dev-server
|
|
161
|
+
// plugin checks for this Symbol per-request and wraps the RSC
|
|
162
|
+
// handler call so getCloudflareBindings() works throughout the
|
|
163
|
+
// request lifecycle (server components, middleware, server actions).
|
|
164
|
+
(server as any)[DEV_REQUEST_WRAPPER_KEY] = <T>(fn: () => T): T => {
|
|
165
|
+
return runWithBindings(proxy.env as Record<string, unknown>, fn);
|
|
166
|
+
};
|
|
167
|
+
|
|
168
|
+
// Dispose the proxy when the dev server closes. This shuts down
|
|
169
|
+
// the local workerd process started by getPlatformProxy().
|
|
170
|
+
server.httpServer?.on('close', () => {
|
|
171
|
+
proxy.dispose().catch(() => {
|
|
172
|
+
// Disposal errors are non-fatal — the process is shutting down.
|
|
173
|
+
});
|
|
174
|
+
});
|
|
175
|
+
},
|
|
176
|
+
};
|
|
177
|
+
}
|
|
@@ -31,8 +31,27 @@ function generateHeadersFile(): string {
|
|
|
31
31
|
// can access KV, D1, DO, R2, Queues, etc. via getCloudflareBindings().
|
|
32
32
|
// No global fallback — if called outside a request, it throws.
|
|
33
33
|
// See design/11-platform.md §"Platform Target" and design/25-production-deployments.md.
|
|
34
|
-
|
|
35
|
-
|
|
34
|
+
//
|
|
35
|
+
// The ALS is stored on globalThis via Symbol.for so it survives Vite's
|
|
36
|
+
// module instance split in dev mode. The Vite host process (where
|
|
37
|
+
// cloudflareDevBindings → runWithBindings runs) and the RSC module runner
|
|
38
|
+
// (where getCloudflareBindings runs) load separate instances of this file,
|
|
39
|
+
// each with their own module-level variables. globalThis + Symbol.for
|
|
40
|
+
// ensures both share the same ALS — same pattern React uses for
|
|
41
|
+
// Symbol.for('react.element').
|
|
42
|
+
|
|
43
|
+
const BINDINGS_ALS_KEY = Symbol.for('timber:cf-bindings-als');
|
|
44
|
+
|
|
45
|
+
function getBindingsAls(): AsyncLocalStorage<Record<string, unknown>> {
|
|
46
|
+
let als = (globalThis as any)[BINDINGS_ALS_KEY] as
|
|
47
|
+
| AsyncLocalStorage<Record<string, unknown>>
|
|
48
|
+
| undefined;
|
|
49
|
+
if (!als) {
|
|
50
|
+
als = new AsyncLocalStorage<Record<string, unknown>>();
|
|
51
|
+
(globalThis as any)[BINDINGS_ALS_KEY] = als;
|
|
52
|
+
}
|
|
53
|
+
return als;
|
|
54
|
+
}
|
|
36
55
|
|
|
37
56
|
/**
|
|
38
57
|
* Get Cloudflare Worker bindings for the current request.
|
|
@@ -55,10 +74,8 @@ const bindingsAls = new AsyncLocalStorage<Record<string, unknown>>();
|
|
|
55
74
|
* }
|
|
56
75
|
* ```
|
|
57
76
|
*/
|
|
58
|
-
export function getCloudflareBindings<
|
|
59
|
-
|
|
60
|
-
>(): T {
|
|
61
|
-
const env = bindingsAls.getStore();
|
|
77
|
+
export function getCloudflareBindings<T = Record<string, unknown>>(): T {
|
|
78
|
+
const env = getBindingsAls().getStore();
|
|
62
79
|
if (!env) {
|
|
63
80
|
throw new Error(
|
|
64
81
|
'getCloudflareBindings() called outside a Cloudflare Workers request context. ' +
|
|
@@ -74,7 +91,85 @@ export function getCloudflareBindings<
|
|
|
74
91
|
* @internal Used by wrapWithExecutionContext.
|
|
75
92
|
*/
|
|
76
93
|
export function runWithBindings<T>(env: Record<string, unknown>, fn: () => T): T {
|
|
77
|
-
return
|
|
94
|
+
return getBindingsAls().run(env, fn);
|
|
95
|
+
}
|
|
96
|
+
|
|
97
|
+
// ─── Binding configuration types ──────────────────────────────────────────────
|
|
98
|
+
// Declarative binding config that maps to wrangler.jsonc binding sections.
|
|
99
|
+
// See design/35-cloudflare-primitives.md §"Binding Declarations in Adapter Config".
|
|
100
|
+
|
|
101
|
+
/** KV namespace binding. `id` defaults to `''` (filled in by wrangler or dashboard). */
|
|
102
|
+
export interface CloudflareKVBinding {
|
|
103
|
+
name: string;
|
|
104
|
+
id?: string;
|
|
105
|
+
}
|
|
106
|
+
|
|
107
|
+
/** D1 database binding. `database_id` is required (from Cloudflare dashboard). */
|
|
108
|
+
export interface CloudflareD1Binding {
|
|
109
|
+
name: string;
|
|
110
|
+
database_id: string;
|
|
111
|
+
}
|
|
112
|
+
|
|
113
|
+
/** R2 bucket binding. `bucket_name` must match the bucket created in Cloudflare. */
|
|
114
|
+
export interface CloudflareR2Binding {
|
|
115
|
+
name: string;
|
|
116
|
+
bucket_name: string;
|
|
117
|
+
}
|
|
118
|
+
|
|
119
|
+
/** Queue producer binding. `queue` is the queue name in Cloudflare. */
|
|
120
|
+
export interface CloudflareQueueProducer {
|
|
121
|
+
name: string;
|
|
122
|
+
queue: string;
|
|
123
|
+
}
|
|
124
|
+
|
|
125
|
+
/** Queue consumer declaration. Attached to the worker, not a binding name. */
|
|
126
|
+
export interface CloudflareQueueConsumer {
|
|
127
|
+
queue: string;
|
|
128
|
+
max_batch_size?: number;
|
|
129
|
+
max_retries?: number;
|
|
130
|
+
dead_letter_queue?: string;
|
|
131
|
+
}
|
|
132
|
+
|
|
133
|
+
/** Durable Object binding. `script_name` is for external DO references. */
|
|
134
|
+
export interface CloudflareDurableObjectBinding {
|
|
135
|
+
name: string;
|
|
136
|
+
class_name: string;
|
|
137
|
+
script_name?: string;
|
|
138
|
+
}
|
|
139
|
+
|
|
140
|
+
/**
|
|
141
|
+
* Declarative Cloudflare bindings configuration.
|
|
142
|
+
*
|
|
143
|
+
* These are converted to the appropriate wrangler.jsonc sections
|
|
144
|
+
* (`kv_namespaces`, `d1_databases`, `r2_buckets`, `queues`, `durable_objects`).
|
|
145
|
+
* The `wrangler` escape hatch overrides any generated binding sections
|
|
146
|
+
* if there's a conflict.
|
|
147
|
+
*
|
|
148
|
+
* @example
|
|
149
|
+
* ```ts
|
|
150
|
+
* cloudflare({
|
|
151
|
+
* bindings: {
|
|
152
|
+
* kv: [{ name: 'TIMBER_CACHE' }],
|
|
153
|
+
* d1: [{ name: 'MY_DB', database_id: 'xxxx' }],
|
|
154
|
+
* r2: [{ name: 'MY_BUCKET', bucket_name: 'my-bucket' }],
|
|
155
|
+
* queues: {
|
|
156
|
+
* producers: [{ name: 'EMAIL_QUEUE', queue: 'email-queue' }],
|
|
157
|
+
* consumers: [{ queue: 'email-queue', max_batch_size: 10 }],
|
|
158
|
+
* },
|
|
159
|
+
* durableObjects: [{ name: 'MY_DO', class_name: 'MyDurableObject' }],
|
|
160
|
+
* },
|
|
161
|
+
* })
|
|
162
|
+
* ```
|
|
163
|
+
*/
|
|
164
|
+
export interface CloudflareBindings {
|
|
165
|
+
kv?: CloudflareKVBinding[];
|
|
166
|
+
d1?: CloudflareD1Binding[];
|
|
167
|
+
r2?: CloudflareR2Binding[];
|
|
168
|
+
queues?: {
|
|
169
|
+
producers?: CloudflareQueueProducer[];
|
|
170
|
+
consumers?: CloudflareQueueConsumer[];
|
|
171
|
+
};
|
|
172
|
+
durableObjects?: CloudflareDurableObjectBinding[];
|
|
78
173
|
}
|
|
79
174
|
|
|
80
175
|
/** Options for the Cloudflare Workers adapter. */
|
|
@@ -91,11 +186,42 @@ export interface CloudflareAdapterOptions {
|
|
|
91
186
|
*/
|
|
92
187
|
compatibilityFlags?: string[];
|
|
93
188
|
|
|
189
|
+
/**
|
|
190
|
+
* Declarative Cloudflare bindings. Generates the appropriate
|
|
191
|
+
* `kv_namespaces`, `d1_databases`, `r2_buckets`, `queues`, and
|
|
192
|
+
* `durable_objects` sections in wrangler.jsonc.
|
|
193
|
+
*
|
|
194
|
+
* If both `bindings` and `wrangler` specify the same section,
|
|
195
|
+
* `wrangler` wins (it's the escape hatch).
|
|
196
|
+
*/
|
|
197
|
+
bindings?: CloudflareBindings;
|
|
198
|
+
|
|
94
199
|
/**
|
|
95
200
|
* Custom wrangler.jsonc fields to merge.
|
|
96
|
-
* Overrides generated values.
|
|
201
|
+
* Overrides generated values (including bindings-generated sections).
|
|
97
202
|
*/
|
|
98
203
|
wrangler?: Record<string, unknown>;
|
|
204
|
+
|
|
205
|
+
/**
|
|
206
|
+
* Path to a module that exports additional Worker handlers (queue, scheduled, email, etc.).
|
|
207
|
+
* The module is imported in the generated `_worker.js` and its named exports are
|
|
208
|
+
* spread into the default export alongside `fetch`.
|
|
209
|
+
*
|
|
210
|
+
* The module receives `(batch/event, env, ctx)` — standard Cloudflare handler signatures.
|
|
211
|
+
* It does NOT have access to timber's request pipeline (no ALS, no getCloudflareBindings).
|
|
212
|
+
* Use `env` directly for bindings.
|
|
213
|
+
*
|
|
214
|
+
* @example
|
|
215
|
+
* ```ts
|
|
216
|
+
* // timber.config.ts
|
|
217
|
+
* adapter: cloudflare({ workerHandlers: './src/worker-handlers.ts' })
|
|
218
|
+
*
|
|
219
|
+
* // src/worker-handlers.ts
|
|
220
|
+
* export async function queue(batch, env) { ... }
|
|
221
|
+
* export async function scheduled(controller, env, ctx) { ... }
|
|
222
|
+
* ```
|
|
223
|
+
*/
|
|
224
|
+
workerHandlers?: string;
|
|
99
225
|
}
|
|
100
226
|
|
|
101
227
|
/**
|
|
@@ -151,9 +277,35 @@ export function cloudflare(options: CloudflareAdapterOptions = {}): TimberPlatfo
|
|
|
151
277
|
await writeFile(join(outDir, '_timber-manifest-init.js'), config.manifestInit);
|
|
152
278
|
}
|
|
153
279
|
|
|
280
|
+
// Compile optional worker handlers (queue, scheduled, etc.)
|
|
281
|
+
// Uses Vite's build API to bundle the TypeScript source into ESM.
|
|
282
|
+
let hasWorkerHandlers = false;
|
|
283
|
+
if (options.workerHandlers) {
|
|
284
|
+
const handlersEntry = join(process.cwd(), options.workerHandlers);
|
|
285
|
+
const { build: viteBuild } = await import('vite');
|
|
286
|
+
await viteBuild({
|
|
287
|
+
configFile: false,
|
|
288
|
+
logLevel: 'info',
|
|
289
|
+
build: {
|
|
290
|
+
lib: {
|
|
291
|
+
entry: handlersEntry,
|
|
292
|
+
formats: ['es'],
|
|
293
|
+
fileName: () => '_worker-handlers.js',
|
|
294
|
+
},
|
|
295
|
+
outDir,
|
|
296
|
+
emptyOutDir: false,
|
|
297
|
+
minify: false,
|
|
298
|
+
rollupOptions: {
|
|
299
|
+
external: [/^node:/],
|
|
300
|
+
},
|
|
301
|
+
},
|
|
302
|
+
});
|
|
303
|
+
hasWorkerHandlers = true;
|
|
304
|
+
}
|
|
305
|
+
|
|
154
306
|
// Generate the Workers entry point
|
|
155
307
|
const hasManifestInit = !!config.manifestInit;
|
|
156
|
-
const workerEntry = generateWorkerEntry(outDir, outDir, hasManifestInit);
|
|
308
|
+
const workerEntry = generateWorkerEntry(outDir, outDir, hasManifestInit, hasWorkerHandlers);
|
|
157
309
|
await writeFile(join(outDir, '_worker.js'), workerEntry);
|
|
158
310
|
|
|
159
311
|
// Generate wrangler.jsonc
|
|
@@ -184,12 +336,12 @@ export function cloudflare(options: CloudflareAdapterOptions = {}): TimberPlatfo
|
|
|
184
336
|
export function wrapWithExecutionContext(
|
|
185
337
|
adapter: TimberPlatformAdapter,
|
|
186
338
|
handler: (req: Request) => Promise<Response>
|
|
187
|
-
):
|
|
339
|
+
): CfExportedHandler<Record<string, unknown>> {
|
|
188
340
|
return {
|
|
189
341
|
async fetch(
|
|
190
342
|
request: Request,
|
|
191
343
|
env: Record<string, unknown>,
|
|
192
|
-
ctx:
|
|
344
|
+
ctx: CfExecutionContext
|
|
193
345
|
): Promise<Response> {
|
|
194
346
|
// Bind the adapter's waitUntil to the Workers execution context
|
|
195
347
|
const originalWaitUntil = adapter.waitUntil;
|
|
@@ -214,7 +366,8 @@ export function wrapWithExecutionContext(
|
|
|
214
366
|
export function generateWorkerEntry(
|
|
215
367
|
buildDir: string,
|
|
216
368
|
outDir: string,
|
|
217
|
-
hasManifestInit = false
|
|
369
|
+
hasManifestInit = false,
|
|
370
|
+
hasWorkerHandlers = false
|
|
218
371
|
): string {
|
|
219
372
|
// The RSC entry is the main request handler — it exports the fetch handler as default.
|
|
220
373
|
// The Vite RSC plugin outputs it to rsc/index.js.
|
|
@@ -229,20 +382,171 @@ export function generateWorkerEntry(
|
|
|
229
382
|
// ESM guarantees imports are evaluated in order.
|
|
230
383
|
const manifestImport = hasManifestInit ? "import './_timber-manifest-init.js'\n" : '';
|
|
231
384
|
|
|
385
|
+
// Optional additional Worker handlers (queue, scheduled, email, etc.)
|
|
386
|
+
// Compiled by buildOutput via Vite's build API into _worker-handlers.js.
|
|
387
|
+
// The module's named exports are spread into the default export alongside fetch.
|
|
388
|
+
const handlersImport = hasWorkerHandlers
|
|
389
|
+
? "import * as workerHandlers from './_worker-handlers.js'\n"
|
|
390
|
+
: '';
|
|
391
|
+
const handlersSpread = hasWorkerHandlers ? ' ...workerHandlers,\n' : '';
|
|
392
|
+
|
|
232
393
|
return `// Generated by @timber-js/app/adapters/cloudflare
|
|
233
394
|
// Do not edit — this file is regenerated on each build.
|
|
234
395
|
|
|
396
|
+
import { AsyncLocalStorage } from 'node:async_hooks'
|
|
235
397
|
${manifestImport}import handler from '${rscEntryRelative}'
|
|
236
|
-
|
|
398
|
+
${handlersImport}
|
|
237
399
|
// Set TIMBER_RUNTIME for instrumentation.ts conditional SDK initialization.
|
|
238
400
|
// See design/25-production-deployments.md §"TIMBER_RUNTIME".
|
|
239
401
|
globalThis.process ??= { env: {} }
|
|
240
402
|
process.env.TIMBER_RUNTIME = 'cloudflare'
|
|
241
403
|
|
|
242
|
-
|
|
404
|
+
// Bind Cloudflare env to ALS so getCloudflareBindings() works at runtime.
|
|
405
|
+
// Uses the same Symbol.for key as getCloudflareBindings() reads from.
|
|
406
|
+
const ALS_KEY = Symbol.for('timber:cf-bindings-als')
|
|
407
|
+
let bindingsAls = globalThis[ALS_KEY]
|
|
408
|
+
if (!bindingsAls) {
|
|
409
|
+
bindingsAls = new AsyncLocalStorage()
|
|
410
|
+
globalThis[ALS_KEY] = bindingsAls
|
|
411
|
+
}
|
|
412
|
+
|
|
413
|
+
export default {
|
|
414
|
+
${handlersSpread} async fetch(request, env, ctx) {
|
|
415
|
+
return bindingsAls.run(env, () => handler(request))
|
|
416
|
+
}
|
|
417
|
+
}
|
|
243
418
|
`;
|
|
244
419
|
}
|
|
245
420
|
|
|
421
|
+
/** Wrangler binding sections generated from `CloudflareBindings`. */
|
|
422
|
+
export interface WranglerBindingsConfig {
|
|
423
|
+
kv_namespaces?: Array<{ binding: string; id: string }>;
|
|
424
|
+
d1_databases?: Array<{ binding: string; database_id: string }>;
|
|
425
|
+
r2_buckets?: Array<{ binding: string; bucket_name: string }>;
|
|
426
|
+
queues?: {
|
|
427
|
+
producers?: Array<{ binding: string; queue: string }>;
|
|
428
|
+
consumers?: Array<Record<string, unknown>>;
|
|
429
|
+
};
|
|
430
|
+
durable_objects?: {
|
|
431
|
+
bindings: Array<Record<string, string>>;
|
|
432
|
+
};
|
|
433
|
+
}
|
|
434
|
+
|
|
435
|
+
/**
|
|
436
|
+
* Convert declarative `CloudflareBindings` into wrangler.jsonc binding sections.
|
|
437
|
+
*
|
|
438
|
+
* Maps:
|
|
439
|
+
* - `kv` → `kv_namespaces`
|
|
440
|
+
* - `d1` → `d1_databases`
|
|
441
|
+
* - `r2` → `r2_buckets`
|
|
442
|
+
* - `queues` → `queues` (producers + consumers)
|
|
443
|
+
* - `durableObjects` → `durable_objects`
|
|
444
|
+
*
|
|
445
|
+
* Empty arrays are omitted from the output.
|
|
446
|
+
*
|
|
447
|
+
* @internal Exported for testing.
|
|
448
|
+
*/
|
|
449
|
+
export function generateBindingsConfig(
|
|
450
|
+
bindings: CloudflareBindings | undefined
|
|
451
|
+
): WranglerBindingsConfig {
|
|
452
|
+
if (!bindings) return {};
|
|
453
|
+
|
|
454
|
+
const result: WranglerBindingsConfig = {};
|
|
455
|
+
|
|
456
|
+
// KV namespaces
|
|
457
|
+
if (bindings.kv && bindings.kv.length > 0) {
|
|
458
|
+
result.kv_namespaces = bindings.kv.map((kv) => ({
|
|
459
|
+
binding: kv.name,
|
|
460
|
+
id: kv.id ?? '',
|
|
461
|
+
}));
|
|
462
|
+
}
|
|
463
|
+
|
|
464
|
+
// D1 databases
|
|
465
|
+
if (bindings.d1 && bindings.d1.length > 0) {
|
|
466
|
+
result.d1_databases = bindings.d1.map((d1) => ({
|
|
467
|
+
binding: d1.name,
|
|
468
|
+
database_id: d1.database_id,
|
|
469
|
+
}));
|
|
470
|
+
}
|
|
471
|
+
|
|
472
|
+
// R2 buckets
|
|
473
|
+
if (bindings.r2 && bindings.r2.length > 0) {
|
|
474
|
+
result.r2_buckets = bindings.r2.map((r2) => ({
|
|
475
|
+
binding: r2.name,
|
|
476
|
+
bucket_name: r2.bucket_name,
|
|
477
|
+
}));
|
|
478
|
+
}
|
|
479
|
+
|
|
480
|
+
// Queues (producers and/or consumers)
|
|
481
|
+
if (bindings.queues) {
|
|
482
|
+
const queues: Record<string, unknown> = {};
|
|
483
|
+
if (bindings.queues.producers && bindings.queues.producers.length > 0) {
|
|
484
|
+
queues.producers = bindings.queues.producers.map((p) => ({
|
|
485
|
+
binding: p.name,
|
|
486
|
+
queue: p.queue,
|
|
487
|
+
}));
|
|
488
|
+
}
|
|
489
|
+
if (bindings.queues.consumers && bindings.queues.consumers.length > 0) {
|
|
490
|
+
queues.consumers = bindings.queues.consumers.map((c) => {
|
|
491
|
+
const consumer: Record<string, unknown> = { queue: c.queue };
|
|
492
|
+
if (c.max_batch_size !== undefined) consumer.max_batch_size = c.max_batch_size;
|
|
493
|
+
if (c.max_retries !== undefined) consumer.max_retries = c.max_retries;
|
|
494
|
+
if (c.dead_letter_queue !== undefined) consumer.dead_letter_queue = c.dead_letter_queue;
|
|
495
|
+
return consumer;
|
|
496
|
+
});
|
|
497
|
+
}
|
|
498
|
+
if (Object.keys(queues).length > 0) {
|
|
499
|
+
result.queues = queues;
|
|
500
|
+
}
|
|
501
|
+
}
|
|
502
|
+
|
|
503
|
+
// Durable Objects
|
|
504
|
+
if (bindings.durableObjects && bindings.durableObjects.length > 0) {
|
|
505
|
+
result.durable_objects = {
|
|
506
|
+
bindings: bindings.durableObjects.map((dobj) => {
|
|
507
|
+
const entry: Record<string, string> = {
|
|
508
|
+
name: dobj.name,
|
|
509
|
+
class_name: dobj.class_name,
|
|
510
|
+
};
|
|
511
|
+
if (dobj.script_name) entry.script_name = dobj.script_name;
|
|
512
|
+
return entry;
|
|
513
|
+
}),
|
|
514
|
+
};
|
|
515
|
+
}
|
|
516
|
+
|
|
517
|
+
return result;
|
|
518
|
+
}
|
|
519
|
+
|
|
520
|
+
/**
|
|
521
|
+
* One-level deep merge: for each key in `override`, if both the base and
|
|
522
|
+
* override values are plain objects (not arrays, not null), merge their
|
|
523
|
+
* keys with override winning on conflicts. Everything else (primitives,
|
|
524
|
+
* arrays, null) is replaced outright — matching the intuitive behavior
|
|
525
|
+
* where `kv_namespaces: [...]` replaces fully but
|
|
526
|
+
* `durable_objects: { migrations }` merges with generated `bindings`.
|
|
527
|
+
*/
|
|
528
|
+
function shallowDeepMerge(
|
|
529
|
+
base: Record<string, unknown>,
|
|
530
|
+
override: Record<string, unknown>
|
|
531
|
+
): Record<string, unknown> {
|
|
532
|
+
const result = { ...base };
|
|
533
|
+
for (const key of Object.keys(override)) {
|
|
534
|
+
const baseVal = result[key];
|
|
535
|
+
const overVal = override[key];
|
|
536
|
+
if (isPlainObject(baseVal) && isPlainObject(overVal)) {
|
|
537
|
+
result[key] = { ...baseVal, ...overVal };
|
|
538
|
+
} else {
|
|
539
|
+
result[key] = overVal;
|
|
540
|
+
}
|
|
541
|
+
}
|
|
542
|
+
return result;
|
|
543
|
+
}
|
|
544
|
+
|
|
545
|
+
/** True for `{}` literals — false for arrays, null, Date, etc. */
|
|
546
|
+
function isPlainObject(val: unknown): val is Record<string, unknown> {
|
|
547
|
+
return val !== null && typeof val === 'object' && !Array.isArray(val);
|
|
548
|
+
}
|
|
549
|
+
|
|
246
550
|
/** @internal Exported for testing. */
|
|
247
551
|
export function generateWranglerConfig(
|
|
248
552
|
config: TimberConfig,
|
|
@@ -267,12 +571,20 @@ export function generateWranglerConfig(
|
|
|
267
571
|
},
|
|
268
572
|
};
|
|
269
573
|
|
|
270
|
-
//
|
|
574
|
+
// Layer 1: merge bindings-generated sections into base
|
|
575
|
+
const bindingsConfig = generateBindingsConfig(options.bindings);
|
|
576
|
+
const merged = { ...base, ...bindingsConfig };
|
|
577
|
+
|
|
578
|
+
// Layer 2: wrangler escape hatch with deep merge for nested plain objects.
|
|
579
|
+
// A shallow spread would replace entire sections like durable_objects and
|
|
580
|
+
// queues — e.g. adding migrations via wrangler would drop the generated
|
|
581
|
+
// durable_objects.bindings. Deep merge preserves generated keys while
|
|
582
|
+
// letting wrangler override on actual key-level conflicts.
|
|
271
583
|
if (options.wrangler) {
|
|
272
|
-
return
|
|
584
|
+
return shallowDeepMerge(merged, options.wrangler);
|
|
273
585
|
}
|
|
274
586
|
|
|
275
|
-
return
|
|
587
|
+
return merged;
|
|
276
588
|
}
|
|
277
589
|
|
|
278
590
|
// ─── Preview ─────────────────────────────────────────────────────────────────
|
|
@@ -310,16 +622,18 @@ function spawnPreviewProcess(command: string, args: string[], cwd: string): Prom
|
|
|
310
622
|
}
|
|
311
623
|
|
|
312
624
|
// ─── Cloudflare Workers type stubs ───────────────────────────────────────────
|
|
313
|
-
//
|
|
314
|
-
//
|
|
315
|
-
|
|
316
|
-
|
|
317
|
-
|
|
318
|
-
|
|
319
|
-
|
|
320
|
-
|
|
625
|
+
// Local type declarations so this file compiles without @cloudflare/workers-types.
|
|
626
|
+
// These are NOT declared globally — that would clash with @cloudflare/workers-types
|
|
627
|
+
// when users install it. Instead, they're module-scoped and used only within
|
|
628
|
+
// this file's function signatures.
|
|
629
|
+
|
|
630
|
+
/** @internal Minimal stub — use @cloudflare/workers-types for full types. */
|
|
631
|
+
export interface CfExecutionContext {
|
|
632
|
+
waitUntil(promise: Promise<unknown>): void;
|
|
633
|
+
passThroughOnException(): void;
|
|
634
|
+
}
|
|
321
635
|
|
|
322
|
-
|
|
323
|
-
|
|
324
|
-
|
|
636
|
+
/** @internal Minimal stub — use @cloudflare/workers-types for full types. */
|
|
637
|
+
export interface CfExportedHandler<Env = Record<string, unknown>> {
|
|
638
|
+
fetch?(request: Request, env: Env, ctx: CfExecutionContext): Promise<Response> | Response;
|
|
325
639
|
}
|
|
@@ -22,9 +22,11 @@
|
|
|
22
22
|
export function generateCompressModule(): string {
|
|
23
23
|
return `// Generated by @timber-js/app — response compression for self-hosted deployments.
|
|
24
24
|
// Do not edit — this file is regenerated on each build.
|
|
25
|
-
// Uses
|
|
26
|
-
//
|
|
27
|
-
//
|
|
25
|
+
// Uses node:zlib createGzip() (C++ native) on Node.js, falls back to
|
|
26
|
+
// CompressionStream (Web API) on other runtimes. Brotli is left to CDNs/reverse
|
|
27
|
+
// proxies — at streaming quality levels its ratio advantage is marginal.
|
|
28
|
+
import { Readable } from 'node:stream';
|
|
29
|
+
import { createGzip, constants } from 'node:zlib';
|
|
28
30
|
|
|
29
31
|
const COMPRESSIBLE_TYPES = new Set([
|
|
30
32
|
'text/html', 'text/css', 'text/plain', 'text/xml', 'text/javascript',
|
|
@@ -71,7 +73,25 @@ function shouldCompress(response) {
|
|
|
71
73
|
}
|
|
72
74
|
|
|
73
75
|
function compressWithGzip(body) {
|
|
74
|
-
|
|
76
|
+
// Use node:zlib (C++ native) for gzip compression. The Web Streams
|
|
77
|
+
// CompressionStream works but every chunk crosses the JS/Promise boundary.
|
|
78
|
+
// node:zlib.createGzip() compresses entirely in C++ with zero per-chunk
|
|
79
|
+
// Promise overhead.
|
|
80
|
+
//
|
|
81
|
+
// Convert: Web ReadableStream → Node Readable → pipe through gzip →
|
|
82
|
+
// Node Readable → Readable.toWeb() → Web ReadableStream
|
|
83
|
+
try {
|
|
84
|
+
const nodeReadable = Readable.fromWeb(body);
|
|
85
|
+
// Z_SYNC_FLUSH ensures each chunk is flushed immediately so the browser
|
|
86
|
+
// receives the HTML shell before Suspense boundaries resolve. Without it,
|
|
87
|
+
// gzip buffers internally and breaks streaming.
|
|
88
|
+
const gzip = createGzip({ flush: constants.Z_SYNC_FLUSH });
|
|
89
|
+
const compressed = nodeReadable.pipe(gzip);
|
|
90
|
+
return Readable.toWeb(compressed);
|
|
91
|
+
} catch {
|
|
92
|
+
// Fallback: CompressionStream (CF Workers, or if node:stream unavailable)
|
|
93
|
+
return body.pipeThrough(new CompressionStream('gzip'));
|
|
94
|
+
}
|
|
75
95
|
}
|
|
76
96
|
|
|
77
97
|
export function compressResponse(request, response) {
|