reqon-dsl 0.2.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/.claude/settings.local.json +31 -0
- package/.claude/skills/api-integration.md +125 -0
- package/.claude/skills/database-schema.md +51 -0
- package/.claude/skills/dsl-design.md +80 -0
- package/.claude/skills/property-testing.md +143 -0
- package/.claude/skills/reqon/SKILL.md +44 -0
- package/.claude/skills/reqon/references/examples.md +206 -0
- package/.claude/skills/reqon/references/syntax.md +263 -0
- package/.claude/skills/vscode-extension.md +113 -0
- package/.github/dependabot.yml +32 -0
- package/.github/pull_request_template.md +21 -0
- package/.github/workflows/ci.yml +174 -0
- package/.github/workflows/release.yml +73 -0
- package/CLAUDE.md +72 -0
- package/CONTRIBUTING.md +161 -0
- package/README.md +235 -0
- package/TODO.md +51 -0
- package/dist/ast/index.d.ts +1 -0
- package/dist/ast/index.js +1 -0
- package/dist/ast/nodes.d.ts +237 -0
- package/dist/ast/nodes.js +12 -0
- package/dist/auth/auth.test.d.ts +1 -0
- package/dist/auth/auth.test.js +255 -0
- package/dist/auth/circuit-breaker.d.ts +115 -0
- package/dist/auth/circuit-breaker.js +267 -0
- package/dist/auth/credentials.d.ts +91 -0
- package/dist/auth/credentials.js +169 -0
- package/dist/auth/index.d.ts +5 -0
- package/dist/auth/index.js +8 -0
- package/dist/auth/oauth2-provider.d.ts +41 -0
- package/dist/auth/oauth2-provider.js +131 -0
- package/dist/auth/rate-limiter.d.ts +61 -0
- package/dist/auth/rate-limiter.js +380 -0
- package/dist/auth/token-store.d.ts +30 -0
- package/dist/auth/token-store.js +148 -0
- package/dist/auth/types.d.ts +142 -0
- package/dist/auth/types.js +1 -0
- package/dist/cli.d.ts +2 -0
- package/dist/cli.js +270 -0
- package/dist/errors/errors.test.d.ts +1 -0
- package/dist/errors/errors.test.js +165 -0
- package/dist/errors/index.d.ts +83 -0
- package/dist/errors/index.js +159 -0
- package/dist/execution/execution.test.d.ts +1 -0
- package/dist/execution/execution.test.js +246 -0
- package/dist/execution/index.d.ts +4 -0
- package/dist/execution/index.js +2 -0
- package/dist/execution/state.d.ts +136 -0
- package/dist/execution/state.js +82 -0
- package/dist/execution/store.d.ts +52 -0
- package/dist/execution/store.js +120 -0
- package/dist/index.d.ts +27 -0
- package/dist/index.js +57 -0
- package/dist/integration.test.d.ts +1 -0
- package/dist/integration.test.js +168 -0
- package/dist/interpreter/context.d.ts +15 -0
- package/dist/interpreter/context.js +29 -0
- package/dist/interpreter/evaluator.d.ts +5 -0
- package/dist/interpreter/evaluator.js +223 -0
- package/dist/interpreter/evaluator.test.d.ts +1 -0
- package/dist/interpreter/evaluator.test.js +512 -0
- package/dist/interpreter/executor.d.ts +131 -0
- package/dist/interpreter/executor.js +663 -0
- package/dist/interpreter/fetch-handler.d.ts +43 -0
- package/dist/interpreter/fetch-handler.js +203 -0
- package/dist/interpreter/http.d.ts +57 -0
- package/dist/interpreter/http.js +210 -0
- package/dist/interpreter/http.test.d.ts +1 -0
- package/dist/interpreter/http.test.js +299 -0
- package/dist/interpreter/index.d.ts +7 -0
- package/dist/interpreter/index.js +7 -0
- package/dist/interpreter/pagination.d.ts +63 -0
- package/dist/interpreter/pagination.js +155 -0
- package/dist/interpreter/progress.test.d.ts +1 -0
- package/dist/interpreter/progress.test.js +216 -0
- package/dist/interpreter/schema-matcher.d.ts +16 -0
- package/dist/interpreter/schema-matcher.js +136 -0
- package/dist/interpreter/schema-matcher.test.d.ts +1 -0
- package/dist/interpreter/schema-matcher.test.js +122 -0
- package/dist/interpreter/signals.d.ts +57 -0
- package/dist/interpreter/signals.js +73 -0
- package/dist/interpreter/step-handlers/for-handler.d.ts +17 -0
- package/dist/interpreter/step-handlers/for-handler.js +51 -0
- package/dist/interpreter/step-handlers/index.d.ts +8 -0
- package/dist/interpreter/step-handlers/index.js +8 -0
- package/dist/interpreter/step-handlers/map-handler.d.ts +10 -0
- package/dist/interpreter/step-handlers/map-handler.js +20 -0
- package/dist/interpreter/step-handlers/match-handler.d.ts +27 -0
- package/dist/interpreter/step-handlers/match-handler.js +61 -0
- package/dist/interpreter/step-handlers/store-handler.d.ts +13 -0
- package/dist/interpreter/step-handlers/store-handler.js +66 -0
- package/dist/interpreter/step-handlers/types.d.ts +15 -0
- package/dist/interpreter/step-handlers/types.js +1 -0
- package/dist/interpreter/step-handlers/validate-handler.d.ts +10 -0
- package/dist/interpreter/step-handlers/validate-handler.js +26 -0
- package/dist/interpreter/step-handlers/webhook-handler.d.ts +36 -0
- package/dist/interpreter/step-handlers/webhook-handler.js +104 -0
- package/dist/lexer/index.d.ts +10 -0
- package/dist/lexer/index.js +12 -0
- package/dist/lexer/lexer.d.ts +24 -0
- package/dist/lexer/lexer.js +264 -0
- package/dist/lexer/lexer.test.d.ts +1 -0
- package/dist/lexer/lexer.test.js +259 -0
- package/dist/lexer/tokens.d.ts +69 -0
- package/dist/lexer/tokens.js +146 -0
- package/dist/loader/index.d.ts +36 -0
- package/dist/loader/index.js +220 -0
- package/dist/loader/loader.test.d.ts +1 -0
- package/dist/loader/loader.test.js +287 -0
- package/dist/oas/index.d.ts +4 -0
- package/dist/oas/index.js +2 -0
- package/dist/oas/loader.d.ts +21 -0
- package/dist/oas/loader.js +82 -0
- package/dist/oas/oas.test.d.ts +1 -0
- package/dist/oas/oas.test.js +218 -0
- package/dist/oas/validator.d.ts +12 -0
- package/dist/oas/validator.js +227 -0
- package/dist/parser/base.d.ts +33 -0
- package/dist/parser/base.js +97 -0
- package/dist/parser/expressions.d.ts +27 -0
- package/dist/parser/expressions.js +248 -0
- package/dist/parser/expressions.test.d.ts +1 -0
- package/dist/parser/expressions.test.js +378 -0
- package/dist/parser/index.d.ts +3 -0
- package/dist/parser/index.js +3 -0
- package/dist/parser/match.test.d.ts +1 -0
- package/dist/parser/match.test.js +254 -0
- package/dist/parser/parser.d.ts +68 -0
- package/dist/parser/parser.js +1229 -0
- package/dist/parser/parser.test.d.ts +1 -0
- package/dist/parser/parser.test.js +333 -0
- package/dist/parser/schedule.test.d.ts +1 -0
- package/dist/parser/schedule.test.js +241 -0
- package/dist/plugin.d.ts +35 -0
- package/dist/plugin.js +68 -0
- package/dist/scheduler/cron-parser.d.ts +32 -0
- package/dist/scheduler/cron-parser.js +198 -0
- package/dist/scheduler/cron-parser.test.d.ts +1 -0
- package/dist/scheduler/cron-parser.test.js +188 -0
- package/dist/scheduler/index.d.ts +3 -0
- package/dist/scheduler/index.js +2 -0
- package/dist/scheduler/scheduler.d.ts +81 -0
- package/dist/scheduler/scheduler.js +376 -0
- package/dist/scheduler/types.d.ts +65 -0
- package/dist/scheduler/types.js +1 -0
- package/dist/stores/factory.d.ts +36 -0
- package/dist/stores/factory.js +73 -0
- package/dist/stores/file.d.ts +60 -0
- package/dist/stores/file.js +173 -0
- package/dist/stores/file.test.d.ts +1 -0
- package/dist/stores/file.test.js +165 -0
- package/dist/stores/index.d.ts +6 -0
- package/dist/stores/index.js +5 -0
- package/dist/stores/memory.d.ts +19 -0
- package/dist/stores/memory.js +51 -0
- package/dist/stores/memory.test.d.ts +1 -0
- package/dist/stores/memory.test.js +157 -0
- package/dist/stores/postgrest.d.ts +55 -0
- package/dist/stores/postgrest.js +217 -0
- package/dist/stores/stores.test.d.ts +1 -0
- package/dist/stores/stores.test.js +158 -0
- package/dist/stores/types.d.ts +31 -0
- package/dist/stores/types.js +26 -0
- package/dist/sync/index.d.ts +4 -0
- package/dist/sync/index.js +2 -0
- package/dist/sync/state.d.ts +69 -0
- package/dist/sync/state.js +66 -0
- package/dist/sync/store.d.ts +49 -0
- package/dist/sync/store.js +93 -0
- package/dist/sync/sync.test.d.ts +1 -0
- package/dist/sync/sync.test.js +221 -0
- package/dist/utils/async.d.ts +7 -0
- package/dist/utils/async.js +9 -0
- package/dist/utils/file.d.ts +38 -0
- package/dist/utils/file.js +92 -0
- package/dist/utils/index.d.ts +4 -0
- package/dist/utils/index.js +4 -0
- package/dist/utils/logger.d.ts +34 -0
- package/dist/utils/logger.js +39 -0
- package/dist/utils/path.d.ts +12 -0
- package/dist/utils/path.js +41 -0
- package/dist/webhook/index.d.ts +8 -0
- package/dist/webhook/index.js +7 -0
- package/dist/webhook/server.d.ts +84 -0
- package/dist/webhook/server.js +319 -0
- package/dist/webhook/store.d.ts +67 -0
- package/dist/webhook/store.js +193 -0
- package/dist/webhook/types.d.ts +88 -0
- package/dist/webhook/types.js +6 -0
- package/docusaurus/README.md +41 -0
- package/docusaurus/docs/advanced/execution-state.md +283 -0
- package/docusaurus/docs/advanced/extending-reqon.md +388 -0
- package/docusaurus/docs/advanced/multi-file-missions.md +250 -0
- package/docusaurus/docs/advanced/parallel-execution.md +353 -0
- package/docusaurus/docs/api-reference.md +443 -0
- package/docusaurus/docs/authentication/api-key.md +339 -0
- package/docusaurus/docs/authentication/basic.md +276 -0
- package/docusaurus/docs/authentication/bearer.md +282 -0
- package/docusaurus/docs/authentication/oauth2.md +317 -0
- package/docusaurus/docs/authentication/overview.md +251 -0
- package/docusaurus/docs/cli.md +229 -0
- package/docusaurus/docs/core-concepts/actions.md +286 -0
- package/docusaurus/docs/core-concepts/missions.md +264 -0
- package/docusaurus/docs/core-concepts/schemas.md +353 -0
- package/docusaurus/docs/core-concepts/sources.md +339 -0
- package/docusaurus/docs/core-concepts/stores.md +332 -0
- package/docusaurus/docs/dsl-syntax/expressions.md +361 -0
- package/docusaurus/docs/dsl-syntax/fetch.md +293 -0
- package/docusaurus/docs/dsl-syntax/for-loops.md +324 -0
- package/docusaurus/docs/dsl-syntax/map.md +345 -0
- package/docusaurus/docs/dsl-syntax/match.md +387 -0
- package/docusaurus/docs/dsl-syntax/pipelines.md +397 -0
- package/docusaurus/docs/dsl-syntax/validate.md +401 -0
- package/docusaurus/docs/error-handling/dead-letter-queues.md +399 -0
- package/docusaurus/docs/error-handling/flow-control.md +337 -0
- package/docusaurus/docs/error-handling/retry-strategies.md +368 -0
- package/docusaurus/docs/examples.md +488 -0
- package/docusaurus/docs/getting-started.md +256 -0
- package/docusaurus/docs/http/circuit-breaker.md +401 -0
- package/docusaurus/docs/http/incremental-sync.md +394 -0
- package/docusaurus/docs/http/pagination.md +361 -0
- package/docusaurus/docs/http/rate-limiting.md +383 -0
- package/docusaurus/docs/http/requests.md +328 -0
- package/docusaurus/docs/http/retry.md +402 -0
- package/docusaurus/docs/intro.md +90 -0
- package/docusaurus/docs/openapi/loading-specs.md +305 -0
- package/docusaurus/docs/openapi/operation-calls.md +314 -0
- package/docusaurus/docs/openapi/overview.md +212 -0
- package/docusaurus/docs/openapi/response-validation.md +344 -0
- package/docusaurus/docs/scheduling/cron.md +305 -0
- package/docusaurus/docs/scheduling/daemon-mode.md +317 -0
- package/docusaurus/docs/scheduling/intervals.md +289 -0
- package/docusaurus/docs/scheduling/overview.md +231 -0
- package/docusaurus/docs/stores/custom-adapters.md +376 -0
- package/docusaurus/docs/stores/file.md +236 -0
- package/docusaurus/docs/stores/memory.md +193 -0
- package/docusaurus/docs/stores/overview.md +274 -0
- package/docusaurus/docs/stores/postgrest.md +316 -0
- package/docusaurus/docusaurus.config.ts +148 -0
- package/docusaurus/package-lock.json +18029 -0
- package/docusaurus/package.json +47 -0
- package/docusaurus/sidebars.ts +155 -0
- package/docusaurus/src/components/HomepageFeatures/index.tsx +105 -0
- package/docusaurus/src/components/HomepageFeatures/styles.module.css +12 -0
- package/docusaurus/src/css/custom.css +169 -0
- package/docusaurus/src/pages/index.module.css +48 -0
- package/docusaurus/src/pages/index.tsx +110 -0
- package/docusaurus/src/pages/markdown-page.md +7 -0
- 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 +10 -0
- package/docusaurus/static/img/undraw_docusaurus_mountain.svg +171 -0
- package/docusaurus/static/img/undraw_docusaurus_react.svg +170 -0
- package/docusaurus/static/img/undraw_docusaurus_tree.svg +40 -0
- package/docusaurus/tsconfig.json +8 -0
- package/examples/README.md +112 -0
- package/examples/error-handling/README.md +150 -0
- package/examples/error-handling/payment-processor.vague +287 -0
- package/examples/github-sync/README.md +74 -0
- package/examples/github-sync/fetch-issues.vague +47 -0
- package/examples/github-sync/fetch-prs.vague +40 -0
- package/examples/github-sync/mission.vague +101 -0
- package/examples/github-sync/normalize.vague +70 -0
- package/examples/jsonplaceholder/README.md +28 -0
- package/examples/jsonplaceholder/posts.vague +48 -0
- package/examples/petstore/README.md +35 -0
- package/examples/petstore/openapi.yaml +97 -0
- package/examples/petstore/sync.vague +52 -0
- package/examples/temporal-comparison/README.md +297 -0
- package/examples/temporal-comparison/reconciliation.vague +355 -0
- package/examples/temporal-comparison/temporal/activities/index.ts +8 -0
- package/examples/temporal-comparison/temporal/activities/shipstation.ts +225 -0
- package/examples/temporal-comparison/temporal/activities/shopify.ts +257 -0
- package/examples/temporal-comparison/temporal/activities/storage.ts +198 -0
- package/examples/temporal-comparison/temporal/activities/stripe.ts +169 -0
- package/examples/temporal-comparison/temporal/activities/validation.ts +205 -0
- package/examples/temporal-comparison/temporal/client/schedule.ts +218 -0
- package/examples/temporal-comparison/temporal/config/retry.ts +63 -0
- package/examples/temporal-comparison/temporal/types/index.ts +129 -0
- package/examples/temporal-comparison/temporal/workers/main.ts +130 -0
- package/examples/temporal-comparison/temporal/workflows/orderReconciliation.ts +262 -0
- package/examples/xero/README.md +88 -0
- package/examples/xero/invoices.vague +189 -0
- package/package.json +40 -0
- package/src/api-integration.test.ts +954 -0
- package/src/ast/index.ts +1 -0
- package/src/ast/nodes.ts +310 -0
- package/src/auth/auth.test.ts +326 -0
- package/src/auth/circuit-breaker.test.ts +390 -0
- package/src/auth/circuit-breaker.ts +379 -0
- package/src/auth/credentials.test.ts +273 -0
- package/src/auth/credentials.ts +246 -0
- package/src/auth/index.ts +40 -0
- package/src/auth/oauth2-provider.ts +177 -0
- package/src/auth/rate-limiter.ts +459 -0
- package/src/auth/token-store.ts +177 -0
- package/src/auth/types.ts +159 -0
- package/src/benchmark/e2e.bench.ts +288 -0
- package/src/benchmark/evaluator.bench.ts +331 -0
- package/src/benchmark/fixtures.ts +295 -0
- package/src/benchmark/index.ts +108 -0
- package/src/benchmark/lexer.bench.ts +69 -0
- package/src/benchmark/parser.bench.ts +103 -0
- package/src/benchmark/resilience.bench.ts +193 -0
- package/src/benchmark/store.bench.ts +147 -0
- package/src/benchmark/utils.ts +230 -0
- package/src/cli.ts +313 -0
- package/src/errors/errors.test.ts +234 -0
- package/src/errors/index.ts +223 -0
- package/src/execution/execution.test.ts +307 -0
- package/src/execution/index.ts +21 -0
- package/src/execution/state.ts +207 -0
- package/src/execution/store.ts +188 -0
- package/src/index.ts +169 -0
- package/src/integration.test.ts +192 -0
- package/src/interpreter/context.ts +57 -0
- package/src/interpreter/evaluator.test.ts +796 -0
- package/src/interpreter/evaluator.ts +245 -0
- package/src/interpreter/executor.ts +946 -0
- package/src/interpreter/fetch-handler.ts +302 -0
- package/src/interpreter/http.test.ts +423 -0
- package/src/interpreter/http.ts +308 -0
- package/src/interpreter/index.ts +32 -0
- package/src/interpreter/pagination.ts +207 -0
- package/src/interpreter/progress.test.ts +276 -0
- package/src/interpreter/schema-matcher.test.ts +160 -0
- package/src/interpreter/schema-matcher.ts +168 -0
- package/src/interpreter/signals.ts +73 -0
- package/src/interpreter/step-handlers/for-handler.ts +65 -0
- package/src/interpreter/step-handlers/index.ts +17 -0
- package/src/interpreter/step-handlers/map-handler.ts +24 -0
- package/src/interpreter/step-handlers/match-handler.ts +101 -0
- package/src/interpreter/step-handlers/store-handler.ts +78 -0
- package/src/interpreter/step-handlers/types.ts +17 -0
- package/src/interpreter/step-handlers/validate-handler.ts +30 -0
- package/src/interpreter/step-handlers/webhook-handler.ts +142 -0
- package/src/lexer/index.ts +18 -0
- package/src/lexer/lexer.test.ts +316 -0
- package/src/lexer/tokens.ts +179 -0
- package/src/loader/index.ts +288 -0
- package/src/loader/loader.test.ts +360 -0
- package/src/oas/index.ts +4 -0
- package/src/oas/loader.ts +126 -0
- package/src/oas/oas.test.ts +254 -0
- package/src/oas/validator.ts +299 -0
- package/src/parser/base.ts +124 -0
- package/src/parser/expressions.test.ts +525 -0
- package/src/parser/expressions.ts +314 -0
- package/src/parser/index.ts +3 -0
- package/src/parser/match.test.ts +296 -0
- package/src/parser/parser.test.ts +739 -0
- package/src/parser/parser.ts +1469 -0
- package/src/parser/schedule.test.ts +287 -0
- package/src/parser/webhook.test.ts +248 -0
- package/src/plugin.ts +83 -0
- package/src/scheduler/cron-parser.test.ts +236 -0
- package/src/scheduler/cron-parser.ts +236 -0
- package/src/scheduler/index.ts +10 -0
- package/src/scheduler/scheduler.ts +443 -0
- package/src/scheduler/types.ts +71 -0
- package/src/stores/factory.ts +104 -0
- package/src/stores/file.test.ts +276 -0
- package/src/stores/file.ts +211 -0
- package/src/stores/index.ts +6 -0
- package/src/stores/memory.test.ts +238 -0
- package/src/stores/memory.ts +63 -0
- package/src/stores/postgrest.test.ts +488 -0
- package/src/stores/postgrest.ts +263 -0
- package/src/stores/stores.test.ts +197 -0
- package/src/stores/types.ts +58 -0
- package/src/sync/index.ts +16 -0
- package/src/sync/state.ts +126 -0
- package/src/sync/store.ts +139 -0
- package/src/sync/sync.test.ts +271 -0
- package/src/utils/async.ts +10 -0
- package/src/utils/file.ts +106 -0
- package/src/utils/index.ts +14 -0
- package/src/utils/logger.ts +53 -0
- package/src/utils/path.ts +47 -0
- package/src/webhook/index.ts +15 -0
- package/src/webhook/server.test.ts +253 -0
- package/src/webhook/server.ts +389 -0
- package/src/webhook/store.ts +239 -0
- package/src/webhook/types.ts +93 -0
- package/tsconfig.json +17 -0
- package/vitest.config.ts +39 -0
|
@@ -0,0 +1,276 @@
|
|
|
1
|
+
import { describe, it, expect, vi, beforeEach } from 'vitest';
|
|
2
|
+
import { MissionExecutor } from './executor.js';
|
|
3
|
+
import type {
|
|
4
|
+
ProgressCallbacks,
|
|
5
|
+
ExecutionStartEvent,
|
|
6
|
+
ExecutionCompleteEvent,
|
|
7
|
+
StageStartEvent,
|
|
8
|
+
StageCompleteEvent,
|
|
9
|
+
} from './executor.js';
|
|
10
|
+
import type { ReqonProgram, MissionDefinition } from '../ast/nodes.js';
|
|
11
|
+
|
|
12
|
+
function createTestProgram(stages: string[]): ReqonProgram {
|
|
13
|
+
const actions = stages.map((name) => ({
|
|
14
|
+
type: 'ActionDefinition' as const,
|
|
15
|
+
name,
|
|
16
|
+
steps: [],
|
|
17
|
+
}));
|
|
18
|
+
|
|
19
|
+
const pipeline = {
|
|
20
|
+
type: 'PipelineDefinition' as const,
|
|
21
|
+
stages: stages.map((action) => ({ action })),
|
|
22
|
+
};
|
|
23
|
+
|
|
24
|
+
const mission: MissionDefinition = {
|
|
25
|
+
type: 'MissionDefinition',
|
|
26
|
+
name: 'TestMission',
|
|
27
|
+
sources: [],
|
|
28
|
+
stores: [],
|
|
29
|
+
schemas: [],
|
|
30
|
+
actions,
|
|
31
|
+
pipeline,
|
|
32
|
+
};
|
|
33
|
+
|
|
34
|
+
return {
|
|
35
|
+
type: 'ReqonProgram',
|
|
36
|
+
statements: [mission],
|
|
37
|
+
};
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
describe('Progress Callbacks', () => {
|
|
41
|
+
describe('onExecutionStart', () => {
|
|
42
|
+
it('is called when execution begins', async () => {
|
|
43
|
+
const onExecutionStart = vi.fn();
|
|
44
|
+
const executor = new MissionExecutor({
|
|
45
|
+
progress: { onExecutionStart },
|
|
46
|
+
});
|
|
47
|
+
|
|
48
|
+
await executor.execute(createTestProgram(['StepA', 'StepB']));
|
|
49
|
+
|
|
50
|
+
expect(onExecutionStart).toHaveBeenCalledOnce();
|
|
51
|
+
expect(onExecutionStart).toHaveBeenCalledWith(
|
|
52
|
+
expect.objectContaining({
|
|
53
|
+
mission: 'TestMission',
|
|
54
|
+
stageCount: 2,
|
|
55
|
+
isResume: false,
|
|
56
|
+
})
|
|
57
|
+
);
|
|
58
|
+
});
|
|
59
|
+
|
|
60
|
+
it('includes executionId from ephemeral execution', async () => {
|
|
61
|
+
const onExecutionStart = vi.fn();
|
|
62
|
+
const executor = new MissionExecutor({
|
|
63
|
+
progress: { onExecutionStart },
|
|
64
|
+
});
|
|
65
|
+
|
|
66
|
+
await executor.execute(createTestProgram(['StepA']));
|
|
67
|
+
|
|
68
|
+
expect(onExecutionStart).toHaveBeenCalledWith(
|
|
69
|
+
expect.objectContaining({
|
|
70
|
+
executionId: 'ephemeral',
|
|
71
|
+
})
|
|
72
|
+
);
|
|
73
|
+
});
|
|
74
|
+
|
|
75
|
+
it('includes executionId when persistence enabled', async () => {
|
|
76
|
+
const onExecutionStart = vi.fn();
|
|
77
|
+
const executor = new MissionExecutor({
|
|
78
|
+
persistState: true,
|
|
79
|
+
dataDir: '.reqon-test-progress',
|
|
80
|
+
progress: { onExecutionStart },
|
|
81
|
+
});
|
|
82
|
+
|
|
83
|
+
await executor.execute(createTestProgram(['StepA']));
|
|
84
|
+
|
|
85
|
+
expect(onExecutionStart).toHaveBeenCalledWith(
|
|
86
|
+
expect.objectContaining({
|
|
87
|
+
executionId: expect.stringMatching(/^exec_/),
|
|
88
|
+
})
|
|
89
|
+
);
|
|
90
|
+
});
|
|
91
|
+
|
|
92
|
+
it('includes metadata if provided', async () => {
|
|
93
|
+
const onExecutionStart = vi.fn();
|
|
94
|
+
const executor = new MissionExecutor({
|
|
95
|
+
metadata: { tenant: 'acme', userId: '123' },
|
|
96
|
+
progress: { onExecutionStart },
|
|
97
|
+
});
|
|
98
|
+
|
|
99
|
+
await executor.execute(createTestProgram(['StepA']));
|
|
100
|
+
|
|
101
|
+
expect(onExecutionStart).toHaveBeenCalledWith(
|
|
102
|
+
expect.objectContaining({
|
|
103
|
+
metadata: { tenant: 'acme', userId: '123' },
|
|
104
|
+
})
|
|
105
|
+
);
|
|
106
|
+
});
|
|
107
|
+
});
|
|
108
|
+
|
|
109
|
+
describe('onExecutionComplete', () => {
|
|
110
|
+
it('is called when execution finishes successfully', async () => {
|
|
111
|
+
const onExecutionComplete = vi.fn();
|
|
112
|
+
const executor = new MissionExecutor({
|
|
113
|
+
progress: { onExecutionComplete },
|
|
114
|
+
});
|
|
115
|
+
|
|
116
|
+
await executor.execute(createTestProgram(['StepA', 'StepB']));
|
|
117
|
+
|
|
118
|
+
expect(onExecutionComplete).toHaveBeenCalledOnce();
|
|
119
|
+
expect(onExecutionComplete).toHaveBeenCalledWith(
|
|
120
|
+
expect.objectContaining({
|
|
121
|
+
mission: 'TestMission',
|
|
122
|
+
success: true,
|
|
123
|
+
stagesCompleted: 2,
|
|
124
|
+
stagesFailed: 0,
|
|
125
|
+
errors: [],
|
|
126
|
+
})
|
|
127
|
+
);
|
|
128
|
+
});
|
|
129
|
+
|
|
130
|
+
it('includes duration', async () => {
|
|
131
|
+
const onExecutionComplete = vi.fn();
|
|
132
|
+
const executor = new MissionExecutor({
|
|
133
|
+
progress: { onExecutionComplete },
|
|
134
|
+
});
|
|
135
|
+
|
|
136
|
+
await executor.execute(createTestProgram(['StepA']));
|
|
137
|
+
|
|
138
|
+
const event = onExecutionComplete.mock.calls[0][0] as ExecutionCompleteEvent;
|
|
139
|
+
expect(event.duration).toBeGreaterThanOrEqual(0);
|
|
140
|
+
});
|
|
141
|
+
});
|
|
142
|
+
|
|
143
|
+
describe('onStageStart', () => {
|
|
144
|
+
it('is called for each stage', async () => {
|
|
145
|
+
const onStageStart = vi.fn();
|
|
146
|
+
const executor = new MissionExecutor({
|
|
147
|
+
progress: { onStageStart },
|
|
148
|
+
});
|
|
149
|
+
|
|
150
|
+
await executor.execute(createTestProgram(['StepA', 'StepB', 'StepC']));
|
|
151
|
+
|
|
152
|
+
expect(onStageStart).toHaveBeenCalledTimes(3);
|
|
153
|
+
});
|
|
154
|
+
|
|
155
|
+
it('includes stage index and name', async () => {
|
|
156
|
+
const events: StageStartEvent[] = [];
|
|
157
|
+
const executor = new MissionExecutor({
|
|
158
|
+
progress: {
|
|
159
|
+
onStageStart: (e) => events.push(e),
|
|
160
|
+
},
|
|
161
|
+
});
|
|
162
|
+
|
|
163
|
+
await executor.execute(createTestProgram(['Fetch', 'Process', 'Store']));
|
|
164
|
+
|
|
165
|
+
expect(events[0]).toMatchObject({
|
|
166
|
+
stageIndex: 0,
|
|
167
|
+
stageName: 'Fetch',
|
|
168
|
+
totalStages: 3,
|
|
169
|
+
});
|
|
170
|
+
expect(events[1]).toMatchObject({
|
|
171
|
+
stageIndex: 1,
|
|
172
|
+
stageName: 'Process',
|
|
173
|
+
totalStages: 3,
|
|
174
|
+
});
|
|
175
|
+
expect(events[2]).toMatchObject({
|
|
176
|
+
stageIndex: 2,
|
|
177
|
+
stageName: 'Store',
|
|
178
|
+
totalStages: 3,
|
|
179
|
+
});
|
|
180
|
+
});
|
|
181
|
+
});
|
|
182
|
+
|
|
183
|
+
describe('onStageComplete', () => {
|
|
184
|
+
it('is called for each stage', async () => {
|
|
185
|
+
const onStageComplete = vi.fn();
|
|
186
|
+
const executor = new MissionExecutor({
|
|
187
|
+
progress: { onStageComplete },
|
|
188
|
+
});
|
|
189
|
+
|
|
190
|
+
await executor.execute(createTestProgram(['StepA', 'StepB']));
|
|
191
|
+
|
|
192
|
+
expect(onStageComplete).toHaveBeenCalledTimes(2);
|
|
193
|
+
});
|
|
194
|
+
|
|
195
|
+
it('includes success status and duration', async () => {
|
|
196
|
+
const events: StageCompleteEvent[] = [];
|
|
197
|
+
const executor = new MissionExecutor({
|
|
198
|
+
progress: {
|
|
199
|
+
onStageComplete: (e) => events.push(e),
|
|
200
|
+
},
|
|
201
|
+
});
|
|
202
|
+
|
|
203
|
+
await executor.execute(createTestProgram(['StepA']));
|
|
204
|
+
|
|
205
|
+
expect(events[0]).toMatchObject({
|
|
206
|
+
stageIndex: 0,
|
|
207
|
+
stageName: 'StepA',
|
|
208
|
+
success: true,
|
|
209
|
+
});
|
|
210
|
+
expect(events[0].duration).toBeGreaterThanOrEqual(0);
|
|
211
|
+
expect(events[0].error).toBeUndefined();
|
|
212
|
+
});
|
|
213
|
+
});
|
|
214
|
+
|
|
215
|
+
describe('callback order', () => {
|
|
216
|
+
it('fires callbacks in correct order', async () => {
|
|
217
|
+
const callOrder: string[] = [];
|
|
218
|
+
|
|
219
|
+
const executor = new MissionExecutor({
|
|
220
|
+
progress: {
|
|
221
|
+
onExecutionStart: () => callOrder.push('execStart'),
|
|
222
|
+
onExecutionComplete: () => callOrder.push('execComplete'),
|
|
223
|
+
onStageStart: (e) => callOrder.push(`stageStart:${e.stageName}`),
|
|
224
|
+
onStageComplete: (e) => callOrder.push(`stageComplete:${e.stageName}`),
|
|
225
|
+
},
|
|
226
|
+
});
|
|
227
|
+
|
|
228
|
+
await executor.execute(createTestProgram(['A', 'B']));
|
|
229
|
+
|
|
230
|
+
expect(callOrder).toEqual([
|
|
231
|
+
'execStart',
|
|
232
|
+
'stageStart:A',
|
|
233
|
+
'stageComplete:A',
|
|
234
|
+
'stageStart:B',
|
|
235
|
+
'stageComplete:B',
|
|
236
|
+
'execComplete',
|
|
237
|
+
]);
|
|
238
|
+
});
|
|
239
|
+
});
|
|
240
|
+
|
|
241
|
+
describe('real-time UI example', () => {
|
|
242
|
+
it('can build a progress tracker', async () => {
|
|
243
|
+
const progressLog: string[] = [];
|
|
244
|
+
|
|
245
|
+
const executor = new MissionExecutor({
|
|
246
|
+
progress: {
|
|
247
|
+
onExecutionStart: (e) => {
|
|
248
|
+
progressLog.push(`Starting ${e.mission} (${e.stageCount} stages)`);
|
|
249
|
+
},
|
|
250
|
+
onStageStart: (e) => {
|
|
251
|
+
const pct = Math.round((e.stageIndex / e.totalStages) * 100);
|
|
252
|
+
progressLog.push(`[${pct}%] Running ${e.stageName}...`);
|
|
253
|
+
},
|
|
254
|
+
onStageComplete: (e) => {
|
|
255
|
+
const pct = Math.round(((e.stageIndex + 1) / e.totalStages) * 100);
|
|
256
|
+
progressLog.push(`[${pct}%] ${e.stageName} ${e.success ? 'done' : 'failed'}`);
|
|
257
|
+
},
|
|
258
|
+
onExecutionComplete: (e) => {
|
|
259
|
+
progressLog.push(`Finished: ${e.success ? 'SUCCESS' : 'FAILED'} in ${e.duration}ms`);
|
|
260
|
+
},
|
|
261
|
+
},
|
|
262
|
+
});
|
|
263
|
+
|
|
264
|
+
await executor.execute(createTestProgram(['Fetch', 'Process', 'Store']));
|
|
265
|
+
|
|
266
|
+
expect(progressLog[0]).toBe('Starting TestMission (3 stages)');
|
|
267
|
+
expect(progressLog[1]).toBe('[0%] Running Fetch...');
|
|
268
|
+
expect(progressLog[2]).toBe('[33%] Fetch done');
|
|
269
|
+
expect(progressLog[3]).toBe('[33%] Running Process...');
|
|
270
|
+
expect(progressLog[4]).toBe('[67%] Process done');
|
|
271
|
+
expect(progressLog[5]).toBe('[67%] Running Store...');
|
|
272
|
+
expect(progressLog[6]).toBe('[100%] Store done');
|
|
273
|
+
expect(progressLog[7]).toMatch(/^Finished: SUCCESS in \d+ms$/);
|
|
274
|
+
});
|
|
275
|
+
});
|
|
276
|
+
});
|
|
@@ -0,0 +1,160 @@
|
|
|
1
|
+
import { describe, it, expect } from 'vitest';
|
|
2
|
+
import { matchesSchema, findMatchingSchema } from './schema-matcher.js';
|
|
3
|
+
import type { SchemaDefinition } from 'vague-lang';
|
|
4
|
+
|
|
5
|
+
describe('Schema Matcher', () => {
|
|
6
|
+
describe('matchesSchema', () => {
|
|
7
|
+
const userSchema: SchemaDefinition = {
|
|
8
|
+
type: 'SchemaDefinition',
|
|
9
|
+
name: 'User',
|
|
10
|
+
fields: [
|
|
11
|
+
{
|
|
12
|
+
type: 'FieldDefinition',
|
|
13
|
+
name: 'id',
|
|
14
|
+
fieldType: { type: 'PrimitiveType', name: 'int' },
|
|
15
|
+
},
|
|
16
|
+
{
|
|
17
|
+
type: 'FieldDefinition',
|
|
18
|
+
name: 'name',
|
|
19
|
+
fieldType: { type: 'PrimitiveType', name: 'string' },
|
|
20
|
+
},
|
|
21
|
+
{
|
|
22
|
+
type: 'FieldDefinition',
|
|
23
|
+
name: 'email',
|
|
24
|
+
fieldType: { type: 'PrimitiveType', name: 'string' },
|
|
25
|
+
optional: true,
|
|
26
|
+
},
|
|
27
|
+
],
|
|
28
|
+
};
|
|
29
|
+
|
|
30
|
+
it('matches valid object with all required fields', () => {
|
|
31
|
+
const value = { id: 1, name: 'Alice' };
|
|
32
|
+
expect(matchesSchema(value, userSchema)).toBe(true);
|
|
33
|
+
});
|
|
34
|
+
|
|
35
|
+
it('matches valid object with optional field', () => {
|
|
36
|
+
const value = { id: 1, name: 'Alice', email: 'alice@example.com' };
|
|
37
|
+
expect(matchesSchema(value, userSchema)).toBe(true);
|
|
38
|
+
});
|
|
39
|
+
|
|
40
|
+
it('matches object with extra fields (open schema)', () => {
|
|
41
|
+
const value = { id: 1, name: 'Alice', extra: 'ignored' };
|
|
42
|
+
expect(matchesSchema(value, userSchema)).toBe(true);
|
|
43
|
+
});
|
|
44
|
+
|
|
45
|
+
it('rejects object missing required field', () => {
|
|
46
|
+
const value = { id: 1 }; // missing 'name'
|
|
47
|
+
expect(matchesSchema(value, userSchema)).toBe(false);
|
|
48
|
+
});
|
|
49
|
+
|
|
50
|
+
it('rejects wrong type for field', () => {
|
|
51
|
+
const value = { id: 'not-a-number', name: 'Alice' };
|
|
52
|
+
expect(matchesSchema(value, userSchema)).toBe(false);
|
|
53
|
+
});
|
|
54
|
+
|
|
55
|
+
it('rejects null value', () => {
|
|
56
|
+
expect(matchesSchema(null, userSchema)).toBe(false);
|
|
57
|
+
});
|
|
58
|
+
|
|
59
|
+
it('rejects primitive value', () => {
|
|
60
|
+
expect(matchesSchema('string', userSchema)).toBe(false);
|
|
61
|
+
});
|
|
62
|
+
|
|
63
|
+
it('allows null for optional fields', () => {
|
|
64
|
+
const value = { id: 1, name: 'Alice', email: null };
|
|
65
|
+
expect(matchesSchema(value, userSchema)).toBe(true);
|
|
66
|
+
});
|
|
67
|
+
});
|
|
68
|
+
|
|
69
|
+
describe('findMatchingSchema', () => {
|
|
70
|
+
const successSchema: SchemaDefinition = {
|
|
71
|
+
type: 'SchemaDefinition',
|
|
72
|
+
name: 'SuccessResponse',
|
|
73
|
+
fields: [
|
|
74
|
+
{
|
|
75
|
+
type: 'FieldDefinition',
|
|
76
|
+
name: 'data',
|
|
77
|
+
fieldType: { type: 'ReferenceType', path: { type: 'QualifiedName', parts: ['object'] } },
|
|
78
|
+
},
|
|
79
|
+
{
|
|
80
|
+
type: 'FieldDefinition',
|
|
81
|
+
name: 'status',
|
|
82
|
+
fieldType: { type: 'PrimitiveType', name: 'string' },
|
|
83
|
+
},
|
|
84
|
+
],
|
|
85
|
+
};
|
|
86
|
+
|
|
87
|
+
const errorSchema: SchemaDefinition = {
|
|
88
|
+
type: 'SchemaDefinition',
|
|
89
|
+
name: 'ErrorResponse',
|
|
90
|
+
fields: [
|
|
91
|
+
{
|
|
92
|
+
type: 'FieldDefinition',
|
|
93
|
+
name: 'error',
|
|
94
|
+
fieldType: { type: 'PrimitiveType', name: 'string' },
|
|
95
|
+
},
|
|
96
|
+
{
|
|
97
|
+
type: 'FieldDefinition',
|
|
98
|
+
name: 'code',
|
|
99
|
+
fieldType: { type: 'PrimitiveType', name: 'int' },
|
|
100
|
+
},
|
|
101
|
+
],
|
|
102
|
+
};
|
|
103
|
+
|
|
104
|
+
const schemas = new Map<string, SchemaDefinition>([
|
|
105
|
+
['SuccessResponse', successSchema],
|
|
106
|
+
['ErrorResponse', errorSchema],
|
|
107
|
+
]);
|
|
108
|
+
|
|
109
|
+
it('finds matching schema in order', () => {
|
|
110
|
+
const successValue = { data: { id: 1 }, status: 'ok' };
|
|
111
|
+
const result = findMatchingSchema(
|
|
112
|
+
successValue,
|
|
113
|
+
schemas,
|
|
114
|
+
['SuccessResponse', 'ErrorResponse']
|
|
115
|
+
);
|
|
116
|
+
expect(result).toBe('SuccessResponse');
|
|
117
|
+
});
|
|
118
|
+
|
|
119
|
+
it('finds second schema if first does not match', () => {
|
|
120
|
+
const errorValue = { error: 'Not found', code: 404 };
|
|
121
|
+
const result = findMatchingSchema(
|
|
122
|
+
errorValue,
|
|
123
|
+
schemas,
|
|
124
|
+
['SuccessResponse', 'ErrorResponse']
|
|
125
|
+
);
|
|
126
|
+
expect(result).toBe('ErrorResponse');
|
|
127
|
+
});
|
|
128
|
+
|
|
129
|
+
it('returns undefined if no schema matches', () => {
|
|
130
|
+
const value = { random: 'value' };
|
|
131
|
+
const result = findMatchingSchema(
|
|
132
|
+
value,
|
|
133
|
+
schemas,
|
|
134
|
+
['SuccessResponse', 'ErrorResponse']
|
|
135
|
+
);
|
|
136
|
+
expect(result).toBeUndefined();
|
|
137
|
+
});
|
|
138
|
+
|
|
139
|
+
it('handles wildcard pattern', () => {
|
|
140
|
+
const value = { random: 'value' };
|
|
141
|
+
const result = findMatchingSchema(
|
|
142
|
+
value,
|
|
143
|
+
schemas,
|
|
144
|
+
['SuccessResponse', 'ErrorResponse', '_']
|
|
145
|
+
);
|
|
146
|
+
expect(result).toBe('_');
|
|
147
|
+
});
|
|
148
|
+
|
|
149
|
+
it('respects schema order (first match wins)', () => {
|
|
150
|
+
// Create a schema that matches both
|
|
151
|
+
const ambiguousValue = { data: {}, status: 'ok', error: 'msg', code: 500 };
|
|
152
|
+
const result = findMatchingSchema(
|
|
153
|
+
ambiguousValue,
|
|
154
|
+
schemas,
|
|
155
|
+
['SuccessResponse', 'ErrorResponse']
|
|
156
|
+
);
|
|
157
|
+
expect(result).toBe('SuccessResponse'); // First match wins
|
|
158
|
+
});
|
|
159
|
+
});
|
|
160
|
+
});
|
|
@@ -0,0 +1,168 @@
|
|
|
1
|
+
import type { SchemaDefinition, FieldDefinition, FieldType } from 'vague-lang';
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* Check if a value matches a schema definition.
|
|
5
|
+
*
|
|
6
|
+
* Matching rules:
|
|
7
|
+
* - All required fields (non-optional) must be present
|
|
8
|
+
* - Field types must match (string, int, decimal, boolean, date)
|
|
9
|
+
* - Extra fields are allowed (open schema)
|
|
10
|
+
* - Nested objects are not deeply validated (future enhancement)
|
|
11
|
+
*/
|
|
12
|
+
export function matchesSchema(value: unknown, schema: SchemaDefinition): boolean {
|
|
13
|
+
if (typeof value !== 'object' || value === null) {
|
|
14
|
+
return false;
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
const obj = value as Record<string, unknown>;
|
|
18
|
+
|
|
19
|
+
for (const field of schema.fields) {
|
|
20
|
+
const fieldValue = obj[field.name];
|
|
21
|
+
const isOptional = field.optional === true;
|
|
22
|
+
|
|
23
|
+
// Required field must be present
|
|
24
|
+
if (fieldValue === undefined) {
|
|
25
|
+
if (!isOptional) {
|
|
26
|
+
return false;
|
|
27
|
+
}
|
|
28
|
+
continue;
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
// Null is allowed for optional fields
|
|
32
|
+
if (fieldValue === null && isOptional) {
|
|
33
|
+
continue;
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
// Check type if present
|
|
37
|
+
if (!matchesFieldType(fieldValue, field.fieldType)) {
|
|
38
|
+
return false;
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
return true;
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
/**
|
|
46
|
+
* Check if a value matches the expected field type
|
|
47
|
+
*/
|
|
48
|
+
function matchesFieldType(value: unknown, fieldType: FieldType): boolean {
|
|
49
|
+
// Handle primitive types
|
|
50
|
+
if (fieldType.type === 'PrimitiveType') {
|
|
51
|
+
return matchesPrimitiveType(value, fieldType.name);
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
// Handle collection types (arrays)
|
|
55
|
+
if (fieldType.type === 'CollectionType') {
|
|
56
|
+
if (!Array.isArray(value)) {
|
|
57
|
+
return false;
|
|
58
|
+
}
|
|
59
|
+
// For now, don't validate element types
|
|
60
|
+
return true;
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
// Handle object/reference types
|
|
64
|
+
if (fieldType.type === 'ReferenceType') {
|
|
65
|
+
// For now, just check it's an object
|
|
66
|
+
return typeof value === 'object' && value !== null;
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
// Handle superposition types (unions) - any option matching is ok
|
|
70
|
+
if (fieldType.type === 'SuperpositionType') {
|
|
71
|
+
// Would need to check each option - be permissive for now
|
|
72
|
+
return true;
|
|
73
|
+
}
|
|
74
|
+
|
|
75
|
+
// Handle generator types (faker, etc.) - can't validate statically
|
|
76
|
+
if (fieldType.type === 'GeneratorType') {
|
|
77
|
+
return true;
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
// Handle expression types - can't validate statically
|
|
81
|
+
if (fieldType.type === 'ExpressionType') {
|
|
82
|
+
return true;
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
// Handle range types - check it's a number in range
|
|
86
|
+
if (fieldType.type === 'RangeType') {
|
|
87
|
+
return typeof value === 'number';
|
|
88
|
+
}
|
|
89
|
+
|
|
90
|
+
// Handle ordered sequence types (tuples)
|
|
91
|
+
if (fieldType.type === 'OrderedSequenceType') {
|
|
92
|
+
return Array.isArray(value);
|
|
93
|
+
}
|
|
94
|
+
|
|
95
|
+
// Unknown type - be permissive
|
|
96
|
+
return true;
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
/**
|
|
100
|
+
* Check if a value matches a primitive type
|
|
101
|
+
*/
|
|
102
|
+
function matchesPrimitiveType(value: unknown, typeName: string): boolean {
|
|
103
|
+
|
|
104
|
+
switch (typeName) {
|
|
105
|
+
case 'string':
|
|
106
|
+
return typeof value === 'string';
|
|
107
|
+
|
|
108
|
+
case 'int':
|
|
109
|
+
case 'integer':
|
|
110
|
+
return typeof value === 'number' && Number.isInteger(value);
|
|
111
|
+
|
|
112
|
+
case 'decimal':
|
|
113
|
+
case 'number':
|
|
114
|
+
case 'float':
|
|
115
|
+
case 'double':
|
|
116
|
+
return typeof value === 'number';
|
|
117
|
+
|
|
118
|
+
case 'boolean':
|
|
119
|
+
case 'bool':
|
|
120
|
+
return typeof value === 'boolean';
|
|
121
|
+
|
|
122
|
+
case 'date':
|
|
123
|
+
case 'datetime':
|
|
124
|
+
// Accept strings (ISO format) or Date objects
|
|
125
|
+
if (value instanceof Date) return true;
|
|
126
|
+
if (typeof value === 'string') {
|
|
127
|
+
const parsed = Date.parse(value);
|
|
128
|
+
return !isNaN(parsed);
|
|
129
|
+
}
|
|
130
|
+
return false;
|
|
131
|
+
|
|
132
|
+
case 'any':
|
|
133
|
+
return true;
|
|
134
|
+
|
|
135
|
+
default:
|
|
136
|
+
// Unknown primitive - be permissive
|
|
137
|
+
return true;
|
|
138
|
+
}
|
|
139
|
+
}
|
|
140
|
+
|
|
141
|
+
/**
|
|
142
|
+
* Find the first matching schema from a list.
|
|
143
|
+
* Returns the schema name or undefined if no match.
|
|
144
|
+
*/
|
|
145
|
+
export function findMatchingSchema(
|
|
146
|
+
value: unknown,
|
|
147
|
+
schemas: Map<string, SchemaDefinition>,
|
|
148
|
+
schemaNames: string[]
|
|
149
|
+
): string | undefined {
|
|
150
|
+
for (const name of schemaNames) {
|
|
151
|
+
// Handle wildcard
|
|
152
|
+
if (name === '_') {
|
|
153
|
+
return '_';
|
|
154
|
+
}
|
|
155
|
+
|
|
156
|
+
const schema = schemas.get(name);
|
|
157
|
+
if (!schema) {
|
|
158
|
+
// Schema not found - skip
|
|
159
|
+
continue;
|
|
160
|
+
}
|
|
161
|
+
|
|
162
|
+
if (matchesSchema(value, schema)) {
|
|
163
|
+
return name;
|
|
164
|
+
}
|
|
165
|
+
}
|
|
166
|
+
|
|
167
|
+
return undefined;
|
|
168
|
+
}
|
|
@@ -0,0 +1,73 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Execution control flow signals
|
|
3
|
+
*
|
|
4
|
+
* These are used for non-exceptional control flow during pipeline execution.
|
|
5
|
+
* They extend Error for easy stack-based propagation but represent expected
|
|
6
|
+
* flow control, not actual errors.
|
|
7
|
+
*/
|
|
8
|
+
|
|
9
|
+
/**
|
|
10
|
+
* Signal thrown when a match arm triggers skip
|
|
11
|
+
*/
|
|
12
|
+
export class SkipSignal extends Error {
|
|
13
|
+
constructor() {
|
|
14
|
+
super('Skip remaining steps');
|
|
15
|
+
this.name = 'SkipSignal';
|
|
16
|
+
}
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
/**
|
|
20
|
+
* Signal thrown when a match arm triggers retry
|
|
21
|
+
*/
|
|
22
|
+
export class RetrySignal extends Error {
|
|
23
|
+
constructor(public backoff?: { maxAttempts: number; backoff: string; initialDelay: number }) {
|
|
24
|
+
super('Retry action');
|
|
25
|
+
this.name = 'RetrySignal';
|
|
26
|
+
}
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
/**
|
|
30
|
+
* Signal thrown when a match arm triggers jump
|
|
31
|
+
*/
|
|
32
|
+
export class JumpSignal extends Error {
|
|
33
|
+
constructor(
|
|
34
|
+
public action: string,
|
|
35
|
+
public then?: 'retry' | 'continue'
|
|
36
|
+
) {
|
|
37
|
+
super(`Jump to action: ${action}`);
|
|
38
|
+
this.name = 'JumpSignal';
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
/**
|
|
43
|
+
* Signal thrown when a match arm triggers queue
|
|
44
|
+
*/
|
|
45
|
+
export class QueueSignal extends Error {
|
|
46
|
+
constructor(
|
|
47
|
+
public value: unknown,
|
|
48
|
+
public target?: string
|
|
49
|
+
) {
|
|
50
|
+
super('Queue for later processing');
|
|
51
|
+
this.name = 'QueueSignal';
|
|
52
|
+
}
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
/**
|
|
56
|
+
* Error thrown when a match step has no matching arm
|
|
57
|
+
*/
|
|
58
|
+
export class NoMatchError extends Error {
|
|
59
|
+
constructor(public value: unknown) {
|
|
60
|
+
super('No matching schema found for response');
|
|
61
|
+
this.name = 'NoMatchError';
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
/**
|
|
66
|
+
* Error thrown when a match arm triggers an abort
|
|
67
|
+
*/
|
|
68
|
+
export class AbortError extends Error {
|
|
69
|
+
constructor(message?: string) {
|
|
70
|
+
super(message ?? 'Execution aborted');
|
|
71
|
+
this.name = 'AbortError';
|
|
72
|
+
}
|
|
73
|
+
}
|