@methodts/runtime 0.1.0
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/dist/__fixtures__/executor-fixtures.d.ts +10 -0
- package/dist/__fixtures__/executor-fixtures.d.ts.map +1 -0
- package/dist/__fixtures__/executor-fixtures.js +36 -0
- package/dist/__fixtures__/executor-fixtures.js.map +1 -0
- package/dist/architecture.test.d.ts +2 -0
- package/dist/architecture.test.d.ts.map +1 -0
- package/dist/architecture.test.js +143 -0
- package/dist/architecture.test.js.map +1 -0
- package/dist/config/cost-governor-config.d.ts +40 -0
- package/dist/config/cost-governor-config.d.ts.map +1 -0
- package/dist/config/cost-governor-config.js +48 -0
- package/dist/config/cost-governor-config.js.map +1 -0
- package/dist/config/index.d.ts +7 -0
- package/dist/config/index.d.ts.map +1 -0
- package/dist/config/index.js +6 -0
- package/dist/config/index.js.map +1 -0
- package/dist/config/sessions-config.d.ts +29 -0
- package/dist/config/sessions-config.d.ts.map +1 -0
- package/dist/config/sessions-config.js +37 -0
- package/dist/config/sessions-config.js.map +1 -0
- package/dist/config/strategies-config.d.ts +38 -0
- package/dist/config/strategies-config.d.ts.map +1 -0
- package/dist/config/strategies-config.js +41 -0
- package/dist/config/strategies-config.js.map +1 -0
- package/dist/cost-governor/backpressure-queue.d.ts +20 -0
- package/dist/cost-governor/backpressure-queue.d.ts.map +1 -0
- package/dist/cost-governor/backpressure-queue.js +78 -0
- package/dist/cost-governor/backpressure-queue.js.map +1 -0
- package/dist/cost-governor/backpressure-queue.test.d.ts +2 -0
- package/dist/cost-governor/backpressure-queue.test.d.ts.map +1 -0
- package/dist/cost-governor/backpressure-queue.test.js +48 -0
- package/dist/cost-governor/backpressure-queue.test.js.map +1 -0
- package/dist/cost-governor/cost-events.d.ts +65 -0
- package/dist/cost-governor/cost-events.d.ts.map +1 -0
- package/dist/cost-governor/cost-events.js +48 -0
- package/dist/cost-governor/cost-events.js.map +1 -0
- package/dist/cost-governor/cost-governor-app-id.test.d.ts +19 -0
- package/dist/cost-governor/cost-governor-app-id.test.d.ts.map +1 -0
- package/dist/cost-governor/cost-governor-app-id.test.js +201 -0
- package/dist/cost-governor/cost-governor-app-id.test.js.map +1 -0
- package/dist/cost-governor/cost-oracle-impl.d.ts +19 -0
- package/dist/cost-governor/cost-oracle-impl.d.ts.map +1 -0
- package/dist/cost-governor/cost-oracle-impl.js +52 -0
- package/dist/cost-governor/cost-oracle-impl.js.map +1 -0
- package/dist/cost-governor/estimator.d.ts +22 -0
- package/dist/cost-governor/estimator.d.ts.map +1 -0
- package/dist/cost-governor/estimator.js +119 -0
- package/dist/cost-governor/estimator.js.map +1 -0
- package/dist/cost-governor/estimator.test.d.ts +2 -0
- package/dist/cost-governor/estimator.test.d.ts.map +1 -0
- package/dist/cost-governor/estimator.test.js +141 -0
- package/dist/cost-governor/estimator.test.js.map +1 -0
- package/dist/cost-governor/index.d.ts +75 -0
- package/dist/cost-governor/index.d.ts.map +1 -0
- package/dist/cost-governor/index.js +120 -0
- package/dist/cost-governor/index.js.map +1 -0
- package/dist/cost-governor/observations-store.d.ts +49 -0
- package/dist/cost-governor/observations-store.d.ts.map +1 -0
- package/dist/cost-governor/observations-store.js +179 -0
- package/dist/cost-governor/observations-store.js.map +1 -0
- package/dist/cost-governor/observations-store.test.d.ts +2 -0
- package/dist/cost-governor/observations-store.test.d.ts.map +1 -0
- package/dist/cost-governor/observations-store.test.js +191 -0
- package/dist/cost-governor/observations-store.test.js.map +1 -0
- package/dist/cost-governor/percentile.d.ts +17 -0
- package/dist/cost-governor/percentile.d.ts.map +1 -0
- package/dist/cost-governor/percentile.js +33 -0
- package/dist/cost-governor/percentile.js.map +1 -0
- package/dist/cost-governor/percentile.test.d.ts +2 -0
- package/dist/cost-governor/percentile.test.d.ts.map +1 -0
- package/dist/cost-governor/percentile.test.js +46 -0
- package/dist/cost-governor/percentile.test.js.map +1 -0
- package/dist/cost-governor/rate-governor-impl.d.ts +73 -0
- package/dist/cost-governor/rate-governor-impl.d.ts.map +1 -0
- package/dist/cost-governor/rate-governor-impl.js +148 -0
- package/dist/cost-governor/rate-governor-impl.js.map +1 -0
- package/dist/cost-governor/signature-builder.d.ts +22 -0
- package/dist/cost-governor/signature-builder.d.ts.map +1 -0
- package/dist/cost-governor/signature-builder.js +43 -0
- package/dist/cost-governor/signature-builder.js.map +1 -0
- package/dist/cost-governor/signature-builder.test.d.ts +2 -0
- package/dist/cost-governor/signature-builder.test.d.ts.map +1 -0
- package/dist/cost-governor/signature-builder.test.js +58 -0
- package/dist/cost-governor/signature-builder.test.js.map +1 -0
- package/dist/cost-governor/token-bucket.d.ts +57 -0
- package/dist/cost-governor/token-bucket.d.ts.map +1 -0
- package/dist/cost-governor/token-bucket.js +109 -0
- package/dist/cost-governor/token-bucket.js.map +1 -0
- package/dist/cost-governor/token-bucket.test.d.ts +2 -0
- package/dist/cost-governor/token-bucket.test.d.ts.map +1 -0
- package/dist/cost-governor/token-bucket.test.js +67 -0
- package/dist/cost-governor/token-bucket.test.js.map +1 -0
- package/dist/dlq/cortex-dlq-observer.d.ts +22 -0
- package/dist/dlq/cortex-dlq-observer.d.ts.map +1 -0
- package/dist/dlq/cortex-dlq-observer.js +29 -0
- package/dist/dlq/cortex-dlq-observer.js.map +1 -0
- package/dist/dlq/dlq-observer.test.d.ts +8 -0
- package/dist/dlq/dlq-observer.test.d.ts.map +1 -0
- package/dist/dlq/dlq-observer.test.js +103 -0
- package/dist/dlq/dlq-observer.test.js.map +1 -0
- package/dist/dlq/index.d.ts +6 -0
- package/dist/dlq/index.d.ts.map +1 -0
- package/dist/dlq/index.js +6 -0
- package/dist/dlq/index.js.map +1 -0
- package/dist/event-bus/adapters.d.ts +50 -0
- package/dist/event-bus/adapters.d.ts.map +1 -0
- package/dist/event-bus/adapters.js +51 -0
- package/dist/event-bus/adapters.js.map +1 -0
- package/dist/event-bus/adapters.test.d.ts +5 -0
- package/dist/event-bus/adapters.test.d.ts.map +1 -0
- package/dist/event-bus/adapters.test.js +73 -0
- package/dist/event-bus/adapters.test.js.map +1 -0
- package/dist/event-bus/agent-event-adapter.d.ts +22 -0
- package/dist/event-bus/agent-event-adapter.d.ts.map +1 -0
- package/dist/event-bus/agent-event-adapter.js +49 -0
- package/dist/event-bus/agent-event-adapter.js.map +1 -0
- package/dist/event-bus/agent-event-adapter.test.d.ts +5 -0
- package/dist/event-bus/agent-event-adapter.test.d.ts.map +1 -0
- package/dist/event-bus/agent-event-adapter.test.js +170 -0
- package/dist/event-bus/agent-event-adapter.test.js.map +1 -0
- package/dist/event-bus/channel-sink.d.ts +71 -0
- package/dist/event-bus/channel-sink.d.ts.map +1 -0
- package/dist/event-bus/channel-sink.js +159 -0
- package/dist/event-bus/channel-sink.js.map +1 -0
- package/dist/event-bus/channel-sink.test.d.ts +5 -0
- package/dist/event-bus/channel-sink.test.d.ts.map +1 -0
- package/dist/event-bus/channel-sink.test.js +234 -0
- package/dist/event-bus/channel-sink.test.js.map +1 -0
- package/dist/event-bus/event-types.snapshot.test.d.ts +27 -0
- package/dist/event-bus/event-types.snapshot.test.d.ts.map +1 -0
- package/dist/event-bus/event-types.snapshot.test.js +165 -0
- package/dist/event-bus/event-types.snapshot.test.js.map +1 -0
- package/dist/event-bus/genesis-sink.d.ts +55 -0
- package/dist/event-bus/genesis-sink.d.ts.map +1 -0
- package/dist/event-bus/genesis-sink.js +141 -0
- package/dist/event-bus/genesis-sink.js.map +1 -0
- package/dist/event-bus/genesis-sink.test.d.ts +5 -0
- package/dist/event-bus/genesis-sink.test.d.ts.map +1 -0
- package/dist/event-bus/genesis-sink.test.js +160 -0
- package/dist/event-bus/genesis-sink.test.js.map +1 -0
- package/dist/event-bus/in-memory-event-bus.d.ts +60 -0
- package/dist/event-bus/in-memory-event-bus.d.ts.map +1 -0
- package/dist/event-bus/in-memory-event-bus.js +274 -0
- package/dist/event-bus/in-memory-event-bus.js.map +1 -0
- package/dist/event-bus/in-memory-event-bus.test.d.ts +5 -0
- package/dist/event-bus/in-memory-event-bus.test.d.ts.map +1 -0
- package/dist/event-bus/in-memory-event-bus.test.js +457 -0
- package/dist/event-bus/in-memory-event-bus.test.js.map +1 -0
- package/dist/event-bus/index.d.ts +22 -0
- package/dist/event-bus/index.d.ts.map +1 -0
- package/dist/event-bus/index.js +17 -0
- package/dist/event-bus/index.js.map +1 -0
- package/dist/event-bus/persistence-sink.d.ts +74 -0
- package/dist/event-bus/persistence-sink.d.ts.map +1 -0
- package/dist/event-bus/persistence-sink.js +193 -0
- package/dist/event-bus/persistence-sink.js.map +1 -0
- package/dist/event-bus/persistence-sink.test.d.ts +6 -0
- package/dist/event-bus/persistence-sink.test.d.ts.map +1 -0
- package/dist/event-bus/persistence-sink.test.js +319 -0
- package/dist/event-bus/persistence-sink.test.js.map +1 -0
- package/dist/event-bus/session-checkpoint-sink.d.ts +91 -0
- package/dist/event-bus/session-checkpoint-sink.d.ts.map +1 -0
- package/dist/event-bus/session-checkpoint-sink.js +107 -0
- package/dist/event-bus/session-checkpoint-sink.js.map +1 -0
- package/dist/event-bus/session-checkpoint-sink.test.d.ts +5 -0
- package/dist/event-bus/session-checkpoint-sink.test.d.ts.map +1 -0
- package/dist/event-bus/session-checkpoint-sink.test.js +215 -0
- package/dist/event-bus/session-checkpoint-sink.test.js.map +1 -0
- package/dist/event-bus/webhook-connector.d.ts +59 -0
- package/dist/event-bus/webhook-connector.d.ts.map +1 -0
- package/dist/event-bus/webhook-connector.js +191 -0
- package/dist/event-bus/webhook-connector.js.map +1 -0
- package/dist/event-bus/webhook-connector.test.d.ts +5 -0
- package/dist/event-bus/webhook-connector.test.d.ts.map +1 -0
- package/dist/event-bus/webhook-connector.test.js +214 -0
- package/dist/event-bus/webhook-connector.test.js.map +1 -0
- package/dist/executors/cortex-job-backed-executor.d.ts +137 -0
- package/dist/executors/cortex-job-backed-executor.d.ts.map +1 -0
- package/dist/executors/cortex-job-backed-executor.js +441 -0
- package/dist/executors/cortex-job-backed-executor.js.map +1 -0
- package/dist/executors/cortex-job-backed-executor.test.d.ts +13 -0
- package/dist/executors/cortex-job-backed-executor.test.d.ts.map +1 -0
- package/dist/executors/cortex-job-backed-executor.test.js +303 -0
- package/dist/executors/cortex-job-backed-executor.test.js.map +1 -0
- package/dist/executors/index.d.ts +9 -0
- package/dist/executors/index.d.ts.map +1 -0
- package/dist/executors/index.js +9 -0
- package/dist/executors/index.js.map +1 -0
- package/dist/index.d.ts +2 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +13 -0
- package/dist/index.js.map +1 -0
- package/dist/ports/checkpoint-sink.d.ts +69 -0
- package/dist/ports/checkpoint-sink.d.ts.map +1 -0
- package/dist/ports/checkpoint-sink.js +16 -0
- package/dist/ports/checkpoint-sink.js.map +1 -0
- package/dist/ports/checkpoint.d.ts +68 -0
- package/dist/ports/checkpoint.d.ts.map +1 -0
- package/dist/ports/checkpoint.js +14 -0
- package/dist/ports/checkpoint.js.map +1 -0
- package/dist/ports/continuation-envelope-cross-app.test.d.ts +10 -0
- package/dist/ports/continuation-envelope-cross-app.test.d.ts.map +1 -0
- package/dist/ports/continuation-envelope-cross-app.test.js +95 -0
- package/dist/ports/continuation-envelope-cross-app.test.js.map +1 -0
- package/dist/ports/continuation-envelope.d.ts +199 -0
- package/dist/ports/continuation-envelope.d.ts.map +1 -0
- package/dist/ports/continuation-envelope.js +69 -0
- package/dist/ports/continuation-envelope.js.map +1 -0
- package/dist/ports/conversation.d.ts +64 -0
- package/dist/ports/conversation.d.ts.map +1 -0
- package/dist/ports/conversation.js +19 -0
- package/dist/ports/conversation.js.map +1 -0
- package/dist/ports/cost-oracle.d.ts +26 -0
- package/dist/ports/cost-oracle.d.ts.map +1 -0
- package/dist/ports/cost-oracle.js +9 -0
- package/dist/ports/cost-oracle.js.map +1 -0
- package/dist/ports/cross-app-invoker.d.ts +198 -0
- package/dist/ports/cross-app-invoker.d.ts.map +1 -0
- package/dist/ports/cross-app-invoker.js +157 -0
- package/dist/ports/cross-app-invoker.js.map +1 -0
- package/dist/ports/dlq-observer.d.ts +40 -0
- package/dist/ports/dlq-observer.d.ts.map +1 -0
- package/dist/ports/dlq-observer.js +20 -0
- package/dist/ports/dlq-observer.js.map +1 -0
- package/dist/ports/event-bus.d.ts +169 -0
- package/dist/ports/event-bus.d.ts.map +1 -0
- package/dist/ports/event-bus.js +20 -0
- package/dist/ports/event-bus.js.map +1 -0
- package/dist/ports/event-reader.d.ts +21 -0
- package/dist/ports/event-reader.d.ts.map +1 -0
- package/dist/ports/event-reader.js +12 -0
- package/dist/ports/event-reader.js.map +1 -0
- package/dist/ports/event-rotator.d.ts +39 -0
- package/dist/ports/event-rotator.d.ts.map +1 -0
- package/dist/ports/event-rotator.js +15 -0
- package/dist/ports/event-rotator.js.map +1 -0
- package/dist/ports/file-system.d.ts +48 -0
- package/dist/ports/file-system.d.ts.map +1 -0
- package/dist/ports/file-system.js +10 -0
- package/dist/ports/file-system.js.map +1 -0
- package/dist/ports/historical-observations.d.ts +39 -0
- package/dist/ports/historical-observations.d.ts.map +1 -0
- package/dist/ports/historical-observations.js +12 -0
- package/dist/ports/historical-observations.js.map +1 -0
- package/dist/ports/in-memory-source.d.ts +46 -0
- package/dist/ports/in-memory-source.d.ts.map +1 -0
- package/dist/ports/in-memory-source.js +69 -0
- package/dist/ports/in-memory-source.js.map +1 -0
- package/dist/ports/index.d.ts +32 -0
- package/dist/ports/index.d.ts.map +1 -0
- package/dist/ports/index.js +10 -0
- package/dist/ports/index.js.map +1 -0
- package/dist/ports/job-backed-executor.d.ts +139 -0
- package/dist/ports/job-backed-executor.d.ts.map +1 -0
- package/dist/ports/job-backed-executor.js +56 -0
- package/dist/ports/job-backed-executor.js.map +1 -0
- package/dist/ports/methodology-source.d.ts +95 -0
- package/dist/ports/methodology-source.d.ts.map +1 -0
- package/dist/ports/methodology-source.js +26 -0
- package/dist/ports/methodology-source.js.map +1 -0
- package/dist/ports/native-session-discovery.d.ts +20 -0
- package/dist/ports/native-session-discovery.d.ts.map +1 -0
- package/dist/ports/native-session-discovery.js +13 -0
- package/dist/ports/native-session-discovery.js.map +1 -0
- package/dist/ports/projection-store.d.ts +48 -0
- package/dist/ports/projection-store.d.ts.map +1 -0
- package/dist/ports/projection-store.js +17 -0
- package/dist/ports/projection-store.js.map +1 -0
- package/dist/ports/projection.d.ts +29 -0
- package/dist/ports/projection.d.ts.map +1 -0
- package/dist/ports/projection.js +13 -0
- package/dist/ports/projection.js.map +1 -0
- package/dist/ports/rate-governor.d.ts +17 -0
- package/dist/ports/rate-governor.d.ts.map +1 -0
- package/dist/ports/rate-governor.js +11 -0
- package/dist/ports/rate-governor.js.map +1 -0
- package/dist/ports/schedule-client.d.ts +29 -0
- package/dist/ports/schedule-client.d.ts.map +1 -0
- package/dist/ports/schedule-client.js +2 -0
- package/dist/ports/schedule-client.js.map +1 -0
- package/dist/ports/session-pool.d.ts +162 -0
- package/dist/ports/session-pool.d.ts.map +1 -0
- package/dist/ports/session-pool.js +21 -0
- package/dist/ports/session-pool.js.map +1 -0
- package/dist/ports/session-store-errors.d.ts +22 -0
- package/dist/ports/session-store-errors.d.ts.map +1 -0
- package/dist/ports/session-store-errors.js +29 -0
- package/dist/ports/session-store-errors.js.map +1 -0
- package/dist/ports/session-store-types.d.ts +132 -0
- package/dist/ports/session-store-types.d.ts.map +1 -0
- package/dist/ports/session-store-types.js +14 -0
- package/dist/ports/session-store-types.js.map +1 -0
- package/dist/ports/session-store.d.ts +69 -0
- package/dist/ports/session-store.d.ts.map +1 -0
- package/dist/ports/session-store.js +23 -0
- package/dist/ports/session-store.js.map +1 -0
- package/dist/ports/yaml-loader.d.ts +15 -0
- package/dist/ports/yaml-loader.d.ts.map +1 -0
- package/dist/ports/yaml-loader.js +13 -0
- package/dist/ports/yaml-loader.js.map +1 -0
- package/dist/scheduling/index.d.ts +6 -0
- package/dist/scheduling/index.d.ts.map +1 -0
- package/dist/scheduling/index.js +6 -0
- package/dist/scheduling/index.js.map +1 -0
- package/dist/scheduling/scheduled-pact.d.ts +97 -0
- package/dist/scheduling/scheduled-pact.d.ts.map +1 -0
- package/dist/scheduling/scheduled-pact.js +89 -0
- package/dist/scheduling/scheduled-pact.js.map +1 -0
- package/dist/sessions/__tests__/cognitive-modules.test.d.ts +10 -0
- package/dist/sessions/__tests__/cognitive-modules.test.d.ts.map +1 -0
- package/dist/sessions/__tests__/cognitive-modules.test.js +535 -0
- package/dist/sessions/__tests__/cognitive-modules.test.js.map +1 -0
- package/dist/sessions/__tests__/cognitive-provider.test.d.ts +13 -0
- package/dist/sessions/__tests__/cognitive-provider.test.d.ts.map +1 -0
- package/dist/sessions/__tests__/cognitive-provider.test.js +331 -0
- package/dist/sessions/__tests__/cognitive-provider.test.js.map +1 -0
- package/dist/sessions/__tests__/cognitive-sink.test.d.ts +19 -0
- package/dist/sessions/__tests__/cognitive-sink.test.d.ts.map +1 -0
- package/dist/sessions/__tests__/cognitive-sink.test.js +334 -0
- package/dist/sessions/__tests__/cognitive-sink.test.js.map +1 -0
- package/dist/sessions/__tests__/runtime-tools.test.d.ts +2 -0
- package/dist/sessions/__tests__/runtime-tools.test.d.ts.map +1 -0
- package/dist/sessions/__tests__/runtime-tools.test.js +83 -0
- package/dist/sessions/__tests__/runtime-tools.test.js.map +1 -0
- package/dist/sessions/auto-retro.d.ts +29 -0
- package/dist/sessions/auto-retro.d.ts.map +1 -0
- package/dist/sessions/auto-retro.js +181 -0
- package/dist/sessions/auto-retro.js.map +1 -0
- package/dist/sessions/auto-retro.test.d.ts +2 -0
- package/dist/sessions/auto-retro.test.d.ts.map +1 -0
- package/dist/sessions/auto-retro.test.js +361 -0
- package/dist/sessions/auto-retro.test.js.map +1 -0
- package/dist/sessions/channels.d.ts +55 -0
- package/dist/sessions/channels.d.ts.map +1 -0
- package/dist/sessions/channels.js +118 -0
- package/dist/sessions/channels.js.map +1 -0
- package/dist/sessions/channels.test.d.ts +2 -0
- package/dist/sessions/channels.test.d.ts.map +1 -0
- package/dist/sessions/channels.test.js +285 -0
- package/dist/sessions/channels.test.js.map +1 -0
- package/dist/sessions/cognitive-modules.d.ts +100 -0
- package/dist/sessions/cognitive-modules.d.ts.map +1 -0
- package/dist/sessions/cognitive-modules.js +458 -0
- package/dist/sessions/cognitive-modules.js.map +1 -0
- package/dist/sessions/cognitive-provider.d.ts +42 -0
- package/dist/sessions/cognitive-provider.d.ts.map +1 -0
- package/dist/sessions/cognitive-provider.js +208 -0
- package/dist/sessions/cognitive-provider.js.map +1 -0
- package/dist/sessions/cognitive-sink.d.ts +73 -0
- package/dist/sessions/cognitive-sink.d.ts.map +1 -0
- package/dist/sessions/cognitive-sink.js +154 -0
- package/dist/sessions/cognitive-sink.js.map +1 -0
- package/dist/sessions/diagnostics.d.ts +70 -0
- package/dist/sessions/diagnostics.d.ts.map +1 -0
- package/dist/sessions/diagnostics.js +129 -0
- package/dist/sessions/diagnostics.js.map +1 -0
- package/dist/sessions/diagnostics.test.d.ts +2 -0
- package/dist/sessions/diagnostics.test.d.ts.map +1 -0
- package/dist/sessions/diagnostics.test.js +135 -0
- package/dist/sessions/diagnostics.test.js.map +1 -0
- package/dist/sessions/index.d.ts +32 -0
- package/dist/sessions/index.d.ts.map +1 -0
- package/dist/sessions/index.js +33 -0
- package/dist/sessions/index.js.map +1 -0
- package/dist/sessions/pool.d.ts +218 -0
- package/dist/sessions/pool.d.ts.map +1 -0
- package/dist/sessions/pool.js +991 -0
- package/dist/sessions/pool.js.map +1 -0
- package/dist/sessions/pool.test.d.ts +2 -0
- package/dist/sessions/pool.test.d.ts.map +1 -0
- package/dist/sessions/pool.test.js +633 -0
- package/dist/sessions/pool.test.js.map +1 -0
- package/dist/sessions/print-session.d.ts +142 -0
- package/dist/sessions/print-session.d.ts.map +1 -0
- package/dist/sessions/print-session.js +325 -0
- package/dist/sessions/print-session.js.map +1 -0
- package/dist/sessions/print-session.test.d.ts +2 -0
- package/dist/sessions/print-session.test.d.ts.map +1 -0
- package/dist/sessions/print-session.test.js +418 -0
- package/dist/sessions/print-session.test.js.map +1 -0
- package/dist/sessions/runtime-tools.d.ts +22 -0
- package/dist/sessions/runtime-tools.d.ts.map +1 -0
- package/dist/sessions/runtime-tools.js +162 -0
- package/dist/sessions/runtime-tools.js.map +1 -0
- package/dist/sessions/scope-hook.d.ts +77 -0
- package/dist/sessions/scope-hook.d.ts.map +1 -0
- package/dist/sessions/scope-hook.js +323 -0
- package/dist/sessions/scope-hook.js.map +1 -0
- package/dist/sessions/scope-hook.test.d.ts +2 -0
- package/dist/sessions/scope-hook.test.d.ts.map +1 -0
- package/dist/sessions/scope-hook.test.js +249 -0
- package/dist/sessions/scope-hook.test.js.map +1 -0
- package/dist/sessions/session-store/checkpoint-sink-impl.d.ts +16 -0
- package/dist/sessions/session-store/checkpoint-sink-impl.d.ts.map +1 -0
- package/dist/sessions/session-store/checkpoint-sink-impl.js +191 -0
- package/dist/sessions/session-store/checkpoint-sink-impl.js.map +1 -0
- package/dist/sessions/session-store/checkpoint-sink-impl.test.d.ts +5 -0
- package/dist/sessions/session-store/checkpoint-sink-impl.test.d.ts.map +1 -0
- package/dist/sessions/session-store/checkpoint-sink-impl.test.js +137 -0
- package/dist/sessions/session-store/checkpoint-sink-impl.test.js.map +1 -0
- package/dist/sessions/session-store/conformance.d.ts +59 -0
- package/dist/sessions/session-store/conformance.d.ts.map +1 -0
- package/dist/sessions/session-store/conformance.js +172 -0
- package/dist/sessions/session-store/conformance.js.map +1 -0
- package/dist/sessions/session-store/conformance.test.d.ts +7 -0
- package/dist/sessions/session-store/conformance.test.d.ts.map +1 -0
- package/dist/sessions/session-store/conformance.test.js +22 -0
- package/dist/sessions/session-store/conformance.test.js.map +1 -0
- package/dist/sessions/session-store/in-memory-session-store.d.ts +23 -0
- package/dist/sessions/session-store/in-memory-session-store.d.ts.map +1 -0
- package/dist/sessions/session-store/in-memory-session-store.js +197 -0
- package/dist/sessions/session-store/in-memory-session-store.js.map +1 -0
- package/dist/sessions/session-store/in-memory-session-store.test.d.ts +6 -0
- package/dist/sessions/session-store/in-memory-session-store.test.d.ts.map +1 -0
- package/dist/sessions/session-store/in-memory-session-store.test.js +183 -0
- package/dist/sessions/session-store/in-memory-session-store.test.js.map +1 -0
- package/dist/sessions/session-store/index.d.ts +20 -0
- package/dist/sessions/session-store/index.d.ts.map +1 -0
- package/dist/sessions/session-store/index.js +15 -0
- package/dist/sessions/session-store/index.js.map +1 -0
- package/dist/sessions/session-store/resume.d.ts +88 -0
- package/dist/sessions/session-store/resume.d.ts.map +1 -0
- package/dist/sessions/session-store/resume.js +96 -0
- package/dist/sessions/session-store/resume.js.map +1 -0
- package/dist/sessions/session-store/resume.test.d.ts +5 -0
- package/dist/sessions/session-store/resume.test.d.ts.map +1 -0
- package/dist/sessions/session-store/resume.test.js +119 -0
- package/dist/sessions/session-store/resume.test.js.map +1 -0
- package/dist/sessions/spawn-queue.d.ts +28 -0
- package/dist/sessions/spawn-queue.d.ts.map +1 -0
- package/dist/sessions/spawn-queue.js +63 -0
- package/dist/sessions/spawn-queue.js.map +1 -0
- package/dist/sessions/spawn-queue.test.d.ts +2 -0
- package/dist/sessions/spawn-queue.test.d.ts.map +1 -0
- package/dist/sessions/spawn-queue.test.js +65 -0
- package/dist/sessions/spawn-queue.test.js.map +1 -0
- package/dist/sessions/types.d.ts +16 -0
- package/dist/sessions/types.d.ts.map +1 -0
- package/dist/sessions/types.js +11 -0
- package/dist/sessions/types.js.map +1 -0
- package/dist/sessions/worktree-stale.test.d.ts +2 -0
- package/dist/sessions/worktree-stale.test.d.ts.map +1 -0
- package/dist/sessions/worktree-stale.test.js +468 -0
- package/dist/sessions/worktree-stale.test.js.map +1 -0
- package/dist/strategy/artifact-store.d.ts +12 -0
- package/dist/strategy/artifact-store.d.ts.map +1 -0
- package/dist/strategy/artifact-store.js +12 -0
- package/dist/strategy/artifact-store.js.map +1 -0
- package/dist/strategy/artifact-store.test.d.ts +2 -0
- package/dist/strategy/artifact-store.test.d.ts.map +1 -0
- package/dist/strategy/artifact-store.test.js +170 -0
- package/dist/strategy/artifact-store.test.js.map +1 -0
- package/dist/strategy/context-load-executor.d.ts +23 -0
- package/dist/strategy/context-load-executor.d.ts.map +1 -0
- package/dist/strategy/context-load-executor.js +87 -0
- package/dist/strategy/context-load-executor.js.map +1 -0
- package/dist/strategy/context-load-executor.test.d.ts +16 -0
- package/dist/strategy/context-load-executor.test.d.ts.map +1 -0
- package/dist/strategy/context-load-executor.test.js +158 -0
- package/dist/strategy/context-load-executor.test.js.map +1 -0
- package/dist/strategy/cortex-cross-app-invoker.stub.d.ts +65 -0
- package/dist/strategy/cortex-cross-app-invoker.stub.d.ts.map +1 -0
- package/dist/strategy/cortex-cross-app-invoker.stub.js +68 -0
- package/dist/strategy/cortex-cross-app-invoker.stub.js.map +1 -0
- package/dist/strategy/cross-app-node-executor.d.ts +54 -0
- package/dist/strategy/cross-app-node-executor.d.ts.map +1 -0
- package/dist/strategy/cross-app-node-executor.js +98 -0
- package/dist/strategy/cross-app-node-executor.js.map +1 -0
- package/dist/strategy/cross-app-node-executor.test.d.ts +13 -0
- package/dist/strategy/cross-app-node-executor.test.d.ts.map +1 -0
- package/dist/strategy/cross-app-node-executor.test.js +160 -0
- package/dist/strategy/cross-app-node-executor.test.js.map +1 -0
- package/dist/strategy/gates.d.ts +13 -0
- package/dist/strategy/gates.d.ts.map +1 -0
- package/dist/strategy/gates.js +13 -0
- package/dist/strategy/gates.js.map +1 -0
- package/dist/strategy/gates.test.d.ts +2 -0
- package/dist/strategy/gates.test.d.ts.map +1 -0
- package/dist/strategy/gates.test.js +299 -0
- package/dist/strategy/gates.test.js.map +1 -0
- package/dist/strategy/human-approval-resolver.d.ts +23 -0
- package/dist/strategy/human-approval-resolver.d.ts.map +1 -0
- package/dist/strategy/human-approval-resolver.js +94 -0
- package/dist/strategy/human-approval-resolver.js.map +1 -0
- package/dist/strategy/human-approval-resolver.test.d.ts +16 -0
- package/dist/strategy/human-approval-resolver.test.d.ts.map +1 -0
- package/dist/strategy/human-approval-resolver.test.js +200 -0
- package/dist/strategy/human-approval-resolver.test.js.map +1 -0
- package/dist/strategy/in-process-cross-app-invoker.d.ts +105 -0
- package/dist/strategy/in-process-cross-app-invoker.d.ts.map +1 -0
- package/dist/strategy/in-process-cross-app-invoker.js +206 -0
- package/dist/strategy/in-process-cross-app-invoker.js.map +1 -0
- package/dist/strategy/in-process-cross-app-invoker.test.d.ts +15 -0
- package/dist/strategy/in-process-cross-app-invoker.test.d.ts.map +1 -0
- package/dist/strategy/in-process-cross-app-invoker.test.js +190 -0
- package/dist/strategy/in-process-cross-app-invoker.test.js.map +1 -0
- package/dist/strategy/index.d.ts +29 -0
- package/dist/strategy/index.d.ts.map +1 -0
- package/dist/strategy/index.js +29 -0
- package/dist/strategy/index.js.map +1 -0
- package/dist/strategy/pacta-strategy.d.ts +97 -0
- package/dist/strategy/pacta-strategy.d.ts.map +1 -0
- package/dist/strategy/pacta-strategy.js +117 -0
- package/dist/strategy/pacta-strategy.js.map +1 -0
- package/dist/strategy/pacta-strategy.test.d.ts +2 -0
- package/dist/strategy/pacta-strategy.test.d.ts.map +1 -0
- package/dist/strategy/pacta-strategy.test.js +234 -0
- package/dist/strategy/pacta-strategy.test.js.map +1 -0
- package/dist/strategy/retro-generator.d.ts +18 -0
- package/dist/strategy/retro-generator.d.ts.map +1 -0
- package/dist/strategy/retro-generator.js +22 -0
- package/dist/strategy/retro-generator.js.map +1 -0
- package/dist/strategy/retro-writer.d.ts +25 -0
- package/dist/strategy/retro-writer.d.ts.map +1 -0
- package/dist/strategy/retro-writer.js +65 -0
- package/dist/strategy/retro-writer.js.map +1 -0
- package/dist/strategy/strategy-executor.d.ts +39 -0
- package/dist/strategy/strategy-executor.d.ts.map +1 -0
- package/dist/strategy/strategy-executor.js +253 -0
- package/dist/strategy/strategy-executor.js.map +1 -0
- package/dist/strategy/strategy-executor.test.d.ts +8 -0
- package/dist/strategy/strategy-executor.test.d.ts.map +1 -0
- package/dist/strategy/strategy-executor.test.js +1301 -0
- package/dist/strategy/strategy-executor.test.js.map +1 -0
- package/dist/strategy/strategy-parser.d.ts +30 -0
- package/dist/strategy/strategy-parser.d.ts.map +1 -0
- package/dist/strategy/strategy-parser.js +30 -0
- package/dist/strategy/strategy-parser.js.map +1 -0
- package/dist/strategy/sub-strategy-source.d.ts +27 -0
- package/dist/strategy/sub-strategy-source.d.ts.map +1 -0
- package/dist/strategy/sub-strategy-source.js +77 -0
- package/dist/strategy/sub-strategy-source.js.map +1 -0
- package/dist/strategy/types.d.ts +12 -0
- package/dist/strategy/types.d.ts.map +1 -0
- package/dist/strategy/types.js +9 -0
- package/dist/strategy/types.js.map +1 -0
- package/package.json +87 -0
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"persistence-sink.d.ts","sourceRoot":"","sources":["../../src/event-bus/persistence-sink.ts"],"names":[],"mappings":"AACA;;;;;;;;;;;;;GAaG;AAEH,OAAO,KAAK,EAAE,YAAY,EAAE,SAAS,EAAE,MAAM,uBAAuB,CAAC;AACrE,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,0BAA0B,CAAC;AAC5D,OAAO,KAAK,EAAE,kBAAkB,EAAE,MAAM,yBAAyB,CAAC;AAIlE,MAAM,WAAW,sBAAsB;IACrC,EAAE,EAAE,kBAAkB,CAAC;IACvB,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,iBAAiB,CAAC,EAAE,MAAM,CAAC;IAC3B,eAAe,CAAC,EAAE,MAAM,CAAC;IACzB,cAAc,CAAC,EAAE,MAAM,CAAC;CACzB;AAUD,qBAAa,eAAgB,YAAW,SAAS,EAAE,WAAW;IAC5D,QAAQ,CAAC,IAAI,iBAAiB;IAE9B,OAAO,CAAC,QAAQ,CAAC,EAAE,CAAqB;IACxC,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAS;IACjC,OAAO,CAAC,QAAQ,CAAC,WAAW,CAAS;IACrC,OAAO,CAAC,QAAQ,CAAC,iBAAiB,CAAS;IAC3C,OAAO,CAAC,QAAQ,CAAC,eAAe,CAAS;IACzC,OAAO,CAAC,QAAQ,CAAC,cAAc,CAAS;IAExC,OAAO,CAAC,MAAM,CAAsB;IACpC,OAAO,CAAC,UAAU,CAA8C;IAChE,OAAO,CAAC,QAAQ,CAAS;IACzB,OAAO,CAAC,YAAY,CAAK;IACzB,OAAO,CAAC,gBAAgB,CAAwC;gBAEpD,OAAO,EAAE,sBAAsB;IAS3C;;;OAGG;IACH,mBAAmB,CAAC,EAAE,EAAE,CAAC,GAAG,EAAE,MAAM,KAAK,IAAI,GAAG,IAAI;IAIpD;;OAEG;IACG,IAAI,IAAI,OAAO,CAAC,IAAI,CAAC;IAO3B,OAAO,CAAC,KAAK,EAAE,YAAY,GAAG,IAAI;IAkBlC,OAAO,CAAC,KAAK,EAAE,KAAK,EAAE,KAAK,EAAE,YAAY,GAAG,IAAI;IAM1C,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC;IAmC5B;;;;OAIG;YACW,aAAa;IA0B3B;;;OAGG;IACG,MAAM,IAAI,OAAO,CAAC,YAAY,EAAE,CAAC;IASvC;;;OAGG;IACG,eAAe,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,YAAY,EAAE,CAAC;IAO1D,WAAW,IAAI,OAAO,CAAC,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;IAS9C,WAAW,CAAC,OAAO,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,GAAG,OAAO,CAAC,IAAI,CAAC;YAInD,UAAU;IAYxB,kDAAkD;IAC5C,OAAO,IAAI,OAAO,CAAC,IAAI,CAAC;CAO/B"}
|
|
@@ -0,0 +1,193 @@
|
|
|
1
|
+
// SPDX-License-Identifier: Apache-2.0
|
|
2
|
+
/**
|
|
3
|
+
* PersistenceSink — PRD 026 Phase 3.
|
|
4
|
+
*
|
|
5
|
+
* Writes every RuntimeEvent to a JSONL file via the FileSystemProvider port.
|
|
6
|
+
* Uses write-ahead batching: events accumulate in memory and flush every
|
|
7
|
+
* 1 second or 100 events (whichever comes first). On flush failure, emits
|
|
8
|
+
* system.sink_overflow via a callback — never crashes.
|
|
9
|
+
*
|
|
10
|
+
* Also handles event replay on startup: reads the JSONL file, filters by
|
|
11
|
+
* a configurable replay window, and returns events for import into the bus.
|
|
12
|
+
*
|
|
13
|
+
* Cursor recovery: persists last-processed sequence per sink in a JSON file.
|
|
14
|
+
* On restart, sinks resume from their last cursor.
|
|
15
|
+
*/
|
|
16
|
+
const DEFAULT_LOG_PATH = '.method/events.jsonl';
|
|
17
|
+
const DEFAULT_CURSORS_PATH = '.method/events-cursors.json';
|
|
18
|
+
const DEFAULT_REPLAY_WINDOW_HOURS = 24;
|
|
19
|
+
const DEFAULT_FLUSH_INTERVAL_MS = 1000;
|
|
20
|
+
const DEFAULT_FLUSH_BATCH_SIZE = 100;
|
|
21
|
+
// ── PersistenceSink ─────────────────────────────────────────────
|
|
22
|
+
export class PersistenceSink {
|
|
23
|
+
name = 'persistence';
|
|
24
|
+
fs;
|
|
25
|
+
logPath;
|
|
26
|
+
cursorsPath;
|
|
27
|
+
replayWindowHours;
|
|
28
|
+
flushIntervalMs;
|
|
29
|
+
flushBatchSize;
|
|
30
|
+
buffer = [];
|
|
31
|
+
flushTimer = null;
|
|
32
|
+
flushing = false;
|
|
33
|
+
lastSequence = 0;
|
|
34
|
+
overflowCallback = null;
|
|
35
|
+
constructor(options) {
|
|
36
|
+
this.fs = options.fs;
|
|
37
|
+
this.logPath = options.logPath ?? DEFAULT_LOG_PATH;
|
|
38
|
+
this.cursorsPath = options.cursorsPath ?? DEFAULT_CURSORS_PATH;
|
|
39
|
+
this.replayWindowHours = options.replayWindowHours ?? DEFAULT_REPLAY_WINDOW_HOURS;
|
|
40
|
+
this.flushIntervalMs = options.flushIntervalMs ?? DEFAULT_FLUSH_INTERVAL_MS;
|
|
41
|
+
this.flushBatchSize = options.flushBatchSize ?? DEFAULT_FLUSH_BATCH_SIZE;
|
|
42
|
+
}
|
|
43
|
+
/**
|
|
44
|
+
* Set a callback to notify when flush fails (e.g., emit system.sink_overflow).
|
|
45
|
+
* Called from composition root after bus is created.
|
|
46
|
+
*/
|
|
47
|
+
setOverflowCallback(cb) {
|
|
48
|
+
this.overflowCallback = cb;
|
|
49
|
+
}
|
|
50
|
+
/**
|
|
51
|
+
* Initialize from persisted cursor state. Call before registering as a sink.
|
|
52
|
+
*/
|
|
53
|
+
async init() {
|
|
54
|
+
const cursors = await this.loadCursors();
|
|
55
|
+
this.lastSequence = cursors[this.name] ?? 0;
|
|
56
|
+
}
|
|
57
|
+
// ── EventSink interface ──────────────────────────────────────
|
|
58
|
+
onEvent(event) {
|
|
59
|
+
// Skip events already persisted (cursor recovery on replay)
|
|
60
|
+
if (event.sequence <= this.lastSequence)
|
|
61
|
+
return;
|
|
62
|
+
this.buffer.push(event);
|
|
63
|
+
this.lastSequence = event.sequence;
|
|
64
|
+
// Flush immediately if batch size reached
|
|
65
|
+
if (this.buffer.length >= this.flushBatchSize) {
|
|
66
|
+
this.flush().catch(() => { });
|
|
67
|
+
}
|
|
68
|
+
else if (!this.flushTimer) {
|
|
69
|
+
this.flushTimer = setTimeout(() => {
|
|
70
|
+
this.flushTimer = null;
|
|
71
|
+
this.flush().catch(() => { });
|
|
72
|
+
}, this.flushIntervalMs);
|
|
73
|
+
}
|
|
74
|
+
}
|
|
75
|
+
onError(error, event) {
|
|
76
|
+
console.error(`[persistence-sink] Error processing event ${event.id}:`, error.message);
|
|
77
|
+
}
|
|
78
|
+
// ── Flush ────────────────────────────────────────────────────
|
|
79
|
+
async flush() {
|
|
80
|
+
if (this.flushing || this.buffer.length === 0)
|
|
81
|
+
return;
|
|
82
|
+
this.flushing = true;
|
|
83
|
+
// Swap buffer so new events accumulate in a fresh array during write
|
|
84
|
+
const batch = this.buffer;
|
|
85
|
+
this.buffer = [];
|
|
86
|
+
if (this.flushTimer) {
|
|
87
|
+
clearTimeout(this.flushTimer);
|
|
88
|
+
this.flushTimer = null;
|
|
89
|
+
}
|
|
90
|
+
try {
|
|
91
|
+
const lines = batch.map(e => JSON.stringify(e)).join('\n') + '\n';
|
|
92
|
+
await this.fs.appendFile(this.logPath, lines, 'utf-8');
|
|
93
|
+
await this.saveCursor();
|
|
94
|
+
}
|
|
95
|
+
catch (err) {
|
|
96
|
+
const msg = err instanceof Error ? err.message : String(err);
|
|
97
|
+
console.error(`[persistence-sink] Flush failed: ${msg}`);
|
|
98
|
+
if (this.overflowCallback) {
|
|
99
|
+
try {
|
|
100
|
+
this.overflowCallback(msg);
|
|
101
|
+
}
|
|
102
|
+
catch { /* double fault */ }
|
|
103
|
+
}
|
|
104
|
+
// Put events back for retry on next flush
|
|
105
|
+
this.buffer = [...batch, ...this.buffer];
|
|
106
|
+
}
|
|
107
|
+
finally {
|
|
108
|
+
this.flushing = false;
|
|
109
|
+
}
|
|
110
|
+
}
|
|
111
|
+
// ── Replay ───────────────────────────────────────────────────
|
|
112
|
+
/**
|
|
113
|
+
* Read and parse all events from the JSONL log file.
|
|
114
|
+
* Skips corrupt lines gracefully. Returns empty array if the log is missing.
|
|
115
|
+
* Shared by replay() (time-window filter) and readEventsSince() (cursor filter).
|
|
116
|
+
*/
|
|
117
|
+
async readAllEvents() {
|
|
118
|
+
let content;
|
|
119
|
+
try {
|
|
120
|
+
content = await this.fs.readFile(this.logPath, 'utf-8');
|
|
121
|
+
}
|
|
122
|
+
catch {
|
|
123
|
+
console.warn(`[persistence-sink] No event log at ${this.logPath}, starting fresh`);
|
|
124
|
+
return [];
|
|
125
|
+
}
|
|
126
|
+
const events = [];
|
|
127
|
+
for (const line of content.split('\n')) {
|
|
128
|
+
const trimmed = line.trim();
|
|
129
|
+
if (!trimmed)
|
|
130
|
+
continue;
|
|
131
|
+
try {
|
|
132
|
+
const event = JSON.parse(trimmed);
|
|
133
|
+
events.push(event);
|
|
134
|
+
}
|
|
135
|
+
catch {
|
|
136
|
+
console.warn('[persistence-sink] Skipping corrupt JSONL line');
|
|
137
|
+
}
|
|
138
|
+
}
|
|
139
|
+
return events;
|
|
140
|
+
}
|
|
141
|
+
/**
|
|
142
|
+
* Read events from JSONL file for replay into the bus.
|
|
143
|
+
* Filters by replay window. Skips corrupt lines gracefully.
|
|
144
|
+
*/
|
|
145
|
+
async replay() {
|
|
146
|
+
const all = await this.readAllEvents();
|
|
147
|
+
const windowMs = this.replayWindowHours * 60 * 60 * 1000;
|
|
148
|
+
const cutoff = Date.now() - windowMs;
|
|
149
|
+
return all.filter(e => new Date(e.timestamp).getTime() >= cutoff);
|
|
150
|
+
}
|
|
151
|
+
// ── EventReader interface ────────────────────────────────────
|
|
152
|
+
/**
|
|
153
|
+
* Read events from the persistent log where sequence > sinceSeq.
|
|
154
|
+
* Returns events in append order. Corrupt JSONL lines are skipped gracefully.
|
|
155
|
+
*/
|
|
156
|
+
async readEventsSince(sinceSeq) {
|
|
157
|
+
const all = await this.readAllEvents();
|
|
158
|
+
return all.filter(e => e.sequence > sinceSeq);
|
|
159
|
+
}
|
|
160
|
+
// ── Cursor management ────────────────────────────────────────
|
|
161
|
+
async loadCursors() {
|
|
162
|
+
try {
|
|
163
|
+
const content = await this.fs.readFile(this.cursorsPath, 'utf-8');
|
|
164
|
+
return JSON.parse(content);
|
|
165
|
+
}
|
|
166
|
+
catch {
|
|
167
|
+
return {};
|
|
168
|
+
}
|
|
169
|
+
}
|
|
170
|
+
async saveCursors(cursors) {
|
|
171
|
+
await this.fs.writeFile(this.cursorsPath, JSON.stringify(cursors, null, 2), 'utf-8');
|
|
172
|
+
}
|
|
173
|
+
async saveCursor() {
|
|
174
|
+
try {
|
|
175
|
+
const cursors = await this.loadCursors();
|
|
176
|
+
cursors[this.name] = this.lastSequence;
|
|
177
|
+
await this.saveCursors(cursors);
|
|
178
|
+
}
|
|
179
|
+
catch (err) {
|
|
180
|
+
console.error('[persistence-sink] Failed to save cursor:', err);
|
|
181
|
+
}
|
|
182
|
+
}
|
|
183
|
+
// ── Lifecycle ────────────────────────────────────────────────
|
|
184
|
+
/** Flush remaining buffer and clean up timers. */
|
|
185
|
+
async dispose() {
|
|
186
|
+
if (this.flushTimer) {
|
|
187
|
+
clearTimeout(this.flushTimer);
|
|
188
|
+
this.flushTimer = null;
|
|
189
|
+
}
|
|
190
|
+
await this.flush();
|
|
191
|
+
}
|
|
192
|
+
}
|
|
193
|
+
//# sourceMappingURL=persistence-sink.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"persistence-sink.js","sourceRoot":"","sources":["../../src/event-bus/persistence-sink.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"persistence-sink.test.d.ts","sourceRoot":"","sources":["../../src/event-bus/persistence-sink.test.ts"],"names":[],"mappings":"AACA;;;GAGG"}
|
|
@@ -0,0 +1,319 @@
|
|
|
1
|
+
// SPDX-License-Identifier: Apache-2.0
|
|
2
|
+
/**
|
|
3
|
+
* Unit tests for PersistenceSink (PRD 026 Phase 3).
|
|
4
|
+
* Uses in-memory FileSystemProvider mock — no real I/O.
|
|
5
|
+
*/
|
|
6
|
+
import { describe, it, beforeEach } from 'node:test';
|
|
7
|
+
import assert from 'node:assert/strict';
|
|
8
|
+
import { PersistenceSink } from './persistence-sink.js';
|
|
9
|
+
// ── In-memory FileSystem mock ──────────────────────────────────
|
|
10
|
+
function createMemoryFs() {
|
|
11
|
+
const files = new Map();
|
|
12
|
+
const dirs = new Set();
|
|
13
|
+
const norm = (p) => p.replace(/\\/g, '/');
|
|
14
|
+
return {
|
|
15
|
+
files,
|
|
16
|
+
readFileSync(path, _encoding) {
|
|
17
|
+
const content = files.get(norm(path));
|
|
18
|
+
if (content === undefined)
|
|
19
|
+
throw new Error(`ENOENT: no such file: ${path}`);
|
|
20
|
+
return content;
|
|
21
|
+
},
|
|
22
|
+
writeFileSync(path, content) {
|
|
23
|
+
files.set(norm(path), content);
|
|
24
|
+
},
|
|
25
|
+
existsSync(path) {
|
|
26
|
+
return files.has(norm(path)) || dirs.has(norm(path));
|
|
27
|
+
},
|
|
28
|
+
readdirSync(_path, _options) {
|
|
29
|
+
return [];
|
|
30
|
+
},
|
|
31
|
+
statSync(_path) {
|
|
32
|
+
return { mtimeMs: 0, mtime: new Date(), size: 0, isFile: () => true, isDirectory: () => false };
|
|
33
|
+
},
|
|
34
|
+
unlinkSync(path) {
|
|
35
|
+
files.delete(norm(path));
|
|
36
|
+
},
|
|
37
|
+
mkdirSync(path) {
|
|
38
|
+
dirs.add(norm(path));
|
|
39
|
+
},
|
|
40
|
+
renameSync(oldPath, newPath) {
|
|
41
|
+
const content = files.get(norm(oldPath));
|
|
42
|
+
if (content !== undefined) {
|
|
43
|
+
files.set(norm(newPath), content);
|
|
44
|
+
files.delete(norm(oldPath));
|
|
45
|
+
}
|
|
46
|
+
},
|
|
47
|
+
realpathSync(path) {
|
|
48
|
+
return path;
|
|
49
|
+
},
|
|
50
|
+
async readFile(path, encoding) {
|
|
51
|
+
return this.readFileSync(path, encoding);
|
|
52
|
+
},
|
|
53
|
+
async writeFile(path, content) {
|
|
54
|
+
this.writeFileSync(path, content);
|
|
55
|
+
},
|
|
56
|
+
async appendFile(path, content) {
|
|
57
|
+
const existing = files.get(norm(path)) ?? '';
|
|
58
|
+
files.set(norm(path), existing + content);
|
|
59
|
+
},
|
|
60
|
+
async readdir(path) {
|
|
61
|
+
return this.readdirSync(path);
|
|
62
|
+
},
|
|
63
|
+
async stat(path) {
|
|
64
|
+
return this.statSync(path);
|
|
65
|
+
},
|
|
66
|
+
async access() { },
|
|
67
|
+
async mkdir(path, opts) {
|
|
68
|
+
this.mkdirSync(path, opts);
|
|
69
|
+
},
|
|
70
|
+
};
|
|
71
|
+
}
|
|
72
|
+
// ── Test helpers ───────────────────────────────────────────────
|
|
73
|
+
function makeEvent(seq, overrides = {}) {
|
|
74
|
+
return {
|
|
75
|
+
id: `evt-${seq}`,
|
|
76
|
+
version: 1,
|
|
77
|
+
timestamp: new Date().toISOString(),
|
|
78
|
+
sequence: seq,
|
|
79
|
+
domain: 'session',
|
|
80
|
+
type: 'session.spawned',
|
|
81
|
+
severity: 'info',
|
|
82
|
+
payload: { test: true },
|
|
83
|
+
source: 'test',
|
|
84
|
+
...overrides,
|
|
85
|
+
};
|
|
86
|
+
}
|
|
87
|
+
// ── Tests ──────────────────────────────────────────────────────
|
|
88
|
+
describe('PersistenceSink', () => {
|
|
89
|
+
let fs;
|
|
90
|
+
let sink;
|
|
91
|
+
const logPath = '.method/events.jsonl';
|
|
92
|
+
const cursorsPath = '.method/events-cursors.json';
|
|
93
|
+
beforeEach(async () => {
|
|
94
|
+
fs = createMemoryFs();
|
|
95
|
+
sink = new PersistenceSink({
|
|
96
|
+
fs,
|
|
97
|
+
logPath,
|
|
98
|
+
cursorsPath,
|
|
99
|
+
flushIntervalMs: 50, // fast for testing
|
|
100
|
+
flushBatchSize: 3,
|
|
101
|
+
});
|
|
102
|
+
await sink.init();
|
|
103
|
+
});
|
|
104
|
+
describe('onEvent + flush', () => {
|
|
105
|
+
it('buffers events and flushes at batch size', async () => {
|
|
106
|
+
sink.onEvent(makeEvent(1));
|
|
107
|
+
sink.onEvent(makeEvent(2));
|
|
108
|
+
// Not yet flushed (batch size is 3)
|
|
109
|
+
assert.equal(fs.files.has(logPath), false);
|
|
110
|
+
sink.onEvent(makeEvent(3));
|
|
111
|
+
// Allow async flush to complete
|
|
112
|
+
await new Promise(r => setTimeout(r, 20));
|
|
113
|
+
const content = fs.files.get(logPath);
|
|
114
|
+
assert.ok(content, 'JSONL file should exist after flush');
|
|
115
|
+
const lines = content.trim().split('\n');
|
|
116
|
+
assert.equal(lines.length, 3, 'should have 3 JSON lines');
|
|
117
|
+
const parsed = JSON.parse(lines[0]);
|
|
118
|
+
assert.equal(parsed.id, 'evt-1');
|
|
119
|
+
});
|
|
120
|
+
it('flushes on timer when batch size not reached', async () => {
|
|
121
|
+
sink.onEvent(makeEvent(1));
|
|
122
|
+
// Wait for timer flush (50ms configured)
|
|
123
|
+
await new Promise(r => setTimeout(r, 100));
|
|
124
|
+
const content = fs.files.get(logPath);
|
|
125
|
+
assert.ok(content);
|
|
126
|
+
const lines = content.trim().split('\n');
|
|
127
|
+
assert.equal(lines.length, 1);
|
|
128
|
+
});
|
|
129
|
+
it('appends to existing JSONL file', async () => {
|
|
130
|
+
fs.files.set(logPath, '{"id":"old","version":1,"sequence":0}\n');
|
|
131
|
+
sink.onEvent(makeEvent(1));
|
|
132
|
+
await sink.flush();
|
|
133
|
+
const content = fs.files.get(logPath);
|
|
134
|
+
const lines = content.trim().split('\n');
|
|
135
|
+
assert.equal(lines.length, 2, 'should append, not overwrite');
|
|
136
|
+
assert.ok(lines[0].includes('"old"'));
|
|
137
|
+
assert.ok(lines[1].includes('"evt-1"'));
|
|
138
|
+
});
|
|
139
|
+
});
|
|
140
|
+
describe('cursor recovery', () => {
|
|
141
|
+
it('skips events at or below last cursor', async () => {
|
|
142
|
+
// Simulate prior cursor state
|
|
143
|
+
fs.files.set(cursorsPath, JSON.stringify({ persistence: 5 }));
|
|
144
|
+
const sinkWithCursor = new PersistenceSink({
|
|
145
|
+
fs,
|
|
146
|
+
logPath,
|
|
147
|
+
cursorsPath,
|
|
148
|
+
flushBatchSize: 1,
|
|
149
|
+
});
|
|
150
|
+
await sinkWithCursor.init();
|
|
151
|
+
// Events 1-5 should be skipped
|
|
152
|
+
sinkWithCursor.onEvent(makeEvent(3));
|
|
153
|
+
sinkWithCursor.onEvent(makeEvent(5));
|
|
154
|
+
await new Promise(r => setTimeout(r, 20));
|
|
155
|
+
assert.equal(fs.files.has(logPath), false, 'no events should be written');
|
|
156
|
+
// Event 6 should be persisted
|
|
157
|
+
sinkWithCursor.onEvent(makeEvent(6));
|
|
158
|
+
await new Promise(r => setTimeout(r, 20));
|
|
159
|
+
const content = fs.files.get(logPath);
|
|
160
|
+
assert.ok(content.includes('"evt-6"'));
|
|
161
|
+
});
|
|
162
|
+
it('saves cursor after flush', async () => {
|
|
163
|
+
sink.onEvent(makeEvent(1));
|
|
164
|
+
sink.onEvent(makeEvent(2));
|
|
165
|
+
sink.onEvent(makeEvent(3));
|
|
166
|
+
await new Promise(r => setTimeout(r, 20));
|
|
167
|
+
const cursors = JSON.parse(fs.files.get(cursorsPath));
|
|
168
|
+
assert.equal(cursors.persistence, 3);
|
|
169
|
+
});
|
|
170
|
+
it('returns empty cursors when file missing', async () => {
|
|
171
|
+
const cursors = await sink.loadCursors();
|
|
172
|
+
assert.deepEqual(cursors, {});
|
|
173
|
+
});
|
|
174
|
+
});
|
|
175
|
+
describe('replay', () => {
|
|
176
|
+
it('reads events from JSONL file', async () => {
|
|
177
|
+
const events = [makeEvent(1), makeEvent(2), makeEvent(3)];
|
|
178
|
+
fs.files.set(logPath, events.map(e => JSON.stringify(e)).join('\n') + '\n');
|
|
179
|
+
const replayed = await sink.replay();
|
|
180
|
+
assert.equal(replayed.length, 3);
|
|
181
|
+
assert.equal(replayed[0].id, 'evt-1');
|
|
182
|
+
assert.equal(replayed[2].id, 'evt-3');
|
|
183
|
+
});
|
|
184
|
+
it('filters by replay window', async () => {
|
|
185
|
+
const oldEvent = makeEvent(1, {
|
|
186
|
+
timestamp: new Date(Date.now() - 48 * 60 * 60 * 1000).toISOString(), // 48h ago
|
|
187
|
+
});
|
|
188
|
+
const recentEvent = makeEvent(2, {
|
|
189
|
+
timestamp: new Date().toISOString(),
|
|
190
|
+
});
|
|
191
|
+
fs.files.set(logPath, [oldEvent, recentEvent].map(e => JSON.stringify(e)).join('\n') + '\n');
|
|
192
|
+
const replayed = await sink.replay();
|
|
193
|
+
assert.equal(replayed.length, 1, 'only recent event within 24h window');
|
|
194
|
+
assert.equal(replayed[0].id, 'evt-2');
|
|
195
|
+
});
|
|
196
|
+
it('skips corrupt JSONL lines gracefully', async () => {
|
|
197
|
+
const goodEvent = makeEvent(1);
|
|
198
|
+
fs.files.set(logPath, `${JSON.stringify(goodEvent)}\n{corrupt json\n${JSON.stringify(makeEvent(2))}\n`);
|
|
199
|
+
const replayed = await sink.replay();
|
|
200
|
+
assert.equal(replayed.length, 2, 'should skip corrupt line and parse the rest');
|
|
201
|
+
});
|
|
202
|
+
it('returns empty array when file missing', async () => {
|
|
203
|
+
const replayed = await sink.replay();
|
|
204
|
+
assert.deepEqual(replayed, []);
|
|
205
|
+
});
|
|
206
|
+
it('returns empty array for empty file', async () => {
|
|
207
|
+
fs.files.set(logPath, '');
|
|
208
|
+
const replayed = await sink.replay();
|
|
209
|
+
assert.deepEqual(replayed, []);
|
|
210
|
+
});
|
|
211
|
+
});
|
|
212
|
+
describe('readEventsSince', () => {
|
|
213
|
+
it('returns events with sequence > sinceSeq', async () => {
|
|
214
|
+
const events = [makeEvent(1), makeEvent(2), makeEvent(3), makeEvent(4), makeEvent(5)];
|
|
215
|
+
fs.files.set(logPath, events.map(e => JSON.stringify(e)).join('\n') + '\n');
|
|
216
|
+
const result = await sink.readEventsSince(2);
|
|
217
|
+
assert.equal(result.length, 3, 'should return events 3, 4, 5');
|
|
218
|
+
assert.deepEqual(result.map(e => e.sequence), [3, 4, 5]);
|
|
219
|
+
});
|
|
220
|
+
it('returns empty array when sinceSeq >= max sequence', async () => {
|
|
221
|
+
const events = [makeEvent(1), makeEvent(2), makeEvent(3)];
|
|
222
|
+
fs.files.set(logPath, events.map(e => JSON.stringify(e)).join('\n') + '\n');
|
|
223
|
+
const result = await sink.readEventsSince(99);
|
|
224
|
+
assert.deepEqual(result, []);
|
|
225
|
+
});
|
|
226
|
+
it('returns empty array when log file missing', async () => {
|
|
227
|
+
const result = await sink.readEventsSince(0);
|
|
228
|
+
assert.deepEqual(result, []);
|
|
229
|
+
});
|
|
230
|
+
it('skips corrupt JSONL lines gracefully', async () => {
|
|
231
|
+
const goodEvent1 = makeEvent(1);
|
|
232
|
+
const goodEvent2 = makeEvent(2);
|
|
233
|
+
fs.files.set(logPath, `${JSON.stringify(goodEvent1)}\n{this is not valid json\n${JSON.stringify(goodEvent2)}\n`);
|
|
234
|
+
const result = await sink.readEventsSince(0);
|
|
235
|
+
assert.equal(result.length, 2, 'should skip corrupt line and return the rest');
|
|
236
|
+
assert.deepEqual(result.map(e => e.sequence), [1, 2]);
|
|
237
|
+
});
|
|
238
|
+
it('returns events in append (monotonic) order', async () => {
|
|
239
|
+
const events = [makeEvent(1), makeEvent(2), makeEvent(3), makeEvent(4), makeEvent(5)];
|
|
240
|
+
fs.files.set(logPath, events.map(e => JSON.stringify(e)).join('\n') + '\n');
|
|
241
|
+
const result = await sink.readEventsSince(0);
|
|
242
|
+
assert.equal(result.length, 5);
|
|
243
|
+
for (let i = 1; i < result.length; i++) {
|
|
244
|
+
assert.ok(result[i].sequence > result[i - 1].sequence, 'sequences should be monotonically increasing');
|
|
245
|
+
}
|
|
246
|
+
});
|
|
247
|
+
it('sinceSeq=0 returns all events', async () => {
|
|
248
|
+
const events = [makeEvent(1), makeEvent(2), makeEvent(3)];
|
|
249
|
+
fs.files.set(logPath, events.map(e => JSON.stringify(e)).join('\n') + '\n');
|
|
250
|
+
const result = await sink.readEventsSince(0);
|
|
251
|
+
assert.equal(result.length, 3);
|
|
252
|
+
assert.deepEqual(result.map(e => e.sequence), [1, 2, 3]);
|
|
253
|
+
});
|
|
254
|
+
it('returns empty array for empty file', async () => {
|
|
255
|
+
fs.files.set(logPath, '');
|
|
256
|
+
const result = await sink.readEventsSince(0);
|
|
257
|
+
assert.deepEqual(result, []);
|
|
258
|
+
});
|
|
259
|
+
it('ignores replay window (returns old events outside 24h)', async () => {
|
|
260
|
+
// readEventsSince uses cursor filter, NOT time window — so old events should be included.
|
|
261
|
+
const oldEvent = makeEvent(1, {
|
|
262
|
+
timestamp: new Date(Date.now() - 48 * 60 * 60 * 1000).toISOString(),
|
|
263
|
+
});
|
|
264
|
+
const recentEvent = makeEvent(2, {
|
|
265
|
+
timestamp: new Date().toISOString(),
|
|
266
|
+
});
|
|
267
|
+
fs.files.set(logPath, [oldEvent, recentEvent].map(e => JSON.stringify(e)).join('\n') + '\n');
|
|
268
|
+
const result = await sink.readEventsSince(0);
|
|
269
|
+
assert.equal(result.length, 2, 'cursor-based replay should not filter by time');
|
|
270
|
+
});
|
|
271
|
+
});
|
|
272
|
+
describe('flush failure', () => {
|
|
273
|
+
it('emits overflow callback on write error', async () => {
|
|
274
|
+
const overflows = [];
|
|
275
|
+
sink.setOverflowCallback(msg => overflows.push(msg));
|
|
276
|
+
// Make appendFile throw
|
|
277
|
+
fs.appendFile = async () => { throw new Error('disk full'); };
|
|
278
|
+
sink.onEvent(makeEvent(1));
|
|
279
|
+
sink.onEvent(makeEvent(2));
|
|
280
|
+
sink.onEvent(makeEvent(3));
|
|
281
|
+
// Wait for flush attempt
|
|
282
|
+
await new Promise(r => setTimeout(r, 20));
|
|
283
|
+
assert.equal(overflows.length, 1);
|
|
284
|
+
assert.ok(overflows[0].includes('disk full'));
|
|
285
|
+
});
|
|
286
|
+
it('retains events in buffer after flush failure', async () => {
|
|
287
|
+
// Make appendFile throw once, then succeed
|
|
288
|
+
let callCount = 0;
|
|
289
|
+
const originalAppend = fs.appendFile.bind(fs);
|
|
290
|
+
fs.appendFile = async (path, content, encoding) => {
|
|
291
|
+
callCount++;
|
|
292
|
+
if (callCount === 1)
|
|
293
|
+
throw new Error('transient');
|
|
294
|
+
return originalAppend(path, content, encoding);
|
|
295
|
+
};
|
|
296
|
+
sink.onEvent(makeEvent(1));
|
|
297
|
+
sink.onEvent(makeEvent(2));
|
|
298
|
+
sink.onEvent(makeEvent(3));
|
|
299
|
+
await new Promise(r => setTimeout(r, 20));
|
|
300
|
+
// Events should still be in buffer after first failure
|
|
301
|
+
// Trigger another flush
|
|
302
|
+
await sink.flush();
|
|
303
|
+
const content = fs.files.get(logPath);
|
|
304
|
+
assert.ok(content, 'events should be written on retry');
|
|
305
|
+
assert.equal(content.trim().split('\n').length, 3);
|
|
306
|
+
});
|
|
307
|
+
});
|
|
308
|
+
describe('dispose', () => {
|
|
309
|
+
it('flushes remaining buffer on dispose', async () => {
|
|
310
|
+
sink.onEvent(makeEvent(1));
|
|
311
|
+
// Don't wait for timer — dispose should flush immediately
|
|
312
|
+
await sink.dispose();
|
|
313
|
+
const content = fs.files.get(logPath);
|
|
314
|
+
assert.ok(content);
|
|
315
|
+
assert.equal(content.trim().split('\n').length, 1);
|
|
316
|
+
});
|
|
317
|
+
});
|
|
318
|
+
});
|
|
319
|
+
//# sourceMappingURL=persistence-sink.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"persistence-sink.test.js","sourceRoot":"","sources":["../../src/event-bus/persistence-sink.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,91 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* SessionCheckpointSink — PRD 029 Phase C-2.
|
|
3
|
+
*
|
|
4
|
+
* Persists session snapshots to the session persistence store whenever
|
|
5
|
+
* session-domain lifecycle events fire. Uses debounced writes: multiple
|
|
6
|
+
* events for the same session within a 200ms window collapse into a
|
|
7
|
+
* single save() call.
|
|
8
|
+
*
|
|
9
|
+
* Depends on two narrow callbacks (not the pool directly) to respect
|
|
10
|
+
* G-BOUNDARY. The composition root wires:
|
|
11
|
+
* - poolList(): current session status snapshots
|
|
12
|
+
* - save(): writes a PersistedSession record
|
|
13
|
+
*/
|
|
14
|
+
import type { RuntimeEvent, EventSink } from '../ports/event-bus.js';
|
|
15
|
+
/**
|
|
16
|
+
* Minimal persisted session shape accepted by the save callback.
|
|
17
|
+
*
|
|
18
|
+
* Q2 audit (S2 §14, PRD-057 §10 Q2): The only bridge-specific field was the
|
|
19
|
+
* `mode: 'pty' | 'print'` union — that narrowing is transport-specific (bridge
|
|
20
|
+
* runs PTY and print sessions; agent-runtime uses different modes). Here we
|
|
21
|
+
* keep `mode: string` in the runtime-level port and let the bridge-side save
|
|
22
|
+
* callback narrow it when persisting its own `PersistedSession` record.
|
|
23
|
+
*
|
|
24
|
+
* Every other field (`session_id`, `workdir`, `nickname`, `purpose`, `status`,
|
|
25
|
+
* `created_at`, `last_activity_at`, `prompt_count`, `depth`,
|
|
26
|
+
* `parent_session_id`, `isolation`, `metadata`) is session-generic and stays
|
|
27
|
+
* strongly typed.
|
|
28
|
+
*/
|
|
29
|
+
export interface PersistedSessionInput {
|
|
30
|
+
session_id: string;
|
|
31
|
+
workdir: string;
|
|
32
|
+
nickname: string;
|
|
33
|
+
purpose: string | null;
|
|
34
|
+
/** Session mode — opaque string at the runtime level. Bridge narrows to 'pty' | 'print'; agent-runtime may use its own modes. */
|
|
35
|
+
mode: string;
|
|
36
|
+
status: string;
|
|
37
|
+
created_at: string;
|
|
38
|
+
last_activity_at: string;
|
|
39
|
+
prompt_count: number;
|
|
40
|
+
depth: number;
|
|
41
|
+
parent_session_id: string | null;
|
|
42
|
+
isolation: string;
|
|
43
|
+
metadata?: Record<string, unknown>;
|
|
44
|
+
}
|
|
45
|
+
/**
|
|
46
|
+
* Minimal session status info returned by the poolList callback.
|
|
47
|
+
* Matches the relevant fields of SessionStatusInfo from the sessions domain.
|
|
48
|
+
*/
|
|
49
|
+
export interface SessionStatusInfo {
|
|
50
|
+
sessionId: string;
|
|
51
|
+
nickname: string;
|
|
52
|
+
purpose: string | null;
|
|
53
|
+
status: string;
|
|
54
|
+
promptCount: number;
|
|
55
|
+
lastActivityAt: Date;
|
|
56
|
+
workdir: string;
|
|
57
|
+
mode: string;
|
|
58
|
+
chain: {
|
|
59
|
+
parent_session_id: string | null;
|
|
60
|
+
depth: number;
|
|
61
|
+
};
|
|
62
|
+
worktree: {
|
|
63
|
+
isolation: string;
|
|
64
|
+
};
|
|
65
|
+
metadata?: Record<string, unknown>;
|
|
66
|
+
}
|
|
67
|
+
export interface SessionCheckpointSinkOptions {
|
|
68
|
+
/** Persist a session snapshot. */
|
|
69
|
+
save: (session: PersistedSessionInput) => Promise<void>;
|
|
70
|
+
/** Get current session statuses from the pool. */
|
|
71
|
+
poolList: () => SessionStatusInfo[];
|
|
72
|
+
/** Debounce window in milliseconds (default: 200). */
|
|
73
|
+
debounceMs?: number;
|
|
74
|
+
}
|
|
75
|
+
export declare class SessionCheckpointSink implements EventSink {
|
|
76
|
+
readonly name = "session-checkpoint";
|
|
77
|
+
private readonly save;
|
|
78
|
+
private readonly poolList;
|
|
79
|
+
private readonly debounceMs;
|
|
80
|
+
/** Pending debounce timers keyed by sessionId. */
|
|
81
|
+
private readonly pendingTimers;
|
|
82
|
+
constructor(options: SessionCheckpointSinkOptions);
|
|
83
|
+
onEvent(event: RuntimeEvent): void;
|
|
84
|
+
onError(error: Error, event: RuntimeEvent): void;
|
|
85
|
+
private checkpoint;
|
|
86
|
+
/** Clean up pending timers. Call on shutdown. */
|
|
87
|
+
dispose(): void;
|
|
88
|
+
/** Number of pending debounce timers (for testing/monitoring). */
|
|
89
|
+
get pendingCount(): number;
|
|
90
|
+
}
|
|
91
|
+
//# sourceMappingURL=session-checkpoint-sink.d.ts.map
|