reqon-dsl 0.2.0 → 0.3.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 +22 -0
- package/dist/ast/nodes.d.ts +83 -4
- package/dist/ast/nodes.js +14 -0
- package/dist/auth/circuit-breaker.js +7 -6
- package/dist/auth/rate-limiter.d.ts +4 -0
- package/dist/auth/rate-limiter.js +9 -16
- package/dist/cli.d.ts +13 -0
- package/dist/cli.js +84 -4
- package/dist/config/constants.d.ts +141 -0
- package/dist/config/constants.js +128 -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 +88 -0
- package/dist/control/server.js +238 -0
- package/dist/control/types.d.ts +55 -0
- package/dist/control/types.js +7 -0
- package/dist/debug/cli-debugger.d.ts +17 -0
- package/dist/debug/cli-debugger.js +180 -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/index.d.ts +21 -1
- package/dist/index.js +33 -2
- package/dist/interpreter/context.d.ts +14 -0
- package/dist/interpreter/context.js +15 -0
- package/dist/interpreter/evaluator.d.ts +63 -1
- package/dist/interpreter/evaluator.js +186 -39
- package/dist/interpreter/executor.d.ts +70 -14
- package/dist/interpreter/executor.js +503 -174
- package/dist/interpreter/fetch-handler.d.ts +9 -0
- package/dist/interpreter/fetch-handler.js +133 -24
- package/dist/interpreter/http.d.ts +5 -0
- package/dist/interpreter/http.js +26 -12
- package/dist/interpreter/index.d.ts +3 -1
- package/dist/interpreter/index.js +2 -0
- package/dist/interpreter/pagination.d.ts +11 -2
- package/dist/interpreter/pagination.js +95 -31
- 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 +13 -0
- package/dist/interpreter/step-handlers/for-handler.js +71 -4
- 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 +43 -16
- 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 +11 -1
- package/dist/interpreter/step-handlers/store-handler.js +45 -13
- 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 +4 -2
- package/dist/interpreter/step-handlers/webhook-handler.d.ts +3 -0
- package/dist/interpreter/step-handlers/webhook-handler.js +18 -2
- package/dist/interpreter/store-manager.d.ts +46 -0
- package/dist/interpreter/store-manager.js +66 -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/mcp/index.d.ts +11 -0
- package/dist/mcp/index.js +11 -0
- package/dist/mcp/server.d.ts +17 -0
- package/dist/mcp/server.js +451 -0
- package/dist/oas/index.d.ts +2 -0
- package/dist/oas/index.js +1 -0
- package/dist/oas/mock-generator.d.ts +12 -0
- package/dist/oas/mock-generator.js +187 -0
- package/dist/observability/events.d.ts +244 -0
- package/dist/observability/events.js +90 -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 +259 -0
- package/dist/observability/otel.d.ts +135 -0
- package/dist/observability/otel.js +386 -0
- package/dist/parser/action-parser.d.ts +105 -0
- package/dist/parser/action-parser.js +645 -0
- package/dist/parser/expressions.d.ts +13 -0
- package/dist/parser/expressions.js +72 -2
- package/dist/parser/fetch-parser.d.ts +27 -0
- package/dist/parser/fetch-parser.js +269 -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 +14 -0
- package/dist/pause/index.js +11 -0
- package/dist/pause/manager.d.ts +118 -0
- package/dist/pause/manager.js +245 -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 +156 -0
- package/dist/plugin.d.ts +9 -12
- package/dist/plugin.js +10 -13
- package/dist/stores/factory.d.ts +1 -1
- package/dist/stores/factory.js +3 -2
- package/dist/stores/file.d.ts +26 -0
- package/dist/stores/file.js +64 -10
- 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 +11 -0
- package/dist/stores/types.d.ts +17 -0
- package/dist/stores/types.js +12 -0
- package/dist/trace/index.d.ts +16 -0
- package/dist/trace/index.js +12 -0
- package/dist/trace/recorder.d.ts +71 -0
- package/dist/trace/recorder.js +144 -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 +69 -0
- package/dist/trace/store.js +225 -0
- package/dist/utils/index.d.ts +1 -0
- package/dist/utils/index.js +1 -0
- package/dist/utils/type-guards.d.ts +58 -0
- package/dist/utils/type-guards.js +92 -0
- package/dist/webhook/server.js +7 -6
- package/package.json +55 -6
- 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
|
@@ -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 {};
|
package/dist/integration.test.js
DELETED
|
@@ -1,168 +0,0 @@
|
|
|
1
|
-
import { describe, it, expect } from 'vitest';
|
|
2
|
-
import { parse, execute } from './index.js';
|
|
3
|
-
import { MemoryStore } from './stores/index.js';
|
|
4
|
-
describe('Reqon Integration', () => {
|
|
5
|
-
it('parses and executes a simple mission with dry run', async () => {
|
|
6
|
-
const source = `
|
|
7
|
-
mission TestMission {
|
|
8
|
-
source API {
|
|
9
|
-
auth: bearer,
|
|
10
|
-
base: "https://api.example.com"
|
|
11
|
-
}
|
|
12
|
-
|
|
13
|
-
store items: memory("items")
|
|
14
|
-
|
|
15
|
-
action FetchItems {
|
|
16
|
-
fetch GET "/items"
|
|
17
|
-
|
|
18
|
-
store response -> items {
|
|
19
|
-
key: .id
|
|
20
|
-
}
|
|
21
|
-
}
|
|
22
|
-
|
|
23
|
-
run FetchItems
|
|
24
|
-
}
|
|
25
|
-
`;
|
|
26
|
-
const program = parse(source);
|
|
27
|
-
expect(program.type).toBe('ReqonProgram');
|
|
28
|
-
const result = await execute(source, { dryRun: true, verbose: false });
|
|
29
|
-
expect(result.success).toBe(true);
|
|
30
|
-
expect(result.actionsRun).toContain('FetchItems');
|
|
31
|
-
});
|
|
32
|
-
it('executes a mission with mock data flow', async () => {
|
|
33
|
-
const source = `
|
|
34
|
-
mission MockDataFlow {
|
|
35
|
-
source API {
|
|
36
|
-
auth: bearer,
|
|
37
|
-
base: "https://api.example.com"
|
|
38
|
-
}
|
|
39
|
-
|
|
40
|
-
store raw: memory("raw")
|
|
41
|
-
store processed: memory("processed")
|
|
42
|
-
|
|
43
|
-
action Process {
|
|
44
|
-
for item in raw {
|
|
45
|
-
map item -> Processed {
|
|
46
|
-
id: .id,
|
|
47
|
-
name: .title,
|
|
48
|
-
value: .amount
|
|
49
|
-
}
|
|
50
|
-
|
|
51
|
-
store response -> processed {
|
|
52
|
-
key: .id
|
|
53
|
-
}
|
|
54
|
-
}
|
|
55
|
-
}
|
|
56
|
-
|
|
57
|
-
run Process
|
|
58
|
-
}
|
|
59
|
-
`;
|
|
60
|
-
// Pre-populate the raw store
|
|
61
|
-
const rawStore = new MemoryStore('raw');
|
|
62
|
-
await rawStore.set('1', { id: '1', title: 'Item 1', amount: 100 });
|
|
63
|
-
await rawStore.set('2', { id: '2', title: 'Item 2', amount: 200 });
|
|
64
|
-
const result = await execute(source, {
|
|
65
|
-
dryRun: false,
|
|
66
|
-
verbose: false,
|
|
67
|
-
stores: { raw: rawStore },
|
|
68
|
-
});
|
|
69
|
-
expect(result.success).toBe(true);
|
|
70
|
-
expect(result.actionsRun).toContain('Process');
|
|
71
|
-
// Check processed store
|
|
72
|
-
const processedStore = result.stores.get('processed');
|
|
73
|
-
expect(processedStore).toBeDefined();
|
|
74
|
-
const items = await processedStore.list();
|
|
75
|
-
expect(items).toHaveLength(2);
|
|
76
|
-
expect(items[0]).toMatchObject({ id: '1', name: 'Item 1', value: 100 });
|
|
77
|
-
});
|
|
78
|
-
it('parses the Xero example file', async () => {
|
|
79
|
-
const fs = await import('node:fs/promises');
|
|
80
|
-
const source = await fs.readFile('./examples/xero/invoices.reqon', 'utf-8');
|
|
81
|
-
const program = parse(source);
|
|
82
|
-
expect(program.type).toBe('ReqonProgram');
|
|
83
|
-
expect(program.statements).toHaveLength(1);
|
|
84
|
-
const mission = program.statements[0];
|
|
85
|
-
if (mission.type === 'MissionDefinition') {
|
|
86
|
-
expect(mission.name).toBe('SyncXeroInvoices');
|
|
87
|
-
expect(mission.sources).toHaveLength(1);
|
|
88
|
-
expect(mission.stores).toHaveLength(2);
|
|
89
|
-
expect(mission.actions).toHaveLength(3);
|
|
90
|
-
expect(mission.pipeline.stages).toHaveLength(3);
|
|
91
|
-
}
|
|
92
|
-
});
|
|
93
|
-
it('handles validation failures gracefully', async () => {
|
|
94
|
-
const source = `
|
|
95
|
-
mission ValidationTest {
|
|
96
|
-
source API {
|
|
97
|
-
auth: bearer,
|
|
98
|
-
base: "https://api.example.com"
|
|
99
|
-
}
|
|
100
|
-
|
|
101
|
-
store items: memory("items")
|
|
102
|
-
|
|
103
|
-
action Validate {
|
|
104
|
-
for item in items {
|
|
105
|
-
validate item {
|
|
106
|
-
assume .value > 0
|
|
107
|
-
}
|
|
108
|
-
}
|
|
109
|
-
}
|
|
110
|
-
|
|
111
|
-
run Validate
|
|
112
|
-
}
|
|
113
|
-
`;
|
|
114
|
-
const itemsStore = new MemoryStore('items');
|
|
115
|
-
await itemsStore.set('1', { id: '1', value: -5 }); // Invalid: negative value
|
|
116
|
-
const result = await execute(source, {
|
|
117
|
-
stores: { items: itemsStore },
|
|
118
|
-
});
|
|
119
|
-
// Validation should fail
|
|
120
|
-
expect(result.success).toBe(false);
|
|
121
|
-
expect(result.errors.length).toBeGreaterThan(0);
|
|
122
|
-
});
|
|
123
|
-
it('processes match expressions in map steps', async () => {
|
|
124
|
-
const source = `
|
|
125
|
-
mission MatchTest {
|
|
126
|
-
source API {
|
|
127
|
-
auth: bearer,
|
|
128
|
-
base: "https://api.example.com"
|
|
129
|
-
}
|
|
130
|
-
|
|
131
|
-
store input: memory("input")
|
|
132
|
-
store output: memory("output")
|
|
133
|
-
|
|
134
|
-
action Transform {
|
|
135
|
-
for item in input {
|
|
136
|
-
map item -> Output {
|
|
137
|
-
id: .id,
|
|
138
|
-
status: match .state {
|
|
139
|
-
"A" => "active",
|
|
140
|
-
"I" => "inactive",
|
|
141
|
-
_ => "unknown"
|
|
142
|
-
}
|
|
143
|
-
}
|
|
144
|
-
|
|
145
|
-
store response -> output {
|
|
146
|
-
key: .id
|
|
147
|
-
}
|
|
148
|
-
}
|
|
149
|
-
}
|
|
150
|
-
|
|
151
|
-
run Transform
|
|
152
|
-
}
|
|
153
|
-
`;
|
|
154
|
-
const inputStore = new MemoryStore('input');
|
|
155
|
-
await inputStore.set('1', { id: '1', state: 'A' });
|
|
156
|
-
await inputStore.set('2', { id: '2', state: 'I' });
|
|
157
|
-
await inputStore.set('3', { id: '3', state: 'X' });
|
|
158
|
-
const result = await execute(source, {
|
|
159
|
-
stores: { input: inputStore },
|
|
160
|
-
});
|
|
161
|
-
expect(result.success).toBe(true);
|
|
162
|
-
const outputStore = result.stores.get('output');
|
|
163
|
-
const items = await outputStore.list();
|
|
164
|
-
expect(items).toContainEqual(expect.objectContaining({ id: '1', status: 'active' }));
|
|
165
|
-
expect(items).toContainEqual(expect.objectContaining({ id: '2', status: 'inactive' }));
|
|
166
|
-
expect(items).toContainEqual(expect.objectContaining({ id: '3', status: 'unknown' }));
|
|
167
|
-
});
|
|
168
|
-
});
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
export {};
|