reqon-dsl 0.2.0 → 0.4.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/CHANGELOG.md +7 -0
- package/README.md +45 -3
- package/dist/ast/nodes.d.ts +91 -4
- package/dist/ast/nodes.js +14 -0
- package/dist/auth/circuit-breaker.d.ts +11 -0
- package/dist/auth/circuit-breaker.js +90 -18
- package/dist/auth/credentials.d.ts +6 -1
- package/dist/auth/credentials.js +12 -4
- package/dist/auth/oauth2-provider.js +13 -3
- package/dist/auth/rate-limiter.d.ts +12 -1
- package/dist/auth/rate-limiter.js +39 -26
- package/dist/auth/token-store.js +8 -1
- package/dist/cli.d.ts +24 -1
- package/dist/cli.js +149 -10
- package/dist/config/constants.d.ts +152 -0
- package/dist/config/constants.js +139 -0
- package/dist/config/index.d.ts +4 -0
- package/dist/config/index.js +4 -0
- package/dist/control/index.d.ts +2 -0
- package/dist/control/index.js +1 -0
- package/dist/control/server.d.ts +105 -0
- package/dist/control/server.js +315 -0
- package/dist/control/types.d.ts +61 -0
- package/dist/control/types.js +7 -0
- package/dist/debug/cli-debugger.d.ts +17 -0
- package/dist/debug/cli-debugger.js +185 -0
- package/dist/debug/controller.d.ts +94 -0
- package/dist/debug/controller.js +45 -0
- package/dist/debug/index.d.ts +6 -0
- package/dist/debug/index.js +5 -0
- package/dist/errors/index.d.ts +67 -0
- package/dist/errors/index.js +89 -1
- package/dist/execution/index.d.ts +1 -1
- package/dist/execution/state.d.ts +24 -0
- package/dist/execution/store.js +2 -2
- package/dist/execution-log/events.d.ts +125 -0
- package/dist/execution-log/events.js +17 -0
- package/dist/execution-log/fold.d.ts +38 -0
- package/dist/execution-log/fold.js +54 -0
- package/dist/execution-log/index.d.ts +18 -0
- package/dist/execution-log/index.js +6 -0
- package/dist/execution-log/postgres-store.d.ts +36 -0
- package/dist/execution-log/postgres-store.js +108 -0
- package/dist/execution-log/resume.d.ts +11 -0
- package/dist/execution-log/resume.js +5 -0
- package/dist/execution-log/sqlite-store.d.ts +16 -0
- package/dist/execution-log/sqlite-store.js +101 -0
- package/dist/execution-log/store.d.ts +72 -0
- package/dist/execution-log/store.js +182 -0
- package/dist/index.d.ts +23 -2
- package/dist/index.js +35 -3
- package/dist/interpreter/context.d.ts +29 -0
- package/dist/interpreter/context.js +18 -0
- package/dist/interpreter/evaluator.d.ts +63 -1
- package/dist/interpreter/evaluator.js +219 -42
- package/dist/interpreter/executor.d.ts +132 -14
- package/dist/interpreter/executor.js +883 -178
- package/dist/interpreter/fetch-handler.d.ts +48 -1
- package/dist/interpreter/fetch-handler.js +216 -38
- package/dist/interpreter/http.d.ts +34 -0
- package/dist/interpreter/http.js +203 -28
- package/dist/interpreter/index.d.ts +5 -3
- package/dist/interpreter/index.js +4 -2
- package/dist/interpreter/pagination.d.ts +12 -3
- package/dist/interpreter/pagination.js +102 -32
- package/dist/interpreter/signals.d.ts +8 -0
- package/dist/interpreter/signals.js +12 -0
- package/dist/interpreter/source-manager.d.ts +75 -0
- package/dist/interpreter/source-manager.js +157 -0
- package/dist/interpreter/step-handlers/apply-handler.d.ts +29 -0
- package/dist/interpreter/step-handlers/apply-handler.js +79 -0
- package/dist/interpreter/step-handlers/for-handler.d.ts +16 -0
- package/dist/interpreter/step-handlers/for-handler.js +89 -7
- package/dist/interpreter/step-handlers/index.d.ts +4 -2
- package/dist/interpreter/step-handlers/index.js +4 -2
- package/dist/interpreter/step-handlers/match-handler.d.ts +9 -0
- package/dist/interpreter/step-handlers/match-handler.js +47 -17
- package/dist/interpreter/step-handlers/pause-handler.d.ts +52 -0
- package/dist/interpreter/step-handlers/pause-handler.js +87 -0
- package/dist/interpreter/step-handlers/store-handler.d.ts +17 -1
- package/dist/interpreter/step-handlers/store-handler.js +61 -20
- package/dist/interpreter/step-handlers/types.d.ts +3 -0
- package/dist/interpreter/step-handlers/validate-handler.d.ts +2 -1
- package/dist/interpreter/step-handlers/validate-handler.js +7 -2
- package/dist/interpreter/step-handlers/webhook-handler.d.ts +4 -0
- package/dist/interpreter/step-handlers/webhook-handler.js +31 -5
- package/dist/interpreter/store-manager.d.ts +46 -0
- package/dist/interpreter/store-manager.js +70 -0
- package/dist/lexer/index.d.ts +11 -4
- package/dist/lexer/index.js +11 -4
- package/dist/lexer/tokens.d.ts +17 -1
- package/dist/lexer/tokens.js +36 -0
- package/dist/loader/index.js +5 -8
- package/dist/mcp/index.d.ts +11 -0
- package/dist/mcp/index.js +11 -0
- package/dist/mcp/sandbox.d.ts +41 -0
- package/dist/mcp/sandbox.js +76 -0
- package/dist/mcp/server.d.ts +17 -0
- package/dist/mcp/server.js +504 -0
- package/dist/oas/index.d.ts +2 -0
- package/dist/oas/index.js +1 -0
- package/dist/oas/loader.d.ts +13 -1
- package/dist/oas/loader.js +25 -3
- package/dist/oas/mock-generator.d.ts +12 -0
- package/dist/oas/mock-generator.js +196 -0
- package/dist/oas/validator.js +45 -5
- package/dist/observability/events.d.ts +248 -0
- package/dist/observability/events.js +85 -0
- package/dist/observability/index.d.ts +15 -0
- package/dist/observability/index.js +12 -0
- package/dist/observability/logger.d.ts +106 -0
- package/dist/observability/logger.js +266 -0
- package/dist/observability/otel.d.ts +143 -0
- package/dist/observability/otel.js +421 -0
- package/dist/parser/action-parser.d.ts +105 -0
- package/dist/parser/action-parser.js +645 -0
- package/dist/parser/base.d.ts +7 -0
- package/dist/parser/base.js +11 -0
- package/dist/parser/expressions.d.ts +14 -0
- package/dist/parser/expressions.js +89 -6
- package/dist/parser/fetch-parser.d.ts +27 -0
- package/dist/parser/fetch-parser.js +280 -0
- package/dist/parser/index.d.ts +17 -0
- package/dist/parser/index.js +17 -0
- package/dist/parser/parser.d.ts +44 -46
- package/dist/parser/parser.js +122 -1070
- package/dist/parser/pipeline-parser.d.ts +12 -0
- package/dist/parser/pipeline-parser.js +52 -0
- package/dist/parser/schedule-parser.d.ts +7 -0
- package/dist/parser/schedule-parser.js +137 -0
- package/dist/parser/source-parser.d.ts +9 -0
- package/dist/parser/source-parser.js +151 -0
- package/dist/pause/index.d.ts +15 -0
- package/dist/pause/index.js +12 -0
- package/dist/pause/log-store.d.ts +33 -0
- package/dist/pause/log-store.js +98 -0
- package/dist/pause/manager.d.ts +130 -0
- package/dist/pause/manager.js +294 -0
- package/dist/pause/state.d.ts +93 -0
- package/dist/pause/state.js +103 -0
- package/dist/pause/store.d.ts +61 -0
- package/dist/pause/store.js +158 -0
- package/dist/plugin.d.ts +9 -12
- package/dist/plugin.js +10 -13
- package/dist/scheduler/cron-parser.d.ts +10 -3
- package/dist/scheduler/cron-parser.js +227 -48
- package/dist/scheduler/scheduler.js +56 -22
- package/dist/stores/factory.d.ts +7 -1
- package/dist/stores/factory.js +14 -3
- package/dist/stores/file.d.ts +26 -0
- package/dist/stores/file.js +67 -21
- package/dist/stores/index.d.ts +16 -1
- package/dist/stores/index.js +16 -1
- package/dist/stores/memory.d.ts +4 -0
- package/dist/stores/memory.js +8 -6
- package/dist/stores/postgrest.d.ts +28 -0
- package/dist/stores/postgrest.js +84 -37
- package/dist/stores/types.d.ts +17 -0
- package/dist/stores/types.js +12 -0
- package/dist/sync/index.d.ts +3 -2
- package/dist/sync/index.js +2 -1
- package/dist/sync/log-store.d.ts +30 -0
- package/dist/sync/log-store.js +45 -0
- package/dist/sync/store.js +1 -1
- package/dist/trace/index.d.ts +18 -0
- package/dist/trace/index.js +13 -0
- package/dist/trace/log-view.d.ts +57 -0
- package/dist/trace/log-view.js +76 -0
- package/dist/trace/recorder.d.ts +75 -0
- package/dist/trace/recorder.js +157 -0
- package/dist/trace/replay.d.ts +132 -0
- package/dist/trace/replay.js +264 -0
- package/dist/trace/state.d.ts +102 -0
- package/dist/trace/state.js +86 -0
- package/dist/trace/store.d.ts +75 -0
- package/dist/trace/store.js +250 -0
- package/dist/utils/deep-merge.d.ts +10 -0
- package/dist/utils/deep-merge.js +23 -0
- package/dist/utils/file.d.ts +13 -4
- package/dist/utils/file.js +70 -12
- package/dist/utils/index.d.ts +2 -1
- package/dist/utils/index.js +2 -1
- package/dist/utils/long-timeout.d.ts +19 -0
- package/dist/utils/long-timeout.js +33 -0
- package/dist/utils/path.d.ts +22 -1
- package/dist/utils/path.js +46 -1
- package/dist/utils/redact.d.ts +22 -0
- package/dist/utils/redact.js +42 -0
- package/dist/utils/type-guards.d.ts +58 -0
- package/dist/utils/type-guards.js +92 -0
- package/dist/webhook/server.d.ts +9 -0
- package/dist/webhook/server.js +122 -36
- package/dist/webhook/types.d.ts +9 -1
- package/package.json +76 -9
- package/.claude/settings.local.json +0 -31
- package/.claude/skills/api-integration.md +0 -125
- package/.claude/skills/database-schema.md +0 -51
- package/.claude/skills/dsl-design.md +0 -80
- package/.claude/skills/property-testing.md +0 -143
- package/.claude/skills/reqon/SKILL.md +0 -44
- package/.claude/skills/reqon/references/examples.md +0 -206
- package/.claude/skills/reqon/references/syntax.md +0 -263
- package/.claude/skills/vscode-extension.md +0 -113
- package/.github/dependabot.yml +0 -32
- package/.github/pull_request_template.md +0 -21
- package/.github/workflows/ci.yml +0 -174
- package/.github/workflows/release.yml +0 -73
- package/CLAUDE.md +0 -72
- package/CONTRIBUTING.md +0 -161
- package/TODO.md +0 -51
- package/dist/auth/auth.test.d.ts +0 -1
- package/dist/auth/auth.test.js +0 -255
- package/dist/errors/errors.test.d.ts +0 -1
- package/dist/errors/errors.test.js +0 -165
- package/dist/execution/execution.test.d.ts +0 -1
- package/dist/execution/execution.test.js +0 -246
- package/dist/integration.test.d.ts +0 -1
- package/dist/integration.test.js +0 -168
- package/dist/interpreter/evaluator.test.d.ts +0 -1
- package/dist/interpreter/evaluator.test.js +0 -512
- package/dist/interpreter/http.test.d.ts +0 -1
- package/dist/interpreter/http.test.js +0 -299
- package/dist/interpreter/progress.test.d.ts +0 -1
- package/dist/interpreter/progress.test.js +0 -216
- package/dist/interpreter/schema-matcher.test.d.ts +0 -1
- package/dist/interpreter/schema-matcher.test.js +0 -122
- package/dist/lexer/lexer.d.ts +0 -24
- package/dist/lexer/lexer.js +0 -264
- package/dist/lexer/lexer.test.d.ts +0 -1
- package/dist/lexer/lexer.test.js +0 -259
- package/dist/loader/loader.test.d.ts +0 -1
- package/dist/loader/loader.test.js +0 -287
- package/dist/oas/oas.test.d.ts +0 -1
- package/dist/oas/oas.test.js +0 -218
- package/dist/parser/expressions.test.d.ts +0 -1
- package/dist/parser/expressions.test.js +0 -378
- package/dist/parser/match.test.d.ts +0 -1
- package/dist/parser/match.test.js +0 -254
- package/dist/parser/parser.test.d.ts +0 -1
- package/dist/parser/parser.test.js +0 -333
- package/dist/parser/schedule.test.d.ts +0 -1
- package/dist/parser/schedule.test.js +0 -241
- package/dist/scheduler/cron-parser.test.d.ts +0 -1
- package/dist/scheduler/cron-parser.test.js +0 -188
- package/dist/stores/file.test.d.ts +0 -1
- package/dist/stores/file.test.js +0 -165
- package/dist/stores/memory.test.d.ts +0 -1
- package/dist/stores/memory.test.js +0 -157
- package/dist/stores/stores.test.d.ts +0 -1
- package/dist/stores/stores.test.js +0 -158
- package/dist/sync/sync.test.d.ts +0 -1
- package/dist/sync/sync.test.js +0 -221
- package/docusaurus/README.md +0 -41
- package/docusaurus/docs/advanced/execution-state.md +0 -283
- package/docusaurus/docs/advanced/extending-reqon.md +0 -388
- package/docusaurus/docs/advanced/multi-file-missions.md +0 -250
- package/docusaurus/docs/advanced/parallel-execution.md +0 -353
- package/docusaurus/docs/api-reference.md +0 -443
- package/docusaurus/docs/authentication/api-key.md +0 -339
- package/docusaurus/docs/authentication/basic.md +0 -276
- package/docusaurus/docs/authentication/bearer.md +0 -282
- package/docusaurus/docs/authentication/oauth2.md +0 -317
- package/docusaurus/docs/authentication/overview.md +0 -251
- package/docusaurus/docs/cli.md +0 -229
- package/docusaurus/docs/core-concepts/actions.md +0 -286
- package/docusaurus/docs/core-concepts/missions.md +0 -264
- package/docusaurus/docs/core-concepts/schemas.md +0 -353
- package/docusaurus/docs/core-concepts/sources.md +0 -339
- package/docusaurus/docs/core-concepts/stores.md +0 -332
- package/docusaurus/docs/dsl-syntax/expressions.md +0 -361
- package/docusaurus/docs/dsl-syntax/fetch.md +0 -293
- package/docusaurus/docs/dsl-syntax/for-loops.md +0 -324
- package/docusaurus/docs/dsl-syntax/map.md +0 -345
- package/docusaurus/docs/dsl-syntax/match.md +0 -387
- package/docusaurus/docs/dsl-syntax/pipelines.md +0 -397
- package/docusaurus/docs/dsl-syntax/validate.md +0 -401
- package/docusaurus/docs/error-handling/dead-letter-queues.md +0 -399
- package/docusaurus/docs/error-handling/flow-control.md +0 -337
- package/docusaurus/docs/error-handling/retry-strategies.md +0 -368
- package/docusaurus/docs/examples.md +0 -488
- package/docusaurus/docs/getting-started.md +0 -256
- package/docusaurus/docs/http/circuit-breaker.md +0 -401
- package/docusaurus/docs/http/incremental-sync.md +0 -394
- package/docusaurus/docs/http/pagination.md +0 -361
- package/docusaurus/docs/http/rate-limiting.md +0 -383
- package/docusaurus/docs/http/requests.md +0 -328
- package/docusaurus/docs/http/retry.md +0 -402
- package/docusaurus/docs/intro.md +0 -90
- package/docusaurus/docs/openapi/loading-specs.md +0 -305
- package/docusaurus/docs/openapi/operation-calls.md +0 -314
- package/docusaurus/docs/openapi/overview.md +0 -212
- package/docusaurus/docs/openapi/response-validation.md +0 -344
- package/docusaurus/docs/scheduling/cron.md +0 -305
- package/docusaurus/docs/scheduling/daemon-mode.md +0 -317
- package/docusaurus/docs/scheduling/intervals.md +0 -289
- package/docusaurus/docs/scheduling/overview.md +0 -231
- package/docusaurus/docs/stores/custom-adapters.md +0 -376
- package/docusaurus/docs/stores/file.md +0 -236
- package/docusaurus/docs/stores/memory.md +0 -193
- package/docusaurus/docs/stores/overview.md +0 -274
- package/docusaurus/docs/stores/postgrest.md +0 -316
- package/docusaurus/docusaurus.config.ts +0 -148
- package/docusaurus/package-lock.json +0 -18029
- package/docusaurus/package.json +0 -47
- package/docusaurus/sidebars.ts +0 -155
- package/docusaurus/src/components/HomepageFeatures/index.tsx +0 -105
- package/docusaurus/src/components/HomepageFeatures/styles.module.css +0 -12
- package/docusaurus/src/css/custom.css +0 -169
- package/docusaurus/src/pages/index.module.css +0 -48
- package/docusaurus/src/pages/index.tsx +0 -110
- package/docusaurus/src/pages/markdown-page.md +0 -7
- package/docusaurus/static/.nojekyll +0 -0
- package/docusaurus/static/img/docusaurus-social-card.jpg +0 -0
- package/docusaurus/static/img/docusaurus.png +0 -0
- package/docusaurus/static/img/favicon.ico +0 -0
- package/docusaurus/static/img/logo.svg +0 -10
- package/docusaurus/static/img/undraw_docusaurus_mountain.svg +0 -171
- package/docusaurus/static/img/undraw_docusaurus_react.svg +0 -170
- package/docusaurus/static/img/undraw_docusaurus_tree.svg +0 -40
- package/docusaurus/tsconfig.json +0 -8
- package/examples/README.md +0 -112
- package/examples/error-handling/README.md +0 -150
- package/examples/error-handling/payment-processor.vague +0 -287
- package/examples/github-sync/README.md +0 -74
- package/examples/github-sync/fetch-issues.vague +0 -47
- package/examples/github-sync/fetch-prs.vague +0 -40
- package/examples/github-sync/mission.vague +0 -101
- package/examples/github-sync/normalize.vague +0 -70
- package/examples/jsonplaceholder/README.md +0 -28
- package/examples/jsonplaceholder/posts.vague +0 -48
- package/examples/petstore/README.md +0 -35
- package/examples/petstore/openapi.yaml +0 -97
- package/examples/petstore/sync.vague +0 -52
- package/examples/temporal-comparison/README.md +0 -297
- package/examples/temporal-comparison/reconciliation.vague +0 -355
- package/examples/temporal-comparison/temporal/activities/index.ts +0 -8
- package/examples/temporal-comparison/temporal/activities/shipstation.ts +0 -225
- package/examples/temporal-comparison/temporal/activities/shopify.ts +0 -257
- package/examples/temporal-comparison/temporal/activities/storage.ts +0 -198
- package/examples/temporal-comparison/temporal/activities/stripe.ts +0 -169
- package/examples/temporal-comparison/temporal/activities/validation.ts +0 -205
- package/examples/temporal-comparison/temporal/client/schedule.ts +0 -218
- package/examples/temporal-comparison/temporal/config/retry.ts +0 -63
- package/examples/temporal-comparison/temporal/types/index.ts +0 -129
- package/examples/temporal-comparison/temporal/workers/main.ts +0 -130
- package/examples/temporal-comparison/temporal/workflows/orderReconciliation.ts +0 -262
- package/examples/xero/README.md +0 -88
- package/examples/xero/invoices.vague +0 -189
- package/src/api-integration.test.ts +0 -954
- package/src/ast/index.ts +0 -1
- package/src/ast/nodes.ts +0 -310
- package/src/auth/auth.test.ts +0 -326
- package/src/auth/circuit-breaker.test.ts +0 -390
- package/src/auth/circuit-breaker.ts +0 -379
- package/src/auth/credentials.test.ts +0 -273
- package/src/auth/credentials.ts +0 -246
- package/src/auth/index.ts +0 -40
- package/src/auth/oauth2-provider.ts +0 -177
- package/src/auth/rate-limiter.ts +0 -459
- package/src/auth/token-store.ts +0 -177
- package/src/auth/types.ts +0 -159
- package/src/benchmark/e2e.bench.ts +0 -288
- package/src/benchmark/evaluator.bench.ts +0 -331
- package/src/benchmark/fixtures.ts +0 -295
- package/src/benchmark/index.ts +0 -108
- package/src/benchmark/lexer.bench.ts +0 -69
- package/src/benchmark/parser.bench.ts +0 -103
- package/src/benchmark/resilience.bench.ts +0 -193
- package/src/benchmark/store.bench.ts +0 -147
- package/src/benchmark/utils.ts +0 -230
- package/src/cli.ts +0 -313
- package/src/errors/errors.test.ts +0 -234
- package/src/errors/index.ts +0 -223
- package/src/execution/execution.test.ts +0 -307
- package/src/execution/index.ts +0 -21
- package/src/execution/state.ts +0 -207
- package/src/execution/store.ts +0 -188
- package/src/index.ts +0 -169
- package/src/integration.test.ts +0 -192
- package/src/interpreter/context.ts +0 -57
- package/src/interpreter/evaluator.test.ts +0 -796
- package/src/interpreter/evaluator.ts +0 -245
- package/src/interpreter/executor.ts +0 -946
- package/src/interpreter/fetch-handler.ts +0 -302
- package/src/interpreter/http.test.ts +0 -423
- package/src/interpreter/http.ts +0 -308
- package/src/interpreter/index.ts +0 -32
- package/src/interpreter/pagination.ts +0 -207
- package/src/interpreter/progress.test.ts +0 -276
- package/src/interpreter/schema-matcher.test.ts +0 -160
- package/src/interpreter/schema-matcher.ts +0 -168
- package/src/interpreter/signals.ts +0 -73
- package/src/interpreter/step-handlers/for-handler.ts +0 -65
- package/src/interpreter/step-handlers/index.ts +0 -17
- package/src/interpreter/step-handlers/map-handler.ts +0 -24
- package/src/interpreter/step-handlers/match-handler.ts +0 -101
- package/src/interpreter/step-handlers/store-handler.ts +0 -78
- package/src/interpreter/step-handlers/types.ts +0 -17
- package/src/interpreter/step-handlers/validate-handler.ts +0 -30
- package/src/interpreter/step-handlers/webhook-handler.ts +0 -142
- package/src/lexer/index.ts +0 -18
- package/src/lexer/lexer.test.ts +0 -316
- package/src/lexer/tokens.ts +0 -179
- package/src/loader/index.ts +0 -288
- package/src/loader/loader.test.ts +0 -360
- package/src/oas/index.ts +0 -4
- package/src/oas/loader.ts +0 -126
- package/src/oas/oas.test.ts +0 -254
- package/src/oas/validator.ts +0 -299
- package/src/parser/base.ts +0 -124
- package/src/parser/expressions.test.ts +0 -525
- package/src/parser/expressions.ts +0 -314
- package/src/parser/index.ts +0 -3
- package/src/parser/match.test.ts +0 -296
- package/src/parser/parser.test.ts +0 -739
- package/src/parser/parser.ts +0 -1469
- package/src/parser/schedule.test.ts +0 -287
- package/src/parser/webhook.test.ts +0 -248
- package/src/plugin.ts +0 -83
- package/src/scheduler/cron-parser.test.ts +0 -236
- package/src/scheduler/cron-parser.ts +0 -236
- package/src/scheduler/index.ts +0 -10
- package/src/scheduler/scheduler.ts +0 -443
- package/src/scheduler/types.ts +0 -71
- package/src/stores/factory.ts +0 -104
- package/src/stores/file.test.ts +0 -276
- package/src/stores/file.ts +0 -211
- package/src/stores/index.ts +0 -6
- package/src/stores/memory.test.ts +0 -238
- package/src/stores/memory.ts +0 -63
- package/src/stores/postgrest.test.ts +0 -488
- package/src/stores/postgrest.ts +0 -263
- package/src/stores/stores.test.ts +0 -197
- package/src/stores/types.ts +0 -58
- package/src/sync/index.ts +0 -16
- package/src/sync/state.ts +0 -126
- package/src/sync/store.ts +0 -139
- package/src/sync/sync.test.ts +0 -271
- package/src/utils/async.ts +0 -10
- package/src/utils/file.ts +0 -106
- package/src/utils/index.ts +0 -14
- package/src/utils/logger.ts +0 -53
- package/src/utils/path.ts +0 -47
- package/src/webhook/index.ts +0 -15
- package/src/webhook/server.test.ts +0 -253
- package/src/webhook/server.ts +0 -389
- package/src/webhook/store.ts +0 -239
- package/src/webhook/types.ts +0 -93
- package/tsconfig.json +0 -17
- package/vitest.config.ts +0 -39
package/dist/auth/auth.test.js
DELETED
|
@@ -1,255 +0,0 @@
|
|
|
1
|
-
import { describe, it, expect, beforeEach, vi } from 'vitest';
|
|
2
|
-
import { AdaptiveRateLimiter, parseRateLimitHeaders, RateLimitError, RateLimitTimeoutError, } from './rate-limiter.js';
|
|
3
|
-
import { InMemoryTokenStore } from './token-store.js';
|
|
4
|
-
describe('parseRateLimitHeaders', () => {
|
|
5
|
-
it('parses X-RateLimit headers', () => {
|
|
6
|
-
const headers = {
|
|
7
|
-
'X-RateLimit-Limit': '100',
|
|
8
|
-
'X-RateLimit-Remaining': '42',
|
|
9
|
-
'X-RateLimit-Reset': '1700000000',
|
|
10
|
-
};
|
|
11
|
-
const info = parseRateLimitHeaders(headers);
|
|
12
|
-
expect(info.limit).toBe(100);
|
|
13
|
-
expect(info.remaining).toBe(42);
|
|
14
|
-
expect(info.resetAt).toBeInstanceOf(Date);
|
|
15
|
-
expect(info.resetAt?.getTime()).toBe(1700000000 * 1000);
|
|
16
|
-
});
|
|
17
|
-
it('parses lowercase ratelimit headers', () => {
|
|
18
|
-
const headers = {
|
|
19
|
-
'ratelimit-limit': '60',
|
|
20
|
-
'ratelimit-remaining': '5',
|
|
21
|
-
};
|
|
22
|
-
const info = parseRateLimitHeaders(headers);
|
|
23
|
-
expect(info.limit).toBe(60);
|
|
24
|
-
expect(info.remaining).toBe(5);
|
|
25
|
-
});
|
|
26
|
-
it('parses Retry-After header (seconds)', () => {
|
|
27
|
-
const headers = {
|
|
28
|
-
'Retry-After': '30',
|
|
29
|
-
};
|
|
30
|
-
const info = parseRateLimitHeaders(headers);
|
|
31
|
-
expect(info.retryAfter).toBe(30);
|
|
32
|
-
});
|
|
33
|
-
it('handles missing headers gracefully', () => {
|
|
34
|
-
const info = parseRateLimitHeaders({});
|
|
35
|
-
expect(info.limit).toBeUndefined();
|
|
36
|
-
expect(info.remaining).toBeUndefined();
|
|
37
|
-
expect(info.resetAt).toBeUndefined();
|
|
38
|
-
expect(info.retryAfter).toBeUndefined();
|
|
39
|
-
});
|
|
40
|
-
});
|
|
41
|
-
describe('AdaptiveRateLimiter', () => {
|
|
42
|
-
let limiter;
|
|
43
|
-
beforeEach(() => {
|
|
44
|
-
limiter = new AdaptiveRateLimiter();
|
|
45
|
-
});
|
|
46
|
-
it('allows requests when no limits recorded', async () => {
|
|
47
|
-
const canProceed = await limiter.canProceed('TestAPI');
|
|
48
|
-
expect(canProceed).toBe(true);
|
|
49
|
-
});
|
|
50
|
-
it('blocks requests when remaining is 0', async () => {
|
|
51
|
-
const futureReset = new Date(Date.now() + 60000);
|
|
52
|
-
limiter.recordResponse('TestAPI', {
|
|
53
|
-
remaining: 0,
|
|
54
|
-
limit: 100,
|
|
55
|
-
resetAt: futureReset,
|
|
56
|
-
});
|
|
57
|
-
const canProceed = await limiter.canProceed('TestAPI');
|
|
58
|
-
expect(canProceed).toBe(false);
|
|
59
|
-
});
|
|
60
|
-
it('allows requests after reset time passes', async () => {
|
|
61
|
-
const pastReset = new Date(Date.now() - 1000);
|
|
62
|
-
limiter.recordResponse('TestAPI', {
|
|
63
|
-
remaining: 0,
|
|
64
|
-
limit: 100,
|
|
65
|
-
resetAt: pastReset,
|
|
66
|
-
});
|
|
67
|
-
const canProceed = await limiter.canProceed('TestAPI');
|
|
68
|
-
expect(canProceed).toBe(true);
|
|
69
|
-
});
|
|
70
|
-
it('blocks during retry-after period', async () => {
|
|
71
|
-
limiter.recordResponse('TestAPI', {
|
|
72
|
-
retryAfter: 60, // 60 seconds
|
|
73
|
-
});
|
|
74
|
-
const canProceed = await limiter.canProceed('TestAPI');
|
|
75
|
-
expect(canProceed).toBe(false);
|
|
76
|
-
});
|
|
77
|
-
it('tracks limits per source', async () => {
|
|
78
|
-
limiter.recordResponse('API1', { remaining: 0, resetAt: new Date(Date.now() + 60000) });
|
|
79
|
-
limiter.recordResponse('API2', { remaining: 50, limit: 100 });
|
|
80
|
-
expect(await limiter.canProceed('API1')).toBe(false);
|
|
81
|
-
expect(await limiter.canProceed('API2')).toBe(true);
|
|
82
|
-
});
|
|
83
|
-
it('tracks limits per endpoint within source', async () => {
|
|
84
|
-
limiter.recordResponse('API', { remaining: 0, resetAt: new Date(Date.now() + 60000) }, '/invoices');
|
|
85
|
-
limiter.recordResponse('API', { remaining: 50, limit: 100 }, '/contacts');
|
|
86
|
-
expect(await limiter.canProceed('API', '/invoices')).toBe(false);
|
|
87
|
-
expect(await limiter.canProceed('API', '/contacts')).toBe(true);
|
|
88
|
-
});
|
|
89
|
-
it('provides accurate status', () => {
|
|
90
|
-
limiter.recordResponse('TestAPI', {
|
|
91
|
-
remaining: 42,
|
|
92
|
-
limit: 100,
|
|
93
|
-
resetAt: new Date(Date.now() + 60000),
|
|
94
|
-
});
|
|
95
|
-
const status = limiter.getStatus('TestAPI');
|
|
96
|
-
expect(status.remaining).toBe(42);
|
|
97
|
-
expect(status.limit).toBe(100);
|
|
98
|
-
expect(status.isLimited).toBe(false);
|
|
99
|
-
});
|
|
100
|
-
describe('fail strategy', () => {
|
|
101
|
-
it('throws RateLimitError immediately when rate limited', async () => {
|
|
102
|
-
limiter.configure('FailAPI', { strategy: 'fail' });
|
|
103
|
-
limiter.recordResponse('FailAPI', {
|
|
104
|
-
remaining: 0,
|
|
105
|
-
resetAt: new Date(Date.now() + 60000),
|
|
106
|
-
});
|
|
107
|
-
await expect(limiter.waitForCapacity('FailAPI')).rejects.toThrow(RateLimitError);
|
|
108
|
-
});
|
|
109
|
-
it('includes reset time in error', async () => {
|
|
110
|
-
limiter.configure('FailAPI', { strategy: 'fail' });
|
|
111
|
-
const resetAt = new Date(Date.now() + 60000);
|
|
112
|
-
limiter.recordResponse('FailAPI', { remaining: 0, resetAt });
|
|
113
|
-
try {
|
|
114
|
-
await limiter.waitForCapacity('FailAPI');
|
|
115
|
-
expect.fail('Should have thrown');
|
|
116
|
-
}
|
|
117
|
-
catch (error) {
|
|
118
|
-
expect(error).toBeInstanceOf(RateLimitError);
|
|
119
|
-
expect(error.resetAt).toEqual(resetAt);
|
|
120
|
-
}
|
|
121
|
-
});
|
|
122
|
-
});
|
|
123
|
-
describe('pause strategy', () => {
|
|
124
|
-
it('throws RateLimitTimeoutError when maxWait exceeded', async () => {
|
|
125
|
-
limiter.configure('PauseAPI', { strategy: 'pause', maxWait: 1 });
|
|
126
|
-
limiter.recordResponse('PauseAPI', {
|
|
127
|
-
remaining: 0,
|
|
128
|
-
resetAt: new Date(Date.now() + 60000), // 60s in future
|
|
129
|
-
});
|
|
130
|
-
await expect(limiter.waitForCapacity('PauseAPI')).rejects.toThrow(RateLimitTimeoutError);
|
|
131
|
-
});
|
|
132
|
-
it('proceeds after reset time passes', async () => {
|
|
133
|
-
limiter.configure('PauseAPI', { strategy: 'pause', maxWait: 5 });
|
|
134
|
-
// Reset in 50ms
|
|
135
|
-
limiter.recordResponse('PauseAPI', {
|
|
136
|
-
remaining: 0,
|
|
137
|
-
resetAt: new Date(Date.now() + 50),
|
|
138
|
-
});
|
|
139
|
-
// Should complete without throwing
|
|
140
|
-
await limiter.waitForCapacity('PauseAPI');
|
|
141
|
-
});
|
|
142
|
-
});
|
|
143
|
-
describe('throttle strategy', () => {
|
|
144
|
-
it('calculates delay based on remaining requests and reset time', () => {
|
|
145
|
-
limiter.configure('ThrottleAPI', { strategy: 'throttle' });
|
|
146
|
-
limiter.recordResponse('ThrottleAPI', {
|
|
147
|
-
remaining: 10,
|
|
148
|
-
limit: 100,
|
|
149
|
-
resetAt: new Date(Date.now() + 10000), // 10s left
|
|
150
|
-
});
|
|
151
|
-
const delay = limiter.getThrottleDelay('ThrottleAPI');
|
|
152
|
-
// 10s / 10 remaining = 1s per request = 1000ms
|
|
153
|
-
expect(delay).toBeGreaterThanOrEqual(900);
|
|
154
|
-
expect(delay).toBeLessThanOrEqual(1100);
|
|
155
|
-
});
|
|
156
|
-
it('returns 0 delay when not in throttle mode', () => {
|
|
157
|
-
limiter.configure('PauseAPI', { strategy: 'pause' });
|
|
158
|
-
limiter.recordResponse('PauseAPI', { remaining: 10, limit: 100 });
|
|
159
|
-
const delay = limiter.getThrottleDelay('PauseAPI');
|
|
160
|
-
expect(delay).toBe(0);
|
|
161
|
-
});
|
|
162
|
-
it('uses fallback RPM when no rate limit headers', () => {
|
|
163
|
-
limiter.configure('FallbackAPI', { strategy: 'throttle', fallbackRpm: 60 });
|
|
164
|
-
limiter.recordResponse('FallbackAPI', {}); // No rate limit info
|
|
165
|
-
const delay = limiter.getThrottleDelay('FallbackAPI');
|
|
166
|
-
// 60 RPM = 1 per second = 1000ms intervals
|
|
167
|
-
expect(delay).toBeLessThanOrEqual(1000);
|
|
168
|
-
});
|
|
169
|
-
});
|
|
170
|
-
describe('callbacks', () => {
|
|
171
|
-
it('calls onRateLimited when rate limited', async () => {
|
|
172
|
-
const onRateLimited = vi.fn();
|
|
173
|
-
limiter.setCallbacks({ onRateLimited });
|
|
174
|
-
// Use maxWait of 10s so it doesn't timeout immediately (wait is only 50ms)
|
|
175
|
-
limiter.configure('CallbackAPI', { strategy: 'pause', maxWait: 10 });
|
|
176
|
-
limiter.recordResponse('CallbackAPI', {
|
|
177
|
-
remaining: 0,
|
|
178
|
-
resetAt: new Date(Date.now() + 50), // Resets in 50ms
|
|
179
|
-
});
|
|
180
|
-
await limiter.waitForCapacity('CallbackAPI');
|
|
181
|
-
expect(onRateLimited).toHaveBeenCalledTimes(1);
|
|
182
|
-
expect(onRateLimited).toHaveBeenCalledWith(expect.objectContaining({
|
|
183
|
-
source: 'CallbackAPI',
|
|
184
|
-
strategy: 'pause',
|
|
185
|
-
}));
|
|
186
|
-
});
|
|
187
|
-
it('calls onResumed after waiting completes', async () => {
|
|
188
|
-
const onResumed = vi.fn();
|
|
189
|
-
limiter.setCallbacks({ onResumed });
|
|
190
|
-
limiter.configure('ResumeAPI', { strategy: 'pause', maxWait: 5 });
|
|
191
|
-
// Reset in 50ms
|
|
192
|
-
limiter.recordResponse('ResumeAPI', {
|
|
193
|
-
remaining: 0,
|
|
194
|
-
resetAt: new Date(Date.now() + 50),
|
|
195
|
-
});
|
|
196
|
-
await limiter.waitForCapacity('ResumeAPI');
|
|
197
|
-
expect(onResumed).toHaveBeenCalledTimes(1);
|
|
198
|
-
expect(onResumed).toHaveBeenCalledWith(expect.objectContaining({
|
|
199
|
-
source: 'ResumeAPI',
|
|
200
|
-
}));
|
|
201
|
-
});
|
|
202
|
-
});
|
|
203
|
-
});
|
|
204
|
-
describe('InMemoryTokenStore', () => {
|
|
205
|
-
let store;
|
|
206
|
-
beforeEach(() => {
|
|
207
|
-
store = new InMemoryTokenStore();
|
|
208
|
-
});
|
|
209
|
-
it('stores and retrieves tokens', async () => {
|
|
210
|
-
const tokens = {
|
|
211
|
-
accessToken: 'access123',
|
|
212
|
-
refreshToken: 'refresh456',
|
|
213
|
-
expiresAt: new Date(Date.now() + 3600000),
|
|
214
|
-
};
|
|
215
|
-
await store.set('connection-1', tokens);
|
|
216
|
-
const retrieved = await store.get('connection-1');
|
|
217
|
-
expect(retrieved?.accessToken).toBe('access123');
|
|
218
|
-
expect(retrieved?.refreshToken).toBe('refresh456');
|
|
219
|
-
});
|
|
220
|
-
it('returns null for unknown connections', async () => {
|
|
221
|
-
const result = await store.get('unknown');
|
|
222
|
-
expect(result).toBeNull();
|
|
223
|
-
});
|
|
224
|
-
it('deletes tokens', async () => {
|
|
225
|
-
await store.set('connection-1', { accessToken: 'test' });
|
|
226
|
-
await store.delete('connection-1');
|
|
227
|
-
const result = await store.get('connection-1');
|
|
228
|
-
expect(result).toBeNull();
|
|
229
|
-
});
|
|
230
|
-
it('lists all connections', async () => {
|
|
231
|
-
await store.set('conn-1', { accessToken: 'a' });
|
|
232
|
-
await store.set('conn-2', { accessToken: 'b' });
|
|
233
|
-
await store.set('conn-3', { accessToken: 'c' });
|
|
234
|
-
const connections = await store.list();
|
|
235
|
-
expect(connections).toHaveLength(3);
|
|
236
|
-
expect(connections).toContain('conn-1');
|
|
237
|
-
expect(connections).toContain('conn-2');
|
|
238
|
-
expect(connections).toContain('conn-3');
|
|
239
|
-
});
|
|
240
|
-
it('identifies tokens needing refresh', async () => {
|
|
241
|
-
// Token expiring in 10 seconds (within 5 min buffer)
|
|
242
|
-
await store.set('expiring-soon', {
|
|
243
|
-
accessToken: 'test',
|
|
244
|
-
expiresAt: new Date(Date.now() + 10000),
|
|
245
|
-
});
|
|
246
|
-
// Token not expiring soon
|
|
247
|
-
await store.set('valid', {
|
|
248
|
-
accessToken: 'test',
|
|
249
|
-
expiresAt: new Date(Date.now() + 3600000),
|
|
250
|
-
});
|
|
251
|
-
const needsRefresh = await store.getTokensNeedingRefresh(300);
|
|
252
|
-
expect(needsRefresh).toContain('expiring-soon');
|
|
253
|
-
expect(needsRefresh).not.toContain('valid');
|
|
254
|
-
});
|
|
255
|
-
});
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
export {};
|
|
@@ -1,165 +0,0 @@
|
|
|
1
|
-
import { describe, it, expect } from 'vitest';
|
|
2
|
-
import { ReqonError, ParseError, LexerError, RuntimeError, ValidationError, formatErrors, getSourceLine, getSourceContext, } from './index.js';
|
|
3
|
-
import { ReqonLexer } from '../lexer/index.js';
|
|
4
|
-
import { ReqonParser } from '../parser/index.js';
|
|
5
|
-
describe('Error classes', () => {
|
|
6
|
-
describe('ReqonError', () => {
|
|
7
|
-
it('should format error with location', () => {
|
|
8
|
-
const error = new ReqonError('Something went wrong', { line: 5, column: 10 });
|
|
9
|
-
const formatted = error.format();
|
|
10
|
-
expect(formatted).toContain('ReqonError: Something went wrong');
|
|
11
|
-
expect(formatted).toContain('5:10');
|
|
12
|
-
});
|
|
13
|
-
it('should format error with source context', () => {
|
|
14
|
-
const source = `mission Test {
|
|
15
|
-
source API {
|
|
16
|
-
auth: bearer,
|
|
17
|
-
base: "https://api.example.com"
|
|
18
|
-
}
|
|
19
|
-
}`;
|
|
20
|
-
const error = new ReqonError('Unexpected token', { line: 3, column: 5 }, { source });
|
|
21
|
-
const formatted = error.format();
|
|
22
|
-
expect(formatted).toContain('auth: bearer,');
|
|
23
|
-
expect(formatted).toContain('3 |');
|
|
24
|
-
expect(formatted).toContain('^');
|
|
25
|
-
});
|
|
26
|
-
it('should include file path when provided', () => {
|
|
27
|
-
const error = new ReqonError('Test error', { line: 1, column: 1 }, { source: 'test', filePath: '/path/to/file.reqon' });
|
|
28
|
-
const formatted = error.format();
|
|
29
|
-
expect(formatted).toContain('/path/to/file.reqon:');
|
|
30
|
-
});
|
|
31
|
-
});
|
|
32
|
-
describe('ParseError', () => {
|
|
33
|
-
it('should include token value in format', () => {
|
|
34
|
-
const error = new ParseError("Expected '{'", { line: 2, column: 15 }, undefined, 'identifier');
|
|
35
|
-
const formatted = error.format();
|
|
36
|
-
expect(formatted).toContain("found: 'identifier'");
|
|
37
|
-
});
|
|
38
|
-
it('should format with source context and pointer', () => {
|
|
39
|
-
const source = `mission Test {
|
|
40
|
-
source API auth: bearer
|
|
41
|
-
}`;
|
|
42
|
-
const error = new ParseError("Expected '{'", { line: 2, column: 14 }, { source }, 'auth');
|
|
43
|
-
const formatted = error.format();
|
|
44
|
-
expect(formatted).toContain('ParseError');
|
|
45
|
-
expect(formatted).toContain('2:14');
|
|
46
|
-
expect(formatted).toContain('source API auth: bearer');
|
|
47
|
-
expect(formatted).toContain('^');
|
|
48
|
-
});
|
|
49
|
-
});
|
|
50
|
-
describe('LexerError', () => {
|
|
51
|
-
it('should format with source context', () => {
|
|
52
|
-
const source = 'mission Test { @ }';
|
|
53
|
-
const error = new LexerError("Unexpected character '@'", { line: 1, column: 16 }, { source });
|
|
54
|
-
const formatted = error.format();
|
|
55
|
-
expect(formatted).toContain('LexerError');
|
|
56
|
-
expect(formatted).toContain("Unexpected character '@'");
|
|
57
|
-
});
|
|
58
|
-
});
|
|
59
|
-
describe('RuntimeError', () => {
|
|
60
|
-
it('should include action and step info', () => {
|
|
61
|
-
const error = new RuntimeError('Request failed with status 404', { line: 5, column: 3 }, undefined, { action: 'FetchData', stepType: 'fetch' });
|
|
62
|
-
const formatted = error.format();
|
|
63
|
-
expect(formatted).toContain('in action: FetchData');
|
|
64
|
-
expect(formatted).toContain('at step: fetch');
|
|
65
|
-
});
|
|
66
|
-
});
|
|
67
|
-
describe('ValidationError', () => {
|
|
68
|
-
it('should include severity', () => {
|
|
69
|
-
const error = new ValidationError('Value must be positive', { line: 10, column: 5 }, undefined, { severity: 'warning' });
|
|
70
|
-
expect(error.severity).toBe('warning');
|
|
71
|
-
});
|
|
72
|
-
});
|
|
73
|
-
});
|
|
74
|
-
describe('formatErrors', () => {
|
|
75
|
-
it('should format multiple errors', () => {
|
|
76
|
-
const errors = [
|
|
77
|
-
new ParseError('Error 1', { line: 1, column: 1 }),
|
|
78
|
-
new ParseError('Error 2', { line: 5, column: 10 }),
|
|
79
|
-
];
|
|
80
|
-
const formatted = formatErrors(errors);
|
|
81
|
-
expect(formatted).toContain('Error 1');
|
|
82
|
-
expect(formatted).toContain('Error 2');
|
|
83
|
-
expect(formatted).toContain('1:1');
|
|
84
|
-
expect(formatted).toContain('5:10');
|
|
85
|
-
});
|
|
86
|
-
});
|
|
87
|
-
describe('getSourceLine', () => {
|
|
88
|
-
it('should return the correct line', () => {
|
|
89
|
-
const source = 'line1\nline2\nline3';
|
|
90
|
-
expect(getSourceLine(source, 1)).toBe('line1');
|
|
91
|
-
expect(getSourceLine(source, 2)).toBe('line2');
|
|
92
|
-
expect(getSourceLine(source, 3)).toBe('line3');
|
|
93
|
-
});
|
|
94
|
-
it('should return undefined for out of range', () => {
|
|
95
|
-
const source = 'line1\nline2';
|
|
96
|
-
expect(getSourceLine(source, 5)).toBeUndefined();
|
|
97
|
-
});
|
|
98
|
-
});
|
|
99
|
-
describe('getSourceContext', () => {
|
|
100
|
-
it('should return surrounding lines', () => {
|
|
101
|
-
const source = 'line1\nline2\nline3\nline4\nline5';
|
|
102
|
-
const context = getSourceContext(source, 3, 1);
|
|
103
|
-
expect(context.lines).toHaveLength(3);
|
|
104
|
-
expect(context.lines[0]).toEqual({ num: 2, text: 'line2' });
|
|
105
|
-
expect(context.lines[1]).toEqual({ num: 3, text: 'line3' });
|
|
106
|
-
expect(context.lines[2]).toEqual({ num: 4, text: 'line4' });
|
|
107
|
-
expect(context.errorLineIndex).toBe(1);
|
|
108
|
-
});
|
|
109
|
-
});
|
|
110
|
-
describe('Integration with parser', () => {
|
|
111
|
-
it('should produce ParseError with source context', () => {
|
|
112
|
-
const source = `mission Test {
|
|
113
|
-
source API {
|
|
114
|
-
auth: bearer
|
|
115
|
-
}
|
|
116
|
-
|
|
117
|
-
action Fetch {
|
|
118
|
-
fetch GET "/items
|
|
119
|
-
}
|
|
120
|
-
|
|
121
|
-
run Fetch
|
|
122
|
-
}`;
|
|
123
|
-
const lexer = new ReqonLexer(source);
|
|
124
|
-
expect(() => lexer.tokenize()).toThrow();
|
|
125
|
-
try {
|
|
126
|
-
lexer.tokenize();
|
|
127
|
-
}
|
|
128
|
-
catch (e) {
|
|
129
|
-
expect(e).toBeInstanceOf(Error);
|
|
130
|
-
const error = e;
|
|
131
|
-
expect(error.message).toContain('Unterminated string');
|
|
132
|
-
}
|
|
133
|
-
});
|
|
134
|
-
it('should produce ParseError for syntax errors', () => {
|
|
135
|
-
const source = `mission Test {
|
|
136
|
-
source API {
|
|
137
|
-
auth: bearer,
|
|
138
|
-
base: "https://api.example.com"
|
|
139
|
-
}
|
|
140
|
-
|
|
141
|
-
action Fetch
|
|
142
|
-
fetch GET "/items"
|
|
143
|
-
}
|
|
144
|
-
|
|
145
|
-
run Fetch
|
|
146
|
-
}`;
|
|
147
|
-
const lexer = new ReqonLexer(source);
|
|
148
|
-
const tokens = lexer.tokenize();
|
|
149
|
-
const parser = new ReqonParser(tokens, source, 'test.reqon');
|
|
150
|
-
expect(() => parser.parse()).toThrow();
|
|
151
|
-
try {
|
|
152
|
-
parser.parse();
|
|
153
|
-
}
|
|
154
|
-
catch (e) {
|
|
155
|
-
expect(e).toBeInstanceOf(ParseError);
|
|
156
|
-
const error = e;
|
|
157
|
-
expect(error.location.line).toBeGreaterThan(0);
|
|
158
|
-
expect(error.context?.source).toBe(source);
|
|
159
|
-
expect(error.context?.filePath).toBe('test.reqon');
|
|
160
|
-
const formatted = error.format();
|
|
161
|
-
expect(formatted).toContain('test.reqon:');
|
|
162
|
-
expect(formatted).toContain('ParseError');
|
|
163
|
-
}
|
|
164
|
-
});
|
|
165
|
-
});
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
export {};
|
|
@@ -1,246 +0,0 @@
|
|
|
1
|
-
import { describe, it, expect, beforeEach, afterEach } from 'vitest';
|
|
2
|
-
import { rmSync, existsSync } from 'node:fs';
|
|
3
|
-
import { createExecutionState, findResumePoint, canResume, getProgress, getExecutionSummary, } from './state.js';
|
|
4
|
-
import { FileExecutionStore, MemoryExecutionStore } from './store.js';
|
|
5
|
-
const TEST_DIR = '.reqon-test-executions';
|
|
6
|
-
describe('ExecutionState', () => {
|
|
7
|
-
describe('createExecutionState', () => {
|
|
8
|
-
it('creates initial state with pending stages', () => {
|
|
9
|
-
const state = createExecutionState({
|
|
10
|
-
mission: 'TestMission',
|
|
11
|
-
stages: ['FetchData', 'ProcessData', 'StoreResults'],
|
|
12
|
-
});
|
|
13
|
-
expect(state.mission).toBe('TestMission');
|
|
14
|
-
expect(state.status).toBe('pending');
|
|
15
|
-
expect(state.stages).toHaveLength(3);
|
|
16
|
-
expect(state.stages[0].action).toBe('FetchData');
|
|
17
|
-
expect(state.stages[0].status).toBe('pending');
|
|
18
|
-
expect(state.id).toMatch(/^exec_/);
|
|
19
|
-
});
|
|
20
|
-
it('includes metadata', () => {
|
|
21
|
-
const state = createExecutionState({
|
|
22
|
-
mission: 'Test',
|
|
23
|
-
stages: ['A'],
|
|
24
|
-
metadata: { tenant: 'acme', userId: '123' },
|
|
25
|
-
});
|
|
26
|
-
expect(state.metadata).toEqual({ tenant: 'acme', userId: '123' });
|
|
27
|
-
});
|
|
28
|
-
});
|
|
29
|
-
describe('findResumePoint', () => {
|
|
30
|
-
it('returns 0 for fresh execution', () => {
|
|
31
|
-
const state = createExecutionState({
|
|
32
|
-
mission: 'Test',
|
|
33
|
-
stages: ['A', 'B', 'C'],
|
|
34
|
-
});
|
|
35
|
-
expect(findResumePoint(state)).toBe(0);
|
|
36
|
-
});
|
|
37
|
-
it('returns index of first non-completed stage', () => {
|
|
38
|
-
const state = createExecutionState({
|
|
39
|
-
mission: 'Test',
|
|
40
|
-
stages: ['A', 'B', 'C'],
|
|
41
|
-
});
|
|
42
|
-
state.stages[0].status = 'completed';
|
|
43
|
-
state.stages[1].status = 'failed';
|
|
44
|
-
expect(findResumePoint(state)).toBe(1);
|
|
45
|
-
});
|
|
46
|
-
it('returns -1 when all stages complete', () => {
|
|
47
|
-
const state = createExecutionState({
|
|
48
|
-
mission: 'Test',
|
|
49
|
-
stages: ['A', 'B'],
|
|
50
|
-
});
|
|
51
|
-
state.stages[0].status = 'completed';
|
|
52
|
-
state.stages[1].status = 'completed';
|
|
53
|
-
expect(findResumePoint(state)).toBe(-1);
|
|
54
|
-
});
|
|
55
|
-
it('skips over skipped stages', () => {
|
|
56
|
-
const state = createExecutionState({
|
|
57
|
-
mission: 'Test',
|
|
58
|
-
stages: ['A', 'B', 'C'],
|
|
59
|
-
});
|
|
60
|
-
state.stages[0].status = 'completed';
|
|
61
|
-
state.stages[1].status = 'skipped';
|
|
62
|
-
expect(findResumePoint(state)).toBe(2);
|
|
63
|
-
});
|
|
64
|
-
});
|
|
65
|
-
describe('canResume', () => {
|
|
66
|
-
it('returns true for failed executions', () => {
|
|
67
|
-
const state = createExecutionState({ mission: 'Test', stages: ['A'] });
|
|
68
|
-
state.status = 'failed';
|
|
69
|
-
expect(canResume(state)).toBe(true);
|
|
70
|
-
});
|
|
71
|
-
it('returns true for paused executions', () => {
|
|
72
|
-
const state = createExecutionState({ mission: 'Test', stages: ['A'] });
|
|
73
|
-
state.status = 'paused';
|
|
74
|
-
expect(canResume(state)).toBe(true);
|
|
75
|
-
});
|
|
76
|
-
it('returns false for completed executions', () => {
|
|
77
|
-
const state = createExecutionState({ mission: 'Test', stages: ['A'] });
|
|
78
|
-
state.status = 'completed';
|
|
79
|
-
expect(canResume(state)).toBe(false);
|
|
80
|
-
});
|
|
81
|
-
it('returns false for running executions', () => {
|
|
82
|
-
const state = createExecutionState({ mission: 'Test', stages: ['A'] });
|
|
83
|
-
state.status = 'running';
|
|
84
|
-
expect(canResume(state)).toBe(false);
|
|
85
|
-
});
|
|
86
|
-
});
|
|
87
|
-
describe('getProgress', () => {
|
|
88
|
-
it('returns 0 for no completed stages', () => {
|
|
89
|
-
const state = createExecutionState({
|
|
90
|
-
mission: 'Test',
|
|
91
|
-
stages: ['A', 'B', 'C', 'D'],
|
|
92
|
-
});
|
|
93
|
-
expect(getProgress(state)).toBe(0);
|
|
94
|
-
});
|
|
95
|
-
it('returns 50 for half completed', () => {
|
|
96
|
-
const state = createExecutionState({
|
|
97
|
-
mission: 'Test',
|
|
98
|
-
stages: ['A', 'B', 'C', 'D'],
|
|
99
|
-
});
|
|
100
|
-
state.stages[0].status = 'completed';
|
|
101
|
-
state.stages[1].status = 'completed';
|
|
102
|
-
expect(getProgress(state)).toBe(50);
|
|
103
|
-
});
|
|
104
|
-
it('returns 100 for all completed', () => {
|
|
105
|
-
const state = createExecutionState({
|
|
106
|
-
mission: 'Test',
|
|
107
|
-
stages: ['A', 'B'],
|
|
108
|
-
});
|
|
109
|
-
state.stages[0].status = 'completed';
|
|
110
|
-
state.stages[1].status = 'completed';
|
|
111
|
-
expect(getProgress(state)).toBe(100);
|
|
112
|
-
});
|
|
113
|
-
it('counts skipped as progress', () => {
|
|
114
|
-
const state = createExecutionState({
|
|
115
|
-
mission: 'Test',
|
|
116
|
-
stages: ['A', 'B'],
|
|
117
|
-
});
|
|
118
|
-
state.stages[0].status = 'completed';
|
|
119
|
-
state.stages[1].status = 'skipped';
|
|
120
|
-
expect(getProgress(state)).toBe(100);
|
|
121
|
-
});
|
|
122
|
-
});
|
|
123
|
-
describe('getExecutionSummary', () => {
|
|
124
|
-
it('generates readable summary', () => {
|
|
125
|
-
const state = createExecutionState({
|
|
126
|
-
mission: 'SyncInvoices',
|
|
127
|
-
stages: ['Fetch', 'Process', 'Store'],
|
|
128
|
-
});
|
|
129
|
-
state.stages[0].status = 'completed';
|
|
130
|
-
state.stages[1].status = 'failed';
|
|
131
|
-
state.status = 'failed';
|
|
132
|
-
const summary = getExecutionSummary(state);
|
|
133
|
-
expect(summary).toContain('SyncInvoices');
|
|
134
|
-
expect(summary).toContain('failed');
|
|
135
|
-
expect(summary).toContain('1 completed');
|
|
136
|
-
expect(summary).toContain('1 failed');
|
|
137
|
-
expect(summary).toContain('1 pending');
|
|
138
|
-
});
|
|
139
|
-
});
|
|
140
|
-
});
|
|
141
|
-
describe('MemoryExecutionStore', () => {
|
|
142
|
-
let store;
|
|
143
|
-
beforeEach(() => {
|
|
144
|
-
store = new MemoryExecutionStore();
|
|
145
|
-
});
|
|
146
|
-
it('saves and loads execution state', async () => {
|
|
147
|
-
const state = createExecutionState({
|
|
148
|
-
mission: 'Test',
|
|
149
|
-
stages: ['A', 'B'],
|
|
150
|
-
});
|
|
151
|
-
await store.save(state);
|
|
152
|
-
const loaded = await store.load(state.id);
|
|
153
|
-
expect(loaded).not.toBeNull();
|
|
154
|
-
expect(loaded.id).toBe(state.id);
|
|
155
|
-
expect(loaded.mission).toBe('Test');
|
|
156
|
-
});
|
|
157
|
-
it('returns null for unknown ID', async () => {
|
|
158
|
-
const loaded = await store.load('nonexistent');
|
|
159
|
-
expect(loaded).toBeNull();
|
|
160
|
-
});
|
|
161
|
-
it('lists by mission', async () => {
|
|
162
|
-
const state1 = createExecutionState({ mission: 'A', stages: ['X'] });
|
|
163
|
-
const state2 = createExecutionState({ mission: 'B', stages: ['X'] });
|
|
164
|
-
const state3 = createExecutionState({ mission: 'A', stages: ['X'] });
|
|
165
|
-
await store.save(state1);
|
|
166
|
-
await store.save(state2);
|
|
167
|
-
await store.save(state3);
|
|
168
|
-
const aExecutions = await store.listByMission('A');
|
|
169
|
-
expect(aExecutions).toHaveLength(2);
|
|
170
|
-
});
|
|
171
|
-
it('finds resumable executions', async () => {
|
|
172
|
-
const completed = createExecutionState({ mission: 'Test', stages: ['A'] });
|
|
173
|
-
completed.status = 'completed';
|
|
174
|
-
const failed = createExecutionState({ mission: 'Test', stages: ['A'] });
|
|
175
|
-
failed.status = 'failed';
|
|
176
|
-
const paused = createExecutionState({ mission: 'Test', stages: ['A'] });
|
|
177
|
-
paused.status = 'paused';
|
|
178
|
-
await store.save(completed);
|
|
179
|
-
await store.save(failed);
|
|
180
|
-
await store.save(paused);
|
|
181
|
-
const resumable = await store.findResumable('Test');
|
|
182
|
-
expect(resumable).toHaveLength(2);
|
|
183
|
-
});
|
|
184
|
-
it('deletes execution state', async () => {
|
|
185
|
-
const state = createExecutionState({ mission: 'Test', stages: ['A'] });
|
|
186
|
-
await store.save(state);
|
|
187
|
-
await store.delete(state.id);
|
|
188
|
-
const loaded = await store.load(state.id);
|
|
189
|
-
expect(loaded).toBeNull();
|
|
190
|
-
});
|
|
191
|
-
});
|
|
192
|
-
describe('FileExecutionStore', () => {
|
|
193
|
-
let store;
|
|
194
|
-
beforeEach(() => {
|
|
195
|
-
if (existsSync(TEST_DIR)) {
|
|
196
|
-
rmSync(TEST_DIR, { recursive: true, force: true });
|
|
197
|
-
}
|
|
198
|
-
store = new FileExecutionStore(TEST_DIR);
|
|
199
|
-
});
|
|
200
|
-
afterEach(() => {
|
|
201
|
-
if (existsSync(TEST_DIR)) {
|
|
202
|
-
rmSync(TEST_DIR, { recursive: true, force: true });
|
|
203
|
-
}
|
|
204
|
-
});
|
|
205
|
-
it('creates directory if not exists', async () => {
|
|
206
|
-
// Trigger initialization by calling any async method
|
|
207
|
-
await store.listRecent();
|
|
208
|
-
expect(existsSync(TEST_DIR)).toBe(true);
|
|
209
|
-
});
|
|
210
|
-
it('persists state to disk', async () => {
|
|
211
|
-
const state = createExecutionState({
|
|
212
|
-
mission: 'Persistent',
|
|
213
|
-
stages: ['A', 'B'],
|
|
214
|
-
});
|
|
215
|
-
state.stages[0].status = 'completed';
|
|
216
|
-
await store.save(state);
|
|
217
|
-
// Create new store instance
|
|
218
|
-
const newStore = new FileExecutionStore(TEST_DIR);
|
|
219
|
-
const loaded = await newStore.load(state.id);
|
|
220
|
-
expect(loaded).not.toBeNull();
|
|
221
|
-
expect(loaded.stages[0].status).toBe('completed');
|
|
222
|
-
});
|
|
223
|
-
it('preserves Date objects', async () => {
|
|
224
|
-
const state = createExecutionState({
|
|
225
|
-
mission: 'Test',
|
|
226
|
-
stages: ['A'],
|
|
227
|
-
});
|
|
228
|
-
state.stages[0].startedAt = new Date();
|
|
229
|
-
state.stages[0].completedAt = new Date();
|
|
230
|
-
await store.save(state);
|
|
231
|
-
const loaded = await store.load(state.id);
|
|
232
|
-
expect(loaded.startedAt).toBeInstanceOf(Date);
|
|
233
|
-
expect(loaded.stages[0].startedAt).toBeInstanceOf(Date);
|
|
234
|
-
expect(loaded.stages[0].completedAt).toBeInstanceOf(Date);
|
|
235
|
-
});
|
|
236
|
-
it('finds latest execution for mission', async () => {
|
|
237
|
-
const older = createExecutionState({ mission: 'Test', stages: ['A'] });
|
|
238
|
-
older.startedAt = new Date(Date.now() - 10000);
|
|
239
|
-
const newer = createExecutionState({ mission: 'Test', stages: ['A'] });
|
|
240
|
-
newer.startedAt = new Date();
|
|
241
|
-
await store.save(older);
|
|
242
|
-
await store.save(newer);
|
|
243
|
-
const latest = await store.findLatest('Test');
|
|
244
|
-
expect(latest.id).toBe(newer.id);
|
|
245
|
-
});
|
|
246
|
-
});
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
export {};
|