smol-symphony 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/AGENTS.md +41 -22
- package/DESIGN.md +494 -273
- package/README.md +109 -57
- package/SPEC.md +33 -24
- package/WORKFLOW.minimal.yaml +34 -0
- package/{WORKFLOW.template.md → WORKFLOW.template.yaml} +409 -256
- package/WORKFLOW.yaml +487 -0
- package/assets/skills/symphony-issues/SKILL.md +136 -0
- package/assets/symphony-mise.system.toml +68 -0
- package/dist/src/bin/symphony.js +30 -0
- package/dist/src/bin/symphony.js.map +1 -0
- package/dist/src/core/actions/context.js +109 -0
- package/dist/src/core/actions/context.js.map +1 -0
- package/dist/{actions/parsing.js → src/core/actions/parse.js} +33 -114
- package/dist/src/core/actions/parse.js.map +1 -0
- package/dist/src/core/actions/plan.js +197 -0
- package/dist/src/core/actions/plan.js.map +1 -0
- package/dist/src/core/actions/predicates.js +111 -0
- package/dist/src/core/actions/predicates.js.map +1 -0
- package/dist/src/core/actions/run-fold.js +248 -0
- package/dist/src/core/actions/run-fold.js.map +1 -0
- package/dist/src/core/actions/template.js +118 -0
- package/dist/src/core/actions/template.js.map +1 -0
- package/dist/src/core/cli/args.js +116 -0
- package/dist/src/core/cli/args.js.map +1 -0
- package/dist/src/core/coerce.js +75 -0
- package/dist/src/core/coerce.js.map +1 -0
- package/dist/src/core/credential/account-id.js +20 -0
- package/dist/src/core/credential/account-id.js.map +1 -0
- package/dist/src/core/credential/adapter-config.js +136 -0
- package/dist/src/core/credential/adapter-config.js.map +1 -0
- package/dist/src/core/credential/availability.js +98 -0
- package/dist/src/core/credential/availability.js.map +1 -0
- package/dist/src/core/credential/extract.js +228 -0
- package/dist/src/core/credential/extract.js.map +1 -0
- package/dist/src/core/credential/fake-creds.js +171 -0
- package/dist/src/core/credential/fake-creds.js.map +1 -0
- package/dist/src/core/credential/identity.js +125 -0
- package/dist/src/core/credential/identity.js.map +1 -0
- package/dist/src/core/credential/shape.js +230 -0
- package/dist/src/core/credential/shape.js.map +1 -0
- package/dist/src/core/credential/strings.js +15 -0
- package/dist/src/core/credential/strings.js.map +1 -0
- package/dist/src/core/doctor/checks.js +303 -0
- package/dist/src/core/doctor/checks.js.map +1 -0
- package/dist/src/core/git/result.js +107 -0
- package/dist/src/core/git/result.js.map +1 -0
- package/dist/src/core/http/decisions.js +225 -0
- package/dist/src/core/http/decisions.js.map +1 -0
- package/dist/{http.js → src/core/http/render.js} +472 -738
- package/dist/src/core/http/render.js.map +1 -0
- package/dist/{http-handlers.js → src/core/http/routes.js} +52 -87
- package/dist/src/core/http/routes.js.map +1 -0
- package/dist/src/core/http/views.js +181 -0
- package/dist/src/core/http/views.js.map +1 -0
- package/dist/src/core/image/managed-image.js +95 -0
- package/dist/src/core/image/managed-image.js.map +1 -0
- package/dist/src/core/issue/file.js +149 -0
- package/dist/src/core/issue/file.js.map +1 -0
- package/dist/src/core/issue/parse.js +210 -0
- package/dist/src/core/issue/parse.js.map +1 -0
- package/dist/src/core/mcp/dispatch.js +239 -0
- package/dist/src/core/mcp/dispatch.js.map +1 -0
- package/dist/src/core/mcp/post-move.js +92 -0
- package/dist/src/core/mcp/post-move.js.map +1 -0
- package/dist/src/core/mcp/protocol.js +293 -0
- package/dist/src/core/mcp/protocol.js.map +1 -0
- package/dist/src/core/mcp/url.js +162 -0
- package/dist/src/core/mcp/url.js.map +1 -0
- package/dist/src/core/path.js +63 -0
- package/dist/src/core/path.js.map +1 -0
- package/dist/src/core/reconcile/image-decide.js +48 -0
- package/dist/src/core/reconcile/image-decide.js.map +1 -0
- package/dist/src/core/reconcile/ledger.js +142 -0
- package/dist/src/core/reconcile/ledger.js.map +1 -0
- package/dist/src/core/reconcile/pr-classify.js +62 -0
- package/dist/src/core/reconcile/pr-classify.js.map +1 -0
- package/dist/{reconciler → src/core/reconcile}/pr-decide.js +25 -12
- package/dist/src/core/reconcile/pr-decide.js.map +1 -0
- package/dist/src/core/reconcile/pr-loop.js +161 -0
- package/dist/src/core/reconcile/pr-loop.js.map +1 -0
- package/dist/src/core/reconcile/pr-notes.js +35 -0
- package/dist/src/core/reconcile/pr-notes.js.map +1 -0
- package/dist/src/core/reconcile/vm-decide.js +70 -0
- package/dist/src/core/reconcile/vm-decide.js.map +1 -0
- package/dist/src/core/reconcile/vm-reap.js +207 -0
- package/dist/src/core/reconcile/vm-reap.js.map +1 -0
- package/dist/src/core/reconcile/workspace-decide.js +162 -0
- package/dist/src/core/reconcile/workspace-decide.js.map +1 -0
- package/dist/src/core/runlog/summary.js +231 -0
- package/dist/src/core/runlog/summary.js.map +1 -0
- package/dist/src/core/runner/dispatch-config.js +95 -0
- package/dist/src/core/runner/dispatch-config.js.map +1 -0
- package/dist/src/core/runner/injection.js +61 -0
- package/dist/src/core/runner/injection.js.map +1 -0
- package/dist/src/core/runner/mise.js +210 -0
- package/dist/src/core/runner/mise.js.map +1 -0
- package/dist/src/core/runner/prompt.js +720 -0
- package/dist/src/core/runner/prompt.js.map +1 -0
- package/dist/src/core/runner/turn.js +242 -0
- package/dist/src/core/runner/turn.js.map +1 -0
- package/dist/src/core/runner/vm-plan.js +390 -0
- package/dist/src/core/runner/vm-plan.js.map +1 -0
- package/dist/src/core/schedule/admission.js +123 -0
- package/dist/src/core/schedule/admission.js.map +1 -0
- package/dist/src/core/schedule/circuit-breaker.js +111 -0
- package/dist/src/core/schedule/circuit-breaker.js.map +1 -0
- package/dist/src/core/schedule/eligibility.js +83 -0
- package/dist/src/core/schedule/eligibility.js.map +1 -0
- package/dist/src/core/schedule/reconcile-issue.js +82 -0
- package/dist/src/core/schedule/reconcile-issue.js.map +1 -0
- package/dist/src/core/schedule/retry.js +96 -0
- package/dist/src/core/schedule/retry.js.map +1 -0
- package/dist/src/core/schedule/sleep-cycle.js +133 -0
- package/dist/src/core/schedule/sleep-cycle.js.map +1 -0
- package/dist/src/core/schedule/slots.js +124 -0
- package/dist/src/core/schedule/slots.js.map +1 -0
- package/dist/src/core/schedule/tick.js +553 -0
- package/dist/src/core/schedule/tick.js.map +1 -0
- package/dist/src/core/schedule/token-fold.js +181 -0
- package/dist/src/core/schedule/token-fold.js.map +1 -0
- package/dist/src/core/state-resolve.js +86 -0
- package/dist/src/core/state-resolve.js.map +1 -0
- package/dist/src/core/vm-guards.js +278 -0
- package/dist/src/core/vm-guards.js.map +1 -0
- package/dist/src/core/workflow/derive.js +107 -0
- package/dist/src/core/workflow/derive.js.map +1 -0
- package/dist/src/core/workflow/parse.js +687 -0
- package/dist/src/core/workflow/parse.js.map +1 -0
- package/dist/src/core/workflow/prompt-probe.js +78 -0
- package/dist/src/core/workflow/prompt-probe.js.map +1 -0
- package/dist/src/core/workflow/validate.js +189 -0
- package/dist/src/core/workflow/validate.js.map +1 -0
- package/dist/src/core/workspace-key.js +19 -0
- package/dist/src/core/workspace-key.js.map +1 -0
- package/dist/src/shell/actions-runner.js +356 -0
- package/dist/src/shell/actions-runner.js.map +1 -0
- package/dist/src/shell/adapter/adapter-registry.js +45 -0
- package/dist/src/shell/adapter/adapter-registry.js.map +1 -0
- package/dist/src/shell/adapter/clock-random.js +96 -0
- package/dist/src/shell/adapter/clock-random.js.map +1 -0
- package/dist/src/shell/adapter/gondolin-dispatch-helpers.js +158 -0
- package/dist/src/shell/adapter/gondolin-dispatch-helpers.js.map +1 -0
- package/dist/src/shell/adapter/gondolin-dispatch.js +385 -0
- package/dist/src/shell/adapter/gondolin-dispatch.js.map +1 -0
- package/dist/src/shell/adapter/gondolin-image-converter.js +233 -0
- package/dist/src/shell/adapter/gondolin-image-converter.js.map +1 -0
- package/dist/src/shell/adapter/gondolin-image-fetch.js +180 -0
- package/dist/src/shell/adapter/gondolin-image-fetch.js.map +1 -0
- package/dist/src/shell/adapter/launcher-asset.js +57 -0
- package/dist/src/shell/adapter/launcher-asset.js.map +1 -0
- package/dist/src/shell/adapter/mise-config-asset.js +65 -0
- package/dist/src/shell/adapter/mise-config-asset.js.map +1 -0
- package/dist/src/shell/adapter/workflow-loader.js +304 -0
- package/dist/src/shell/adapter/workflow-loader.js.map +1 -0
- package/dist/src/shell/cli/doctor.js +268 -0
- package/dist/src/shell/cli/doctor.js.map +1 -0
- package/dist/src/shell/effect-interpreter-families.js +314 -0
- package/dist/src/shell/effect-interpreter-families.js.map +1 -0
- package/dist/src/shell/effect-interpreter.js +29 -0
- package/dist/src/shell/effect-interpreter.js.map +1 -0
- package/dist/src/shell/interp/acp-frame.js +137 -0
- package/dist/src/shell/interp/acp-frame.js.map +1 -0
- package/dist/src/shell/interp/acp-ws-conn.js +320 -0
- package/dist/src/shell/interp/acp-ws-conn.js.map +1 -0
- package/dist/src/shell/interp/acp-ws-frames.js +159 -0
- package/dist/src/shell/interp/acp-ws-frames.js.map +1 -0
- package/dist/src/shell/interp/acp-ws.js +197 -0
- package/dist/src/shell/interp/acp-ws.js.map +1 -0
- package/dist/src/shell/interp/acp.js +319 -0
- package/dist/src/shell/interp/acp.js.map +1 -0
- package/dist/src/shell/interp/credential-defaults.js +128 -0
- package/dist/src/shell/interp/credential-defaults.js.map +1 -0
- package/dist/src/shell/interp/credential-hooks.js +149 -0
- package/dist/src/shell/interp/credential-hooks.js.map +1 -0
- package/dist/src/shell/interp/credential-registry.js +226 -0
- package/dist/src/shell/interp/credential-registry.js.map +1 -0
- package/dist/src/shell/interp/credential.js +103 -0
- package/dist/src/shell/interp/credential.js.map +1 -0
- package/dist/src/shell/interp/gh.js +163 -0
- package/dist/src/shell/interp/gh.js.map +1 -0
- package/dist/src/shell/interp/git.js +28 -0
- package/dist/src/shell/interp/git.js.map +1 -0
- package/dist/src/shell/interp/log.js +213 -0
- package/dist/src/shell/interp/log.js.map +1 -0
- package/dist/src/shell/interp/process.js +178 -0
- package/dist/src/shell/interp/process.js.map +1 -0
- package/dist/src/shell/interp/runlog.js +193 -0
- package/dist/src/shell/interp/runlog.js.map +1 -0
- package/dist/src/shell/interp/timer.js +64 -0
- package/dist/src/shell/interp/timer.js.map +1 -0
- package/dist/src/shell/interp/tracker-disk.js +99 -0
- package/dist/src/shell/interp/tracker-disk.js.map +1 -0
- package/dist/src/shell/interp/tracker-parse.js +71 -0
- package/dist/src/shell/interp/tracker-parse.js.map +1 -0
- package/dist/src/shell/interp/tracker-scan.js +238 -0
- package/dist/src/shell/interp/tracker-scan.js.map +1 -0
- package/dist/src/shell/interp/tracker-write.js +91 -0
- package/dist/src/shell/interp/tracker-write.js.map +1 -0
- package/dist/src/shell/interp/tracker.js +41 -0
- package/dist/src/shell/interp/tracker.js.map +1 -0
- package/dist/src/shell/interp/tty.js +48 -0
- package/dist/src/shell/interp/tty.js.map +1 -0
- package/dist/src/shell/interp/vm.js +199 -0
- package/dist/src/shell/interp/vm.js.map +1 -0
- package/dist/src/shell/interp/workspace.js +310 -0
- package/dist/src/shell/interp/workspace.js.map +1 -0
- package/dist/src/shell/main-acp.js +78 -0
- package/dist/src/shell/main-acp.js.map +1 -0
- package/dist/src/shell/main-adapters.js +222 -0
- package/dist/src/shell/main-adapters.js.map +1 -0
- package/dist/src/shell/main-credential.js +122 -0
- package/dist/src/shell/main-credential.js.map +1 -0
- package/dist/src/shell/main-doctor.js +22 -0
- package/dist/src/shell/main-doctor.js.map +1 -0
- package/dist/src/shell/main-entry.js +46 -0
- package/dist/src/shell/main-entry.js.map +1 -0
- package/dist/src/shell/main-http-csrf.js +45 -0
- package/dist/src/shell/main-http-csrf.js.map +1 -0
- package/dist/src/shell/main-http-handler.js +389 -0
- package/dist/src/shell/main-http-handler.js.map +1 -0
- package/dist/src/shell/main-http-mcp.js +122 -0
- package/dist/src/shell/main-http-mcp.js.map +1 -0
- package/dist/src/shell/main-http-views.js +253 -0
- package/dist/src/shell/main-http-views.js.map +1 -0
- package/dist/src/shell/main-http.js +76 -0
- package/dist/src/shell/main-http.js.map +1 -0
- package/dist/src/shell/main-loops.js +130 -0
- package/dist/src/shell/main-loops.js.map +1 -0
- package/dist/src/shell/main-mcp.js +129 -0
- package/dist/src/shell/main-mcp.js.map +1 -0
- package/dist/src/shell/main-orchestrator.js +120 -0
- package/dist/src/shell/main-orchestrator.js.map +1 -0
- package/dist/src/shell/main-preflight.js +43 -0
- package/dist/src/shell/main-preflight.js.map +1 -0
- package/dist/src/shell/main-reconcilers-helpers.js +244 -0
- package/dist/src/shell/main-reconcilers-helpers.js.map +1 -0
- package/dist/src/shell/main-reconcilers-pr.js +148 -0
- package/dist/src/shell/main-reconcilers-pr.js.map +1 -0
- package/dist/src/shell/main-reconcilers.js +225 -0
- package/dist/src/shell/main-reconcilers.js.map +1 -0
- package/dist/src/shell/main-runner.js +355 -0
- package/dist/src/shell/main-runner.js.map +1 -0
- package/dist/src/shell/main-scaffold.js +116 -0
- package/dist/src/shell/main-scaffold.js.map +1 -0
- package/dist/src/shell/main-shutdown.js +115 -0
- package/dist/src/shell/main-shutdown.js.map +1 -0
- package/dist/src/shell/main-startup.js +48 -0
- package/dist/src/shell/main-startup.js.map +1 -0
- package/dist/src/shell/main-substrates.js +43 -0
- package/dist/src/shell/main-substrates.js.map +1 -0
- package/dist/src/shell/main.js +385 -0
- package/dist/src/shell/main.js.map +1 -0
- package/dist/src/shell/orchestrator-feedback.js +69 -0
- package/dist/src/shell/orchestrator-feedback.js.map +1 -0
- package/dist/src/shell/orchestrator-image.js +167 -0
- package/dist/src/shell/orchestrator-image.js.map +1 -0
- package/dist/src/shell/orchestrator-loop.js +468 -0
- package/dist/src/shell/orchestrator-loop.js.map +1 -0
- package/dist/src/shell/orchestrator-reconcile.js +36 -0
- package/dist/src/shell/orchestrator-reconcile.js.map +1 -0
- package/dist/src/shell/reconciler-loop.js +228 -0
- package/dist/src/shell/reconciler-loop.js.map +1 -0
- package/dist/src/shell/runner-loop-turn.js +301 -0
- package/dist/src/shell/runner-loop-turn.js.map +1 -0
- package/dist/src/shell/runner-loop.js +338 -0
- package/dist/src/shell/runner-loop.js.map +1 -0
- package/dist/src/shell/server/http.js +208 -0
- package/dist/src/shell/server/http.js.map +1 -0
- package/dist/src/shell/server/mcp-runtime-effects.js +237 -0
- package/dist/src/shell/server/mcp-runtime-effects.js.map +1 -0
- package/dist/src/shell/server/mcp-runtime.js +99 -0
- package/dist/src/shell/server/mcp-runtime.js.map +1 -0
- package/dist/src/shell/workspace-key.js +14 -0
- package/dist/src/shell/workspace-key.js.map +1 -0
- package/dist/src/types/acp.js +8 -0
- package/dist/src/types/acp.js.map +1 -0
- package/dist/src/types/actions/plan.js +6 -0
- package/dist/src/types/actions/plan.js.map +1 -0
- package/dist/src/types/actions/predicates.js +6 -0
- package/dist/src/types/actions/predicates.js.map +1 -0
- package/dist/src/types/actions/run-fold.js +8 -0
- package/dist/src/types/actions/run-fold.js.map +1 -0
- package/dist/src/types/actions.js +7 -0
- package/dist/src/types/actions.js.map +1 -0
- package/dist/src/types/adapter/clock-random.js +4 -0
- package/dist/src/types/adapter/clock-random.js.map +1 -0
- package/dist/src/types/adapter/gondolin-image-converter.js +5 -0
- package/dist/src/types/adapter/gondolin-image-converter.js.map +1 -0
- package/dist/src/types/adapter/gondolin-image-fetch.js +5 -0
- package/dist/src/types/adapter/gondolin-image-fetch.js.map +1 -0
- package/dist/src/types/adapter/workflow-loader.js +4 -0
- package/dist/src/types/adapter/workflow-loader.js.map +1 -0
- package/dist/src/types/cli/args.js +8 -0
- package/dist/src/types/cli/args.js.map +1 -0
- package/dist/src/types/config.js +8 -0
- package/dist/src/types/config.js.map +1 -0
- package/dist/src/types/credential-interp.js +6 -0
- package/dist/src/types/credential-interp.js.map +1 -0
- package/dist/src/types/credentials.js +10 -0
- package/dist/src/types/credentials.js.map +1 -0
- package/dist/src/types/doctor.js +7 -0
- package/dist/src/types/doctor.js.map +1 -0
- package/dist/src/types/domain.js +7 -0
- package/dist/src/types/domain.js.map +1 -0
- package/dist/src/types/effect.js +15 -0
- package/dist/src/types/effect.js.map +1 -0
- package/dist/src/types/errors.js +39 -0
- package/dist/src/types/errors.js.map +1 -0
- package/dist/src/types/http/decisions.js +6 -0
- package/dist/src/types/http/decisions.js.map +1 -0
- package/dist/src/types/http/render.js +10 -0
- package/dist/src/types/http/render.js.map +1 -0
- package/dist/src/types/http/views.js +6 -0
- package/dist/src/types/http/views.js.map +1 -0
- package/dist/src/types/http.js +9 -0
- package/dist/src/types/http.js.map +1 -0
- package/dist/src/types/image/managed-image.js +7 -0
- package/dist/src/types/image/managed-image.js.map +1 -0
- package/dist/src/types/interp/effect-interpreter.js +8 -0
- package/dist/src/types/interp/effect-interpreter.js.map +1 -0
- package/dist/src/types/interp/tracker.js +7 -0
- package/dist/src/types/interp/tracker.js.map +1 -0
- package/dist/src/types/issue/file.js +6 -0
- package/dist/src/types/issue/file.js.map +1 -0
- package/dist/src/types/issue/parse.js +8 -0
- package/dist/src/types/issue/parse.js.map +1 -0
- package/dist/src/types/main-acp.js +13 -0
- package/dist/src/types/main-acp.js.map +1 -0
- package/dist/src/types/main-adapters.js +5 -0
- package/dist/src/types/main-adapters.js.map +1 -0
- package/dist/src/types/main-credential.js +21 -0
- package/dist/src/types/main-credential.js.map +1 -0
- package/dist/src/types/main-doctor.js +6 -0
- package/dist/src/types/main-doctor.js.map +1 -0
- package/dist/src/types/main-http-handler.js +12 -0
- package/dist/src/types/main-http-handler.js.map +1 -0
- package/dist/src/types/main-http.js +5 -0
- package/dist/src/types/main-http.js.map +1 -0
- package/dist/src/types/main-loops.js +5 -0
- package/dist/src/types/main-loops.js.map +1 -0
- package/dist/src/types/main-mcp.js +12 -0
- package/dist/src/types/main-mcp.js.map +1 -0
- package/dist/src/types/main-orchestrator.js +5 -0
- package/dist/src/types/main-orchestrator.js.map +1 -0
- package/dist/src/types/main-reconcilers.js +11 -0
- package/dist/src/types/main-reconcilers.js.map +1 -0
- package/dist/src/types/main-runner.js +13 -0
- package/dist/src/types/main-runner.js.map +1 -0
- package/dist/src/types/main-startup.js +5 -0
- package/dist/src/types/main-startup.js.map +1 -0
- package/dist/src/types/main-substrates.js +5 -0
- package/dist/src/types/main-substrates.js.map +1 -0
- package/dist/src/types/mcp/dispatch.js +4 -0
- package/dist/src/types/mcp/dispatch.js.map +1 -0
- package/dist/src/types/mcp/post-move.js +7 -0
- package/dist/src/types/mcp/post-move.js.map +1 -0
- package/dist/src/types/mcp.js +9 -0
- package/dist/src/types/mcp.js.map +1 -0
- package/dist/src/types/ports.js +12 -0
- package/dist/src/types/ports.js.map +1 -0
- package/dist/src/types/reconcile/image-decide.js +5 -0
- package/dist/src/types/reconcile/image-decide.js.map +1 -0
- package/dist/src/types/reconcile/ledger.js +7 -0
- package/dist/src/types/reconcile/ledger.js.map +1 -0
- package/dist/src/types/reconcile/pr-loop.js +8 -0
- package/dist/src/types/reconcile/pr-loop.js.map +1 -0
- package/dist/src/types/reconcile/vm-reap.js +8 -0
- package/dist/src/types/reconcile/vm-reap.js.map +1 -0
- package/dist/src/types/reconcile/workspace-decide.js +7 -0
- package/dist/src/types/reconcile/workspace-decide.js.map +1 -0
- package/dist/src/types/reconcile.js +9 -0
- package/dist/src/types/reconcile.js.map +1 -0
- package/dist/src/types/runlog.js +7 -0
- package/dist/src/types/runlog.js.map +1 -0
- package/dist/src/types/runner/actions-runner.js +12 -0
- package/dist/src/types/runner/actions-runner.js.map +1 -0
- package/dist/src/types/runner/gondolin-dispatch.js +5 -0
- package/dist/src/types/runner/gondolin-dispatch.js.map +1 -0
- package/dist/src/types/runner/injection.js +6 -0
- package/dist/src/types/runner/injection.js.map +1 -0
- package/dist/src/types/runner/runner-loop.js +5 -0
- package/dist/src/types/runner/runner-loop.js.map +1 -0
- package/dist/src/types/runner/turn.js +4 -0
- package/dist/src/types/runner/turn.js.map +1 -0
- package/dist/src/types/runner/vm-plan.js +4 -0
- package/dist/src/types/runner/vm-plan.js.map +1 -0
- package/dist/src/types/runtime.js +9 -0
- package/dist/src/types/runtime.js.map +1 -0
- package/dist/src/types/schedule/admission.js +7 -0
- package/dist/src/types/schedule/admission.js.map +1 -0
- package/dist/src/types/schedule/circuit-breaker.js +2 -0
- package/dist/src/types/schedule/circuit-breaker.js.map +1 -0
- package/dist/src/types/schedule/eligibility.js +9 -0
- package/dist/src/types/schedule/eligibility.js.map +1 -0
- package/dist/src/types/schedule/orchestrator-loop.js +10 -0
- package/dist/src/types/schedule/orchestrator-loop.js.map +1 -0
- package/dist/src/types/schedule/sleep-cycle.js +4 -0
- package/dist/src/types/schedule/sleep-cycle.js.map +1 -0
- package/dist/src/types/schedule/slots.js +8 -0
- package/dist/src/types/schedule/slots.js.map +1 -0
- package/dist/src/types/schedule/tick.js +9 -0
- package/dist/src/types/schedule/tick.js.map +1 -0
- package/dist/src/types/server/mcp-runtime.js +8 -0
- package/dist/src/types/server/mcp-runtime.js.map +1 -0
- package/dist/src/types/workflow/parse.js +4 -0
- package/dist/src/types/workflow/parse.js.map +1 -0
- package/dist/tests/core/account-id.test.js +35 -0
- package/dist/tests/core/account-id.test.js.map +1 -0
- package/dist/tests/core/actions-parse.test.js +176 -0
- package/dist/tests/core/actions-parse.test.js.map +1 -0
- package/dist/tests/core/adapter-config.test.js +133 -0
- package/dist/tests/core/adapter-config.test.js.map +1 -0
- package/dist/tests/core/admission.test.js +215 -0
- package/dist/tests/core/admission.test.js.map +1 -0
- package/dist/tests/core/args.test.js +132 -0
- package/dist/tests/core/args.test.js.map +1 -0
- package/dist/tests/core/availability.test.js +62 -0
- package/dist/tests/core/availability.test.js.map +1 -0
- package/dist/tests/core/checks.test.js +395 -0
- package/dist/tests/core/checks.test.js.map +1 -0
- package/dist/tests/core/circuit-breaker.test.js +172 -0
- package/dist/tests/core/circuit-breaker.test.js.map +1 -0
- package/dist/tests/core/coerce.test.js +87 -0
- package/dist/tests/core/coerce.test.js.map +1 -0
- package/dist/tests/core/context.test.js +228 -0
- package/dist/tests/core/context.test.js.map +1 -0
- package/dist/tests/core/decisions.test.js +310 -0
- package/dist/tests/core/decisions.test.js.map +1 -0
- package/dist/tests/core/derive.test.js +205 -0
- package/dist/tests/core/derive.test.js.map +1 -0
- package/dist/tests/core/dispatch-config.test.js +164 -0
- package/dist/tests/core/dispatch-config.test.js.map +1 -0
- package/dist/tests/core/dispatch.test.js +302 -0
- package/dist/tests/core/dispatch.test.js.map +1 -0
- package/dist/tests/core/eligibility.test.js +163 -0
- package/dist/tests/core/eligibility.test.js.map +1 -0
- package/dist/tests/core/extract.test.js +139 -0
- package/dist/tests/core/extract.test.js.map +1 -0
- package/dist/tests/core/fake-creds.test.js +134 -0
- package/dist/tests/core/fake-creds.test.js.map +1 -0
- package/dist/tests/core/file.test.js +197 -0
- package/dist/tests/core/file.test.js.map +1 -0
- package/dist/tests/core/git-result.test.js +113 -0
- package/dist/tests/core/git-result.test.js.map +1 -0
- package/dist/tests/core/identity.test.js +180 -0
- package/dist/tests/core/identity.test.js.map +1 -0
- package/dist/tests/core/image-decide.test.js +59 -0
- package/dist/tests/core/image-decide.test.js.map +1 -0
- package/dist/tests/core/injection.test.js +163 -0
- package/dist/tests/core/injection.test.js.map +1 -0
- package/dist/tests/core/ledger.test.js +218 -0
- package/dist/tests/core/ledger.test.js.map +1 -0
- package/dist/tests/core/managed-image.test.js +68 -0
- package/dist/tests/core/managed-image.test.js.map +1 -0
- package/dist/tests/core/mise.test.js +138 -0
- package/dist/tests/core/mise.test.js.map +1 -0
- package/dist/tests/core/parse.test.js +174 -0
- package/dist/tests/core/parse.test.js.map +1 -0
- package/dist/tests/core/path.test.js +50 -0
- package/dist/tests/core/path.test.js.map +1 -0
- package/dist/tests/core/plan.test.js +218 -0
- package/dist/tests/core/plan.test.js.map +1 -0
- package/dist/tests/core/post-move.test.js +162 -0
- package/dist/tests/core/post-move.test.js.map +1 -0
- package/dist/tests/core/pr-classify.test.js +117 -0
- package/dist/tests/core/pr-classify.test.js.map +1 -0
- package/dist/tests/core/pr-decide.test.js +298 -0
- package/dist/tests/core/pr-decide.test.js.map +1 -0
- package/dist/tests/core/pr-loop.test.js +301 -0
- package/dist/tests/core/pr-loop.test.js.map +1 -0
- package/dist/tests/core/pr-notes.test.js +165 -0
- package/dist/tests/core/pr-notes.test.js.map +1 -0
- package/dist/tests/core/predicates.test.js +154 -0
- package/dist/tests/core/predicates.test.js.map +1 -0
- package/dist/tests/core/prompt.test.js +189 -0
- package/dist/tests/core/prompt.test.js.map +1 -0
- package/dist/tests/core/protocol.test.js +195 -0
- package/dist/tests/core/protocol.test.js.map +1 -0
- package/dist/tests/core/reconcile-issue.test.js +116 -0
- package/dist/tests/core/reconcile-issue.test.js.map +1 -0
- package/dist/tests/core/render.test.js +549 -0
- package/dist/tests/core/render.test.js.map +1 -0
- package/dist/tests/core/retry.test.js +186 -0
- package/dist/tests/core/retry.test.js.map +1 -0
- package/dist/tests/core/routes.test.js +247 -0
- package/dist/tests/core/routes.test.js.map +1 -0
- package/dist/tests/core/run-fold.test.js +299 -0
- package/dist/tests/core/run-fold.test.js.map +1 -0
- package/dist/tests/core/shape.test.js +185 -0
- package/dist/tests/core/shape.test.js.map +1 -0
- package/dist/tests/core/sleep-cycle.test.js +150 -0
- package/dist/tests/core/sleep-cycle.test.js.map +1 -0
- package/dist/tests/core/slots.test.js +201 -0
- package/dist/tests/core/slots.test.js.map +1 -0
- package/dist/tests/core/state-resolve.test.js +80 -0
- package/dist/tests/core/state-resolve.test.js.map +1 -0
- package/dist/tests/core/summary.test.js +200 -0
- package/dist/tests/core/summary.test.js.map +1 -0
- package/dist/tests/core/template.test.js +116 -0
- package/dist/tests/core/template.test.js.map +1 -0
- package/dist/tests/core/tick.test.js +558 -0
- package/dist/tests/core/tick.test.js.map +1 -0
- package/dist/tests/core/token-fold.test.js +176 -0
- package/dist/tests/core/token-fold.test.js.map +1 -0
- package/dist/tests/core/turn.test.js +388 -0
- package/dist/tests/core/turn.test.js.map +1 -0
- package/dist/tests/core/url.test.js +118 -0
- package/dist/tests/core/url.test.js.map +1 -0
- package/dist/tests/core/validate.test.js +247 -0
- package/dist/tests/core/validate.test.js.map +1 -0
- package/dist/tests/core/views.test.js +252 -0
- package/dist/tests/core/views.test.js.map +1 -0
- package/dist/tests/core/vm-decide.test.js +110 -0
- package/dist/tests/core/vm-decide.test.js.map +1 -0
- package/dist/tests/core/vm-guards.test.js +153 -0
- package/dist/tests/core/vm-guards.test.js.map +1 -0
- package/dist/tests/core/vm-plan.test.js +332 -0
- package/dist/tests/core/vm-plan.test.js.map +1 -0
- package/dist/tests/core/vm-reap.test.js +196 -0
- package/dist/tests/core/vm-reap.test.js.map +1 -0
- package/dist/tests/core/workflow-parse.test.js +493 -0
- package/dist/tests/core/workflow-parse.test.js.map +1 -0
- package/dist/tests/core/workspace-decide.test.js +236 -0
- package/dist/tests/core/workspace-decide.test.js.map +1 -0
- package/dist/tests/helpers/fixtures.js +167 -0
- package/dist/tests/helpers/fixtures.js.map +1 -0
- package/dist/tests/shell/acp-substrate.test.js +101 -0
- package/dist/tests/shell/acp-substrate.test.js.map +1 -0
- package/dist/tests/shell/actions-runner-push.test.js +203 -0
- package/dist/tests/shell/actions-runner-push.test.js.map +1 -0
- package/dist/tests/shell/credential-hooks.test.js +36 -0
- package/dist/tests/shell/credential-hooks.test.js.map +1 -0
- package/dist/tests/shell/credential-registry.test.js +165 -0
- package/dist/tests/shell/credential-registry.test.js.map +1 -0
- package/dist/tests/shell/credential-substrate.test.js +179 -0
- package/dist/tests/shell/credential-substrate.test.js.map +1 -0
- package/dist/tests/shell/dockerfile-mise-pin.test.js +51 -0
- package/dist/tests/shell/dockerfile-mise-pin.test.js.map +1 -0
- package/dist/tests/shell/doctor.test.js +101 -0
- package/dist/tests/shell/doctor.test.js.map +1 -0
- package/dist/tests/shell/effect-vm-create.test.js +52 -0
- package/dist/tests/shell/effect-vm-create.test.js.map +1 -0
- package/dist/tests/shell/gh-port.test.js +63 -0
- package/dist/tests/shell/gh-port.test.js.map +1 -0
- package/dist/tests/shell/gondolin-dispatch-guard.test.js +144 -0
- package/dist/tests/shell/gondolin-dispatch-guard.test.js.map +1 -0
- package/dist/tests/shell/gondolin-dispatch-shquote.test.js +168 -0
- package/dist/tests/shell/gondolin-dispatch-shquote.test.js.map +1 -0
- package/dist/tests/shell/gondolin-image-converter.test.js +208 -0
- package/dist/tests/shell/gondolin-image-converter.test.js.map +1 -0
- package/dist/tests/shell/gondolin-image-fetch.test.js +93 -0
- package/dist/tests/shell/gondolin-image-fetch.test.js.map +1 -0
- package/dist/tests/shell/http-handler.test.js +608 -0
- package/dist/tests/shell/http-handler.test.js.map +1 -0
- package/dist/tests/shell/http-server.test.js +53 -0
- package/dist/tests/shell/http-server.test.js.map +1 -0
- package/dist/tests/shell/mcp-runtime.test.js +366 -0
- package/dist/tests/shell/mcp-runtime.test.js.map +1 -0
- package/dist/tests/shell/mise-config-asset.test.js +87 -0
- package/dist/tests/shell/mise-config-asset.test.js.map +1 -0
- package/dist/tests/shell/orchestrator-loop.test.js +583 -0
- package/dist/tests/shell/orchestrator-loop.test.js.map +1 -0
- package/dist/tests/shell/reconciler-passes.test.js +314 -0
- package/dist/tests/shell/reconciler-passes.test.js.map +1 -0
- package/dist/tests/shell/runner-loop-turn.test.js +97 -0
- package/dist/tests/shell/runner-loop-turn.test.js.map +1 -0
- package/dist/tests/shell/runner-slice.test.js +536 -0
- package/dist/tests/shell/runner-slice.test.js.map +1 -0
- package/dist/tests/shell/scaffold.test.js +65 -0
- package/dist/tests/shell/scaffold.test.js.map +1 -0
- package/dist/tests/shell/tick-config.test.js +83 -0
- package/dist/tests/shell/tick-config.test.js.map +1 -0
- package/dist/tests/shell/tracker-parse-dates.test.js +44 -0
- package/dist/tests/shell/tracker-parse-dates.test.js.map +1 -0
- package/dist/tests/shell/tracker-write-issue.test.js +154 -0
- package/dist/tests/shell/tracker-write-issue.test.js.map +1 -0
- package/dist/tests/shell/workflow-prompt-split.test.js +208 -0
- package/dist/tests/shell/workflow-prompt-split.test.js.map +1 -0
- package/dist/tests/shell/workspace-live-config.test.js +140 -0
- package/dist/tests/shell/workspace-live-config.test.js.map +1 -0
- package/package.json +21 -9
- package/patches/@earendil-works+gondolin+0.12.0.patch +173 -0
- package/prompts/Reflect.md +91 -0
- package/prompts/Review.md +97 -0
- package/prompts/Todo.md +96 -0
- package/prompts/_footer.md +41 -0
- package/prompts/_preamble.md +42 -0
- package/prompts-minimal/Todo.md +26 -0
- package/scripts/postinstall.mjs +63 -0
- package/scripts/vm-agent.mjs +312 -90
- package/WORKFLOW.md +0 -744
- package/dist/acp-bridge.js +0 -324
- package/dist/acp-bridge.js.map +0 -1
- package/dist/actions/cache.js +0 -191
- package/dist/actions/cache.js.map +0 -1
- package/dist/actions/effects.js +0 -41
- package/dist/actions/effects.js.map +0 -1
- package/dist/actions/executor.js +0 -570
- package/dist/actions/executor.js.map +0 -1
- package/dist/actions/index.js +0 -13
- package/dist/actions/index.js.map +0 -1
- package/dist/actions/parsing.js.map +0 -1
- package/dist/actions/predicate-env.js +0 -27
- package/dist/actions/predicate-env.js.map +0 -1
- package/dist/actions/predicates.js +0 -49
- package/dist/actions/predicates.js.map +0 -1
- package/dist/actions/templating.js +0 -66
- package/dist/actions/templating.js.map +0 -1
- package/dist/actions/types.js +0 -15
- package/dist/actions/types.js.map +0 -1
- package/dist/agent/acp.js +0 -473
- package/dist/agent/acp.js.map +0 -1
- package/dist/agent/adapter-names.js +0 -159
- package/dist/agent/adapter-names.js.map +0 -1
- package/dist/agent/adapters.js +0 -511
- package/dist/agent/adapters.js.map +0 -1
- package/dist/agent/credential-extractors.js +0 -342
- package/dist/agent/credential-extractors.js.map +0 -1
- package/dist/agent/credential-secrets.js +0 -628
- package/dist/agent/credential-secrets.js.map +0 -1
- package/dist/agent/credential-ticker.js +0 -57
- package/dist/agent/credential-ticker.js.map +0 -1
- package/dist/agent/gondolin-creds-staging.js +0 -356
- package/dist/agent/gondolin-creds-staging.js.map +0 -1
- package/dist/agent/gondolin-dispatch.js +0 -375
- package/dist/agent/gondolin-dispatch.js.map +0 -1
- package/dist/agent/gondolin.js +0 -124
- package/dist/agent/gondolin.js.map +0 -1
- package/dist/agent/runner-decisions.js +0 -134
- package/dist/agent/runner-decisions.js.map +0 -1
- package/dist/agent/runner.js +0 -1456
- package/dist/agent/runner.js.map +0 -1
- package/dist/agent/tool-call-summary.js +0 -102
- package/dist/agent/tool-call-summary.js.map +0 -1
- package/dist/agent/vm-acp-mapping.js +0 -73
- package/dist/agent/vm-acp-mapping.js.map +0 -1
- package/dist/agent/vm-guards.js +0 -262
- package/dist/agent/vm-guards.js.map +0 -1
- package/dist/agent/vm-port.js +0 -22
- package/dist/agent/vm-port.js.map +0 -1
- package/dist/agent/vm-process-registry.js +0 -79
- package/dist/agent/vm-process-registry.js.map +0 -1
- package/dist/bin/cli-args.js +0 -105
- package/dist/bin/cli-args.js.map +0 -1
- package/dist/bin/symphony.js +0 -794
- package/dist/bin/symphony.js.map +0 -1
- package/dist/errors.js +0 -15
- package/dist/errors.js.map +0 -1
- package/dist/http-disk.js +0 -135
- package/dist/http-disk.js.map +0 -1
- package/dist/http-handlers.js.map +0 -1
- package/dist/http.js.map +0 -1
- package/dist/issues.js +0 -178
- package/dist/issues.js.map +0 -1
- package/dist/logging.js +0 -203
- package/dist/logging.js.map +0 -1
- package/dist/mcp.js +0 -706
- package/dist/mcp.js.map +0 -1
- package/dist/memory.js +0 -85
- package/dist/memory.js.map +0 -1
- package/dist/orchestrator-decisions.js +0 -331
- package/dist/orchestrator-decisions.js.map +0 -1
- package/dist/orchestrator.js +0 -1569
- package/dist/orchestrator.js.map +0 -1
- package/dist/prompt.js +0 -65
- package/dist/prompt.js.map +0 -1
- package/dist/reconciler/cache.js +0 -65
- package/dist/reconciler/cache.js.map +0 -1
- package/dist/reconciler/index.js +0 -448
- package/dist/reconciler/index.js.map +0 -1
- package/dist/reconciler/ledger.js +0 -131
- package/dist/reconciler/ledger.js.map +0 -1
- package/dist/reconciler/pr-adapters.js +0 -174
- package/dist/reconciler/pr-adapters.js.map +0 -1
- package/dist/reconciler/pr-decide.js.map +0 -1
- package/dist/reconciler/pr.js +0 -422
- package/dist/reconciler/pr.js.map +0 -1
- package/dist/reconciler/types.js +0 -12
- package/dist/reconciler/types.js.map +0 -1
- package/dist/reconciler/vm.js +0 -243
- package/dist/reconciler/vm.js.map +0 -1
- package/dist/reconciler/workspace-defaults.js +0 -83
- package/dist/reconciler/workspace-defaults.js.map +0 -1
- package/dist/reconciler/workspace.js +0 -272
- package/dist/reconciler/workspace.js.map +0 -1
- package/dist/runlog.js +0 -403
- package/dist/runlog.js.map +0 -1
- package/dist/scaffold.js +0 -165
- package/dist/scaffold.js.map +0 -1
- package/dist/trackers/local.js +0 -445
- package/dist/trackers/local.js.map +0 -1
- package/dist/trackers/types.js +0 -10
- package/dist/trackers/types.js.map +0 -1
- package/dist/types.js +0 -3
- package/dist/types.js.map +0 -1
- package/dist/util/clock.js +0 -12
- package/dist/util/clock.js.map +0 -1
- package/dist/util/crypto.js +0 -25
- package/dist/util/crypto.js.map +0 -1
- package/dist/util/frontmatter.js +0 -70
- package/dist/util/frontmatter.js.map +0 -1
- package/dist/util/fs-issues.js +0 -22
- package/dist/util/fs-issues.js.map +0 -1
- package/dist/util/process.js +0 -152
- package/dist/util/process.js.map +0 -1
- package/dist/util/workspace-key.js +0 -10
- package/dist/util/workspace-key.js.map +0 -1
- package/dist/workflow-loader.js +0 -147
- package/dist/workflow-loader.js.map +0 -1
- package/dist/workflow.js +0 -822
- package/dist/workflow.js.map +0 -1
- package/dist/workspace-types.js +0 -8
- package/dist/workspace-types.js.map +0 -1
- package/dist/workspace.js +0 -443
- package/dist/workspace.js.map +0 -1
|
@@ -0,0 +1,87 @@
|
|
|
1
|
+
// Pure unit tests for src/core/coerce.ts — the canonical scalar coercers.
|
|
2
|
+
// These PIN the behaviours that had silently drifted across the ~5 former core
|
|
3
|
+
// copies (see core/coerce.ts header), so a future re-fork is caught by CI.
|
|
4
|
+
import { test } from 'node:test';
|
|
5
|
+
import assert from 'node:assert/strict';
|
|
6
|
+
import { asString, asInt, asIntOr, asStringList, asStringListOr, asNumber, asTimestamp, } from '../../src/core/coerce.js';
|
|
7
|
+
// ─── asString: NO length guard (the http renderer's guard was the drift) ──────
|
|
8
|
+
test('asString: a string passes through unchanged, including the empty string', () => {
|
|
9
|
+
assert.equal(asString('hi'), 'hi');
|
|
10
|
+
// The canonical asString has NO `length > 0` guard (the http/render.ts copy
|
|
11
|
+
// did; the three parse copies did not). Empty string must pass through.
|
|
12
|
+
assert.equal(asString(''), '');
|
|
13
|
+
});
|
|
14
|
+
test('asString: non-strings (number/null/undefined/object) → null', () => {
|
|
15
|
+
assert.equal(asString(5), null);
|
|
16
|
+
assert.equal(asString(null), null);
|
|
17
|
+
assert.equal(asString(undefined), null);
|
|
18
|
+
assert.equal(asString({}), null);
|
|
19
|
+
assert.equal(asString(['a']), null);
|
|
20
|
+
});
|
|
21
|
+
// ─── asInt: isFinite + Math.trunc (NOT Number.isInteger — the issue drift) ────
|
|
22
|
+
test('asInt: a fractional number truncates toward zero (isFinite + trunc)', () => {
|
|
23
|
+
// The issue/parse.ts copy used Number.isInteger and returned null here; the
|
|
24
|
+
// canonical (matching the 20+ workflow-config call sites) truncates.
|
|
25
|
+
assert.equal(asInt(2.5), 2);
|
|
26
|
+
assert.equal(asInt(-2.9), -2);
|
|
27
|
+
assert.equal(asInt(2), 2);
|
|
28
|
+
assert.equal(asInt(2.0), 2);
|
|
29
|
+
});
|
|
30
|
+
test('asInt: a decimal-digit string parses; a non-integer string → null', () => {
|
|
31
|
+
assert.equal(asInt('42'), 42);
|
|
32
|
+
assert.equal(asInt('-7'), -7);
|
|
33
|
+
assert.equal(asInt('3.5'), null); // not a pure integer string
|
|
34
|
+
assert.equal(asInt('abc'), null);
|
|
35
|
+
});
|
|
36
|
+
test('asInt: Infinity / NaN / non-numbers → null', () => {
|
|
37
|
+
assert.equal(asInt(Infinity), null);
|
|
38
|
+
assert.equal(asInt(NaN), null);
|
|
39
|
+
assert.equal(asInt(null), null);
|
|
40
|
+
assert.equal(asInt(undefined), null);
|
|
41
|
+
assert.equal(asInt({}), null);
|
|
42
|
+
});
|
|
43
|
+
test('asIntOr: returns the fallback when uncoercible, else the coerced int', () => {
|
|
44
|
+
assert.equal(asIntOr(undefined, 30_000), 30_000);
|
|
45
|
+
assert.equal(asIntOr('nope', 5), 5);
|
|
46
|
+
assert.equal(asIntOr('12', 5), 12);
|
|
47
|
+
assert.equal(asIntOr(7.9, 5), 7); // truncates, does NOT fall back
|
|
48
|
+
});
|
|
49
|
+
// ─── asStringList ─────────────────────────────────────────────────────────────
|
|
50
|
+
test('asStringList: keeps only string elements; non-arrays → []', () => {
|
|
51
|
+
assert.deepEqual(asStringList(['a', 1, 'b', null, 'c']), ['a', 'b', 'c']);
|
|
52
|
+
assert.deepEqual(asStringList('not an array'), []);
|
|
53
|
+
assert.deepEqual(asStringList(undefined), []);
|
|
54
|
+
});
|
|
55
|
+
test('asStringListOr: non-array returns the fallback, array is filtered', () => {
|
|
56
|
+
const fb = ['default'];
|
|
57
|
+
assert.equal(asStringListOr(undefined, fb), fb); // same ref → fallback
|
|
58
|
+
assert.deepEqual(asStringListOr(['x', 2], fb), ['x']);
|
|
59
|
+
});
|
|
60
|
+
// ─── asNumber: finite number passes through (no truncation) ───────────────────
|
|
61
|
+
test('asNumber: a finite number (including fractional) passes through; else null', () => {
|
|
62
|
+
assert.equal(asNumber(2.5), 2.5);
|
|
63
|
+
assert.equal(asNumber(0), 0);
|
|
64
|
+
assert.equal(asNumber(Infinity), null);
|
|
65
|
+
assert.equal(asNumber('5'), null); // strings are not numbers here
|
|
66
|
+
assert.equal(asNumber(null), null);
|
|
67
|
+
});
|
|
68
|
+
// ─── asTimestamp: folds the old formatTimestamp (empty string → null) ─────────
|
|
69
|
+
test('asTimestamp: a non-empty string passes through; the empty string → null', () => {
|
|
70
|
+
assert.equal(asTimestamp('2026-01-01T00:00:00.000Z'), '2026-01-01T00:00:00.000Z');
|
|
71
|
+
// Folds the render.ts formatTimestamp length-guard the bare asTimestamp lacked.
|
|
72
|
+
assert.equal(asTimestamp(''), null);
|
|
73
|
+
});
|
|
74
|
+
// Core coercers operate over PLAIN DATA only: a `Date` (which the `yaml` lib would
|
|
75
|
+
// auto-decode) is normalized to an ISO string at the SHELL decode boundary
|
|
76
|
+
// (shell/interp/tracker-parse.ts:normalizeDates) BEFORE core sees it — covered by
|
|
77
|
+
// tests/shell/tracker-parse-dates.test.ts. So core's asTimestamp never sees a Date
|
|
78
|
+
// and treats one as a non-string → null.
|
|
79
|
+
test('asTimestamp: a Date is NOT plain data → null (normalized in the shell decode)', () => {
|
|
80
|
+
assert.equal(asTimestamp(new Date('2026-01-02T03:04:05.000Z')), null);
|
|
81
|
+
});
|
|
82
|
+
test('asTimestamp: non-string → null', () => {
|
|
83
|
+
assert.equal(asTimestamp(123), null);
|
|
84
|
+
assert.equal(asTimestamp(null), null);
|
|
85
|
+
assert.equal(asTimestamp({}), null);
|
|
86
|
+
});
|
|
87
|
+
//# sourceMappingURL=coerce.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"coerce.test.js","sourceRoot":"","sources":["../../../tests/core/coerce.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,228 @@
|
|
|
1
|
+
// Pure unit tests for src/core/actions/context.ts.
|
|
2
|
+
// The module is pure (data-in / data-out), so no mocks are needed.
|
|
3
|
+
import { test } from 'node:test';
|
|
4
|
+
import assert from 'node:assert/strict';
|
|
5
|
+
import { deriveActionContext, computeForwardedEnv, isCredentialEnvName, } from '../../src/core/actions/context.js';
|
|
6
|
+
// ─── fixtures ────────────────────────────────────────────────────────────────
|
|
7
|
+
function issue(over = {}) {
|
|
8
|
+
return {
|
|
9
|
+
id: 'ISS-42',
|
|
10
|
+
identifier: 'iss-42',
|
|
11
|
+
title: 'Fix the widget',
|
|
12
|
+
description: 'body text',
|
|
13
|
+
priority: null,
|
|
14
|
+
state: 'Todo',
|
|
15
|
+
branch_name: null,
|
|
16
|
+
url: null,
|
|
17
|
+
labels: [],
|
|
18
|
+
blocked_by: [],
|
|
19
|
+
created_at: null,
|
|
20
|
+
updated_at: null,
|
|
21
|
+
...over,
|
|
22
|
+
};
|
|
23
|
+
}
|
|
24
|
+
function view(over = {}) {
|
|
25
|
+
return {
|
|
26
|
+
issue_id: 'ISS-42',
|
|
27
|
+
identifier: 'iss-42',
|
|
28
|
+
issue: issue(),
|
|
29
|
+
session_id: null,
|
|
30
|
+
workspace_path: '/ws/iss-42',
|
|
31
|
+
started_at: '2026-06-02T00:00:00Z',
|
|
32
|
+
turn_count: 0,
|
|
33
|
+
retry_attempt: null,
|
|
34
|
+
input_tokens: 0,
|
|
35
|
+
output_tokens: 0,
|
|
36
|
+
total_tokens: 0,
|
|
37
|
+
last_event: null,
|
|
38
|
+
last_event_at: null,
|
|
39
|
+
last_message: null,
|
|
40
|
+
recent_events: [],
|
|
41
|
+
last_error: null,
|
|
42
|
+
transitioned: false,
|
|
43
|
+
steering_requested: false,
|
|
44
|
+
steering_question: null,
|
|
45
|
+
steering_context: null,
|
|
46
|
+
active_tool_call_count: 0,
|
|
47
|
+
resolved_actor: 'claude/default',
|
|
48
|
+
base_branch_at_dispatch: 'main',
|
|
49
|
+
github_repo_at_dispatch: null,
|
|
50
|
+
tracker_root_at_dispatch: null,
|
|
51
|
+
cleanup_workspace_on_exit: false,
|
|
52
|
+
...over,
|
|
53
|
+
};
|
|
54
|
+
}
|
|
55
|
+
// ─── deriveActionContext: all-fallback path (no staged env, no repo) ─────────
|
|
56
|
+
test('deriveActionContext applies every fallback when extraEnv is undefined', () => {
|
|
57
|
+
const ctx = deriveActionContext({ view: view(), extraEnv: undefined });
|
|
58
|
+
assert.equal(ctx.identifier, 'iss-42');
|
|
59
|
+
assert.equal(ctx.workspace, '/ws/iss-42');
|
|
60
|
+
assert.equal(ctx.branch, 'agent/iss-42');
|
|
61
|
+
assert.equal(ctx.base_branch, 'main');
|
|
62
|
+
assert.equal(ctx.issue_title, 'Fix the widget');
|
|
63
|
+
assert.equal(ctx.issue_body, 'body text');
|
|
64
|
+
assert.equal(ctx.repo, null);
|
|
65
|
+
assert.equal(ctx.pr_title, 'ISS-42: Fix the widget');
|
|
66
|
+
assert.equal(ctx.pr_body_file, '');
|
|
67
|
+
});
|
|
68
|
+
test('deriveActionContext: empty description falls back to empty string', () => {
|
|
69
|
+
const ctx = deriveActionContext({
|
|
70
|
+
view: view({ issue: issue({ description: null }) }),
|
|
71
|
+
extraEnv: undefined,
|
|
72
|
+
});
|
|
73
|
+
assert.equal(ctx.issue_body, '');
|
|
74
|
+
});
|
|
75
|
+
// ─── deriveActionContext: staged SYMPHONY_* env wins over fallbacks ──────────
|
|
76
|
+
test('deriveActionContext: staged SYMPHONY_* env overrides every fallback', () => {
|
|
77
|
+
const ctx = deriveActionContext({
|
|
78
|
+
view: view(),
|
|
79
|
+
extraEnv: {
|
|
80
|
+
SYMPHONY_BRANCH: 'feat/custom',
|
|
81
|
+
SYMPHONY_BASE_BRANCH: 'develop',
|
|
82
|
+
SYMPHONY_PR_TITLE: 'Custom PR title',
|
|
83
|
+
SYMPHONY_PR_BODY_FILE: '/tmp/body.md',
|
|
84
|
+
},
|
|
85
|
+
});
|
|
86
|
+
assert.equal(ctx.branch, 'feat/custom');
|
|
87
|
+
assert.equal(ctx.base_branch, 'develop');
|
|
88
|
+
assert.equal(ctx.pr_title, 'Custom PR title');
|
|
89
|
+
assert.equal(ctx.pr_body_file, '/tmp/body.md');
|
|
90
|
+
});
|
|
91
|
+
// ─── deriveActionContext: pr_title default with untitled issue ───────────────
|
|
92
|
+
test('deriveActionContext: untitled issue → pr_title is just the id', () => {
|
|
93
|
+
const ctx = deriveActionContext({
|
|
94
|
+
view: view({ issue: issue({ title: ' ' }) }),
|
|
95
|
+
extraEnv: undefined,
|
|
96
|
+
});
|
|
97
|
+
assert.equal(ctx.pr_title, 'ISS-42');
|
|
98
|
+
});
|
|
99
|
+
// ─── deriveActionContext: repo precedence ────────────────────────────────────
|
|
100
|
+
test('deriveActionContext: repoEnv wins over dispatch-pinned repo', () => {
|
|
101
|
+
const ctx = deriveActionContext({
|
|
102
|
+
view: view({ github_repo_at_dispatch: 'org/pinned' }),
|
|
103
|
+
extraEnv: undefined,
|
|
104
|
+
repoEnv: 'org/from-env',
|
|
105
|
+
});
|
|
106
|
+
assert.equal(ctx.repo, 'org/from-env');
|
|
107
|
+
});
|
|
108
|
+
test('deriveActionContext: empty repoEnv falls through to dispatch-pinned repo', () => {
|
|
109
|
+
const ctx = deriveActionContext({
|
|
110
|
+
view: view({ github_repo_at_dispatch: 'org/pinned' }),
|
|
111
|
+
extraEnv: undefined,
|
|
112
|
+
repoEnv: '',
|
|
113
|
+
});
|
|
114
|
+
assert.equal(ctx.repo, 'org/pinned');
|
|
115
|
+
});
|
|
116
|
+
test('deriveActionContext: no repoEnv and no pinned repo → null', () => {
|
|
117
|
+
const ctx = deriveActionContext({
|
|
118
|
+
view: view({ github_repo_at_dispatch: null }),
|
|
119
|
+
extraEnv: undefined,
|
|
120
|
+
});
|
|
121
|
+
assert.equal(ctx.repo, null);
|
|
122
|
+
});
|
|
123
|
+
test('deriveActionContext: unpinned repo falls back to repoDefault (cfg.workspace.github_repo)', () => {
|
|
124
|
+
const ctx = deriveActionContext({
|
|
125
|
+
view: view({ github_repo_at_dispatch: null }),
|
|
126
|
+
extraEnv: undefined,
|
|
127
|
+
repoDefault: 'org/from-cfg',
|
|
128
|
+
});
|
|
129
|
+
assert.equal(ctx.repo, 'org/from-cfg');
|
|
130
|
+
});
|
|
131
|
+
test('deriveActionContext: repoEnv and pinned repo both win over repoDefault', () => {
|
|
132
|
+
const fromEnv = deriveActionContext({
|
|
133
|
+
view: view({ github_repo_at_dispatch: null }),
|
|
134
|
+
extraEnv: undefined,
|
|
135
|
+
repoEnv: 'org/from-env',
|
|
136
|
+
repoDefault: 'org/from-cfg',
|
|
137
|
+
});
|
|
138
|
+
assert.equal(fromEnv.repo, 'org/from-env');
|
|
139
|
+
const fromPin = deriveActionContext({
|
|
140
|
+
view: view({ github_repo_at_dispatch: 'org/pinned' }),
|
|
141
|
+
extraEnv: undefined,
|
|
142
|
+
repoDefault: 'org/from-cfg',
|
|
143
|
+
});
|
|
144
|
+
assert.equal(fromPin.repo, 'org/pinned');
|
|
145
|
+
});
|
|
146
|
+
test('deriveActionContext: local-only (repoDefault null) keeps repo null', () => {
|
|
147
|
+
const ctx = deriveActionContext({
|
|
148
|
+
view: view({ github_repo_at_dispatch: null }),
|
|
149
|
+
extraEnv: undefined,
|
|
150
|
+
repoDefault: null,
|
|
151
|
+
});
|
|
152
|
+
assert.equal(ctx.repo, null);
|
|
153
|
+
});
|
|
154
|
+
// ─── deriveActionContext: base_branch precedence ─────────────────────────────
|
|
155
|
+
test('deriveActionContext: dispatch-pinned base used when no staged env', () => {
|
|
156
|
+
const ctx = deriveActionContext({
|
|
157
|
+
view: view({ base_branch_at_dispatch: 'release/1.x' }),
|
|
158
|
+
extraEnv: undefined,
|
|
159
|
+
});
|
|
160
|
+
assert.equal(ctx.base_branch, 'release/1.x');
|
|
161
|
+
});
|
|
162
|
+
test('deriveActionContext: empty pinned base → literal main', () => {
|
|
163
|
+
const ctx = deriveActionContext({
|
|
164
|
+
view: view({ base_branch_at_dispatch: '' }),
|
|
165
|
+
extraEnv: undefined,
|
|
166
|
+
});
|
|
167
|
+
assert.equal(ctx.base_branch, 'main');
|
|
168
|
+
});
|
|
169
|
+
test('deriveActionContext: empty pinned base falls back to baseBranchDefault before main', () => {
|
|
170
|
+
const ctx = deriveActionContext({
|
|
171
|
+
view: view({ base_branch_at_dispatch: '' }),
|
|
172
|
+
extraEnv: undefined,
|
|
173
|
+
baseBranchDefault: 'develop',
|
|
174
|
+
});
|
|
175
|
+
assert.equal(ctx.base_branch, 'develop');
|
|
176
|
+
});
|
|
177
|
+
// ─── computeForwardedEnv: forward only non-empty, non-credential vars ────────
|
|
178
|
+
test('computeForwardedEnv forwards non-empty, non-credential vars', () => {
|
|
179
|
+
const store = {
|
|
180
|
+
SYMPHONY_REPO: 'org/repo',
|
|
181
|
+
HOME: '/home/agent',
|
|
182
|
+
EMPTY: '',
|
|
183
|
+
};
|
|
184
|
+
const out = computeForwardedEnv(['SYMPHONY_REPO', 'HOME', 'EMPTY', 'MISSING'], undefined, (k) => store[k]);
|
|
185
|
+
assert.deepEqual(out, { SYMPHONY_REPO: 'org/repo', HOME: '/home/agent' });
|
|
186
|
+
});
|
|
187
|
+
test('computeForwardedEnv drops the omitVar', () => {
|
|
188
|
+
const store = { A: '1', B: '2' };
|
|
189
|
+
const out = computeForwardedEnv(['A', 'B'], 'B', (k) => store[k]);
|
|
190
|
+
assert.deepEqual(out, { A: '1' });
|
|
191
|
+
});
|
|
192
|
+
test('computeForwardedEnv strips credential-bearing vars even if listed', () => {
|
|
193
|
+
const store = {
|
|
194
|
+
ANTHROPIC_AUTH_TOKEN: 'sk-secret',
|
|
195
|
+
OPENAI_API_KEY: 'sk-openai',
|
|
196
|
+
GITHUB_TOKEN: 'ghp-x',
|
|
197
|
+
MY_API_KEY: 'k',
|
|
198
|
+
FOO_SECRET: 's',
|
|
199
|
+
BAR_TOKEN: 't',
|
|
200
|
+
ANTHROPIC_MODEL: 'claude', // ANTHROPIC_ prefix → stripped by boot-env policy
|
|
201
|
+
SAFE_VAR: 'ok',
|
|
202
|
+
};
|
|
203
|
+
const out = computeForwardedEnv(Object.keys(store), undefined, (k) => store[k]);
|
|
204
|
+
assert.deepEqual(out, { SAFE_VAR: 'ok' });
|
|
205
|
+
});
|
|
206
|
+
test('computeForwardedEnv: empty forward list yields empty env', () => {
|
|
207
|
+
const out = computeForwardedEnv([], undefined, () => 'whatever');
|
|
208
|
+
assert.deepEqual(out, {});
|
|
209
|
+
});
|
|
210
|
+
// ─── isCredentialEnvName: each rule family ───────────────────────────────────
|
|
211
|
+
test('isCredentialEnvName: named set, prefixes, and suffixes', () => {
|
|
212
|
+
// named set
|
|
213
|
+
assert.equal(isCredentialEnvName('ANTHROPIC_AUTH_TOKEN'), true);
|
|
214
|
+
assert.equal(isCredentialEnvName('GH_TOKEN'), true);
|
|
215
|
+
// vendor prefix families (case-insensitive)
|
|
216
|
+
assert.equal(isCredentialEnvName('anthropic_model'), true);
|
|
217
|
+
assert.equal(isCredentialEnvName('OPENAI_BASE_URL'), true);
|
|
218
|
+
assert.equal(isCredentialEnvName('GITHUB_REPOSITORY'), true);
|
|
219
|
+
// secret suffix families
|
|
220
|
+
assert.equal(isCredentialEnvName('VENDOR_API_KEY'), true);
|
|
221
|
+
assert.equal(isCredentialEnvName('SOME_TOKEN'), true);
|
|
222
|
+
assert.equal(isCredentialEnvName('A_SECRET'), true);
|
|
223
|
+
// non-credential
|
|
224
|
+
assert.equal(isCredentialEnvName('SYMPHONY_REPO'), false);
|
|
225
|
+
assert.equal(isCredentialEnvName('HOME'), false);
|
|
226
|
+
assert.equal(isCredentialEnvName('PATH'), false);
|
|
227
|
+
});
|
|
228
|
+
//# sourceMappingURL=context.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"context.test.js","sourceRoot":"","sources":["../../../tests/core/context.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,310 @@
|
|
|
1
|
+
// PURE unit tests for src/core/http/decisions.ts — node:test + node:assert.
|
|
2
|
+
// No mocks: every function under test is a pure (data) -> data transform.
|
|
3
|
+
import { test } from 'node:test';
|
|
4
|
+
import assert from 'node:assert/strict';
|
|
5
|
+
import { decideCreateIssue, decideMoveTransition, buildIssueDetailDto, } from '../../src/core/http/decisions.js';
|
|
6
|
+
// A representative workflow: Triage (holding) → Todo (active) → InProgress (active)
|
|
7
|
+
// → Done (terminal) → Cancelled (terminal). Declaration order is significant.
|
|
8
|
+
const STATES = [
|
|
9
|
+
{ name: 'Triage', role: 'holding' },
|
|
10
|
+
{ name: 'Todo', role: 'active' },
|
|
11
|
+
{ name: 'InProgress', role: 'active' },
|
|
12
|
+
{ name: 'Done', role: 'terminal' },
|
|
13
|
+
{ name: 'Cancelled', role: 'terminal' },
|
|
14
|
+
];
|
|
15
|
+
// ─── decideCreateIssue ────────────────────────────────────────────────────────
|
|
16
|
+
test('decideCreateIssue: non-object body → 400 bad_request', () => {
|
|
17
|
+
for (const body of [null, undefined, 42, 'str', ['arr'], true]) {
|
|
18
|
+
const d = decideCreateIssue(body, STATES);
|
|
19
|
+
assert.equal(d.ok, false);
|
|
20
|
+
assert.equal(d.ok === false && d.status, 400);
|
|
21
|
+
assert.equal(d.ok === false && d.code, 'bad_request');
|
|
22
|
+
}
|
|
23
|
+
});
|
|
24
|
+
test('decideCreateIssue: missing/blank title → 400', () => {
|
|
25
|
+
const d = decideCreateIssue({ title: ' ' }, STATES);
|
|
26
|
+
assert.equal(d.ok, false);
|
|
27
|
+
assert.equal(d.ok === false && d.message, 'title is required');
|
|
28
|
+
});
|
|
29
|
+
test('decideCreateIssue: defaults state to first active, trims fields, parses optionals', () => {
|
|
30
|
+
const d = decideCreateIssue({
|
|
31
|
+
title: ' Fix the bug ',
|
|
32
|
+
identifier: ' bug-1 ',
|
|
33
|
+
description: 'details',
|
|
34
|
+
priority: 2,
|
|
35
|
+
labels: ['x', 7, 'y'],
|
|
36
|
+
blocked_by: ['a', null, 'b'],
|
|
37
|
+
}, STATES);
|
|
38
|
+
assert.equal(d.ok, true);
|
|
39
|
+
if (!d.ok)
|
|
40
|
+
return;
|
|
41
|
+
assert.equal(d.title, 'Fix the bug');
|
|
42
|
+
assert.equal(d.identifier, 'bug-1');
|
|
43
|
+
assert.equal(d.state, 'Todo'); // first active
|
|
44
|
+
assert.equal(d.description, 'details');
|
|
45
|
+
assert.equal(d.priority, 2);
|
|
46
|
+
assert.deepEqual(d.labels, ['x', 'y']); // non-strings filtered
|
|
47
|
+
assert.deepEqual(d.blocked_by, ['a', 'b']);
|
|
48
|
+
});
|
|
49
|
+
test('decideCreateIssue: explicit holding state is allowed (non-terminal)', () => {
|
|
50
|
+
const d = decideCreateIssue({ title: 'Idea', state: 'Triage' }, STATES);
|
|
51
|
+
assert.equal(d.ok, true);
|
|
52
|
+
assert.equal(d.ok === true && d.state, 'Triage');
|
|
53
|
+
});
|
|
54
|
+
test('decideCreateIssue: terminal state is rejected with allowed list', () => {
|
|
55
|
+
const d = decideCreateIssue({ title: 'X', state: 'Done' }, STATES);
|
|
56
|
+
assert.equal(d.ok, false);
|
|
57
|
+
if (d.ok)
|
|
58
|
+
return;
|
|
59
|
+
assert.equal(d.status, 400);
|
|
60
|
+
// allowed = non-terminal names only
|
|
61
|
+
assert.equal(d.message, 'state must be one of: Triage, Todo, InProgress');
|
|
62
|
+
});
|
|
63
|
+
test('decideCreateIssue: unknown/path-escape state rejected', () => {
|
|
64
|
+
const d = decideCreateIssue({ title: 'X', state: '../escape' }, STATES);
|
|
65
|
+
assert.equal(d.ok, false);
|
|
66
|
+
});
|
|
67
|
+
test('decideCreateIssue: no active state declared + no explicit state → state required error', () => {
|
|
68
|
+
const noActive = [{ name: 'Done', role: 'terminal' }];
|
|
69
|
+
const d = decideCreateIssue({ title: 'X' }, noActive);
|
|
70
|
+
assert.equal(d.ok, false);
|
|
71
|
+
if (d.ok)
|
|
72
|
+
return;
|
|
73
|
+
assert.equal(d.message, 'state is required (no active states declared to default to)');
|
|
74
|
+
});
|
|
75
|
+
test('decideCreateIssue: non-finite/non-number priority → null; missing optionals → defaults', () => {
|
|
76
|
+
const d = decideCreateIssue({ title: 'X', priority: Number.NaN }, STATES);
|
|
77
|
+
assert.equal(d.ok, true);
|
|
78
|
+
if (!d.ok)
|
|
79
|
+
return;
|
|
80
|
+
assert.equal(d.priority, null);
|
|
81
|
+
assert.equal(d.identifier, ''); // not supplied
|
|
82
|
+
assert.equal(d.description, undefined);
|
|
83
|
+
assert.deepEqual(d.labels, []);
|
|
84
|
+
assert.deepEqual(d.blocked_by, []);
|
|
85
|
+
});
|
|
86
|
+
// ─── decideMoveTransition ─────────────────────────────────────────────────────
|
|
87
|
+
// The operator move replaces the fixed approve/discard pair (issue 126): any
|
|
88
|
+
// declared target, off-flow only WARNED (never blocked), from-state resolved to
|
|
89
|
+
// the file's actual directory (mismatch ⇒ 409).
|
|
90
|
+
// A flow with a declared transition graph so off-flow can be exercised:
|
|
91
|
+
// Triage (holding, open) → Todo ⇄ Review → Done; Cancelled is a dead-end ([]).
|
|
92
|
+
const FLOW = [
|
|
93
|
+
{ name: 'Triage', role: 'holding', allowed_transitions: null }, // open default
|
|
94
|
+
{ name: 'Todo', role: 'active', allowed_transitions: ['Review', 'Cancelled'] },
|
|
95
|
+
{ name: 'Review', role: 'active', allowed_transitions: ['Todo', 'Done'] },
|
|
96
|
+
{ name: 'Done', role: 'terminal', allowed_transitions: null },
|
|
97
|
+
{ name: 'Cancelled', role: 'terminal', allowed_transitions: [] }, // no transitions out
|
|
98
|
+
];
|
|
99
|
+
test('decideMoveTransition: non-object body → 400 bad_request', () => {
|
|
100
|
+
for (const body of [null, undefined, 42, 'str', ['arr'], true]) {
|
|
101
|
+
const d = decideMoveTransition(body, FLOW, 'Todo');
|
|
102
|
+
assert.equal(d.ok, false);
|
|
103
|
+
assert.equal(d.ok === false && d.status, 400);
|
|
104
|
+
assert.equal(d.ok === false && d.code, 'bad_request');
|
|
105
|
+
}
|
|
106
|
+
});
|
|
107
|
+
test('decideMoveTransition: missing/blank target state → 400 state required', () => {
|
|
108
|
+
const d = decideMoveTransition({ state: ' ' }, FLOW, 'Todo');
|
|
109
|
+
assert.equal(d.ok, false);
|
|
110
|
+
if (d.ok)
|
|
111
|
+
return;
|
|
112
|
+
assert.equal(d.status, 400);
|
|
113
|
+
assert.equal(d.message, 'state is required');
|
|
114
|
+
});
|
|
115
|
+
test('decideMoveTransition: undeclared target → 400 with the declared list', () => {
|
|
116
|
+
const d = decideMoveTransition({ state: 'Nowhere' }, FLOW, 'Todo');
|
|
117
|
+
assert.equal(d.ok, false);
|
|
118
|
+
if (d.ok)
|
|
119
|
+
return;
|
|
120
|
+
assert.equal(d.status, 400);
|
|
121
|
+
assert.equal(d.message, 'state must be one of: Triage, Todo, Review, Done, Cancelled');
|
|
122
|
+
});
|
|
123
|
+
test('decideMoveTransition: case-insensitive target canonicalizes; fromState is the actual dir', () => {
|
|
124
|
+
// actualFromState is the on-disk directory (here lower-cased), echoed verbatim
|
|
125
|
+
// so the shell moves the RIGHT file; the target canonicalizes to declared casing.
|
|
126
|
+
const d = decideMoveTransition({ state: 'review' }, FLOW, 'todo');
|
|
127
|
+
assert.equal(d.ok, true);
|
|
128
|
+
if (!d.ok)
|
|
129
|
+
return;
|
|
130
|
+
assert.equal(d.toState, 'Review');
|
|
131
|
+
assert.equal(d.fromState, 'todo');
|
|
132
|
+
assert.equal(d.offFlow, false); // Review ∈ Todo.allowed_transitions
|
|
133
|
+
});
|
|
134
|
+
test('decideMoveTransition: approve-equivalent (Triage → first active) is in-flow', () => {
|
|
135
|
+
// Triage.allowed_transitions === null ⇒ open default ⇒ in-flow.
|
|
136
|
+
const d = decideMoveTransition({ state: 'Todo' }, FLOW, 'Triage');
|
|
137
|
+
assert.equal(d.ok, true);
|
|
138
|
+
if (!d.ok)
|
|
139
|
+
return;
|
|
140
|
+
assert.equal(d.toState, 'Todo');
|
|
141
|
+
assert.equal(d.fromState, 'Triage');
|
|
142
|
+
assert.equal(d.offFlow, false);
|
|
143
|
+
});
|
|
144
|
+
test('decideMoveTransition: discard-equivalent (Todo → Cancelled) is in-flow when allowed', () => {
|
|
145
|
+
const d = decideMoveTransition({ state: 'Cancelled' }, FLOW, 'Todo');
|
|
146
|
+
assert.equal(d.ok, true);
|
|
147
|
+
assert.equal(d.ok === true && d.offFlow, false); // Cancelled ∈ Todo.allowed_transitions
|
|
148
|
+
});
|
|
149
|
+
test('decideMoveTransition: Review → Todo (rework) is in-flow', () => {
|
|
150
|
+
const d = decideMoveTransition({ state: 'Todo' }, FLOW, 'Review');
|
|
151
|
+
assert.equal(d.ok, true);
|
|
152
|
+
assert.equal(d.ok === true && d.offFlow, false);
|
|
153
|
+
});
|
|
154
|
+
test('decideMoveTransition: off-flow target still applies but sets offFlow=true', () => {
|
|
155
|
+
// Todo only declares [Review, Cancelled]; Done is declared but off-flow.
|
|
156
|
+
const d = decideMoveTransition({ state: 'Done' }, FLOW, 'Todo');
|
|
157
|
+
assert.equal(d.ok, true);
|
|
158
|
+
if (!d.ok)
|
|
159
|
+
return;
|
|
160
|
+
assert.equal(d.toState, 'Done');
|
|
161
|
+
assert.equal(d.offFlow, true);
|
|
162
|
+
});
|
|
163
|
+
test('decideMoveTransition: an empty allowed_transitions list makes every target off-flow', () => {
|
|
164
|
+
// Cancelled.allowed_transitions === [] ⇒ no in-flow targets; yanking it back to
|
|
165
|
+
// Todo is allowed (operator is never blocked) but flagged off-flow.
|
|
166
|
+
const d = decideMoveTransition({ state: 'Todo' }, FLOW, 'Cancelled');
|
|
167
|
+
assert.equal(d.ok, true);
|
|
168
|
+
assert.equal(d.ok === true && d.offFlow, true);
|
|
169
|
+
});
|
|
170
|
+
test('decideMoveTransition: a null/open from-state list is in-flow for any declared target', () => {
|
|
171
|
+
// Triage (open) → Done jumps past the active states yet is in-flow (open default).
|
|
172
|
+
const d = decideMoveTransition({ state: 'Done' }, FLOW, 'Triage');
|
|
173
|
+
assert.equal(d.ok, true);
|
|
174
|
+
assert.equal(d.ok === true && d.offFlow, false);
|
|
175
|
+
});
|
|
176
|
+
test('decideMoveTransition: an undeclared (orphan) from-state is in-flow (gate skipped)', () => {
|
|
177
|
+
const d = decideMoveTransition({ state: 'Todo' }, FLOW, 'Ghost');
|
|
178
|
+
assert.equal(d.ok, true);
|
|
179
|
+
if (!d.ok)
|
|
180
|
+
return;
|
|
181
|
+
assert.equal(d.fromState, 'Ghost');
|
|
182
|
+
assert.equal(d.offFlow, false);
|
|
183
|
+
});
|
|
184
|
+
test('decideMoveTransition: self-transition is in-flow even from a dead-end state', () => {
|
|
185
|
+
const d = decideMoveTransition({ state: 'Cancelled' }, FLOW, 'Cancelled');
|
|
186
|
+
assert.equal(d.ok, true);
|
|
187
|
+
if (!d.ok)
|
|
188
|
+
return;
|
|
189
|
+
assert.equal(d.toState, 'Cancelled');
|
|
190
|
+
assert.equal(d.offFlow, false);
|
|
191
|
+
});
|
|
192
|
+
test('decideMoveTransition: claimed from_state mismatching the actual dir → 409', () => {
|
|
193
|
+
const d = decideMoveTransition({ state: 'Review', from_state: 'Triage' }, FLOW, 'Todo');
|
|
194
|
+
assert.equal(d.ok, false);
|
|
195
|
+
if (d.ok)
|
|
196
|
+
return;
|
|
197
|
+
assert.equal(d.status, 409);
|
|
198
|
+
assert.equal(d.code, 'from_state_mismatch');
|
|
199
|
+
assert.match(d.message, /issue is in "Todo", not "Triage"/);
|
|
200
|
+
});
|
|
201
|
+
test('decideMoveTransition: claimed from_state matches the actual dir case-insensitively', () => {
|
|
202
|
+
const d = decideMoveTransition({ state: 'Review', from_state: 'todo' }, FLOW, 'Todo');
|
|
203
|
+
assert.equal(d.ok, true);
|
|
204
|
+
assert.equal(d.ok === true && d.toState, 'Review');
|
|
205
|
+
});
|
|
206
|
+
test('decideMoveTransition: a workflow with no declared states rejects any target (400)', () => {
|
|
207
|
+
const d = decideMoveTransition({ state: 'X' }, [], 'Todo');
|
|
208
|
+
assert.equal(d.ok, false);
|
|
209
|
+
if (d.ok)
|
|
210
|
+
return;
|
|
211
|
+
assert.equal(d.message, 'state must be one of: <none configured>');
|
|
212
|
+
});
|
|
213
|
+
// ─── buildIssueDetailDto ──────────────────────────────────────────────────────
|
|
214
|
+
function makeEntry(over = {}) {
|
|
215
|
+
return {
|
|
216
|
+
issue_id: 'iss-1',
|
|
217
|
+
identifier: 'BUG-1',
|
|
218
|
+
workspace_path: '/ws/BUG-1',
|
|
219
|
+
session_id: 'sess-9',
|
|
220
|
+
turn_count: 3,
|
|
221
|
+
state: 'InProgress',
|
|
222
|
+
started_at: '2026-06-02T00:00:00.000Z',
|
|
223
|
+
last_event: 'tool_call',
|
|
224
|
+
last_message: 'working',
|
|
225
|
+
last_event_at: '2026-06-02T00:01:00.000Z',
|
|
226
|
+
input_tokens: 100,
|
|
227
|
+
output_tokens: 50,
|
|
228
|
+
total_tokens: 150,
|
|
229
|
+
recent_events: [],
|
|
230
|
+
last_error: null,
|
|
231
|
+
...over,
|
|
232
|
+
};
|
|
233
|
+
}
|
|
234
|
+
function makeRetry(over = {}) {
|
|
235
|
+
return {
|
|
236
|
+
issue_id: 'iss-1',
|
|
237
|
+
identifier: 'BUG-1',
|
|
238
|
+
attempt: 2,
|
|
239
|
+
due_at_ms: 1_700_000_000_000,
|
|
240
|
+
error: 'boom',
|
|
241
|
+
kind: 'failure',
|
|
242
|
+
...over,
|
|
243
|
+
};
|
|
244
|
+
}
|
|
245
|
+
test('buildIssueDetailDto: no entry and no retry → null', () => {
|
|
246
|
+
assert.equal(buildIssueDetailDto('BUG-1', null, null), null);
|
|
247
|
+
});
|
|
248
|
+
test('buildIssueDetailDto: live entry → status running, running section populated', () => {
|
|
249
|
+
const dto = buildIssueDetailDto('BUG-1', makeEntry(), null);
|
|
250
|
+
assert.ok(dto);
|
|
251
|
+
assert.equal(dto.status, 'running');
|
|
252
|
+
assert.equal(dto.issue_identifier, 'BUG-1');
|
|
253
|
+
assert.equal(dto.issue_id, 'iss-1');
|
|
254
|
+
assert.deepEqual(dto.workspace, { path: '/ws/BUG-1' });
|
|
255
|
+
assert.equal(dto.attempts.current_retry_attempt, null);
|
|
256
|
+
assert.equal(dto.retry, null);
|
|
257
|
+
assert.ok(dto.running);
|
|
258
|
+
assert.equal(dto.running.session_id, 'sess-9');
|
|
259
|
+
assert.equal(dto.running.turn_count, 3);
|
|
260
|
+
assert.deepEqual(dto.running.tokens, {
|
|
261
|
+
input_tokens: 100,
|
|
262
|
+
output_tokens: 50,
|
|
263
|
+
total_tokens: 150,
|
|
264
|
+
});
|
|
265
|
+
assert.deepEqual(dto.tracked, {});
|
|
266
|
+
});
|
|
267
|
+
test('buildIssueDetailDto: entry takes precedence over retry (status running)', () => {
|
|
268
|
+
const dto = buildIssueDetailDto('BUG-1', makeEntry(), makeRetry({ kind: 'continuation' }));
|
|
269
|
+
assert.ok(dto);
|
|
270
|
+
// entry present → running even though a continuation retry also exists
|
|
271
|
+
assert.equal(dto.status, 'running');
|
|
272
|
+
assert.ok(dto.running);
|
|
273
|
+
assert.ok(dto.retry); // retry section still rendered
|
|
274
|
+
// last_error = entry.last_error ?? retry.error ?? null. entry.last_error is
|
|
275
|
+
// null here so the `??` chain falls through to the retry's error ('boom').
|
|
276
|
+
assert.equal(dto.last_error, 'boom');
|
|
277
|
+
// current_retry_attempt still sourced from the retry
|
|
278
|
+
assert.equal(dto.attempts.current_retry_attempt, 2);
|
|
279
|
+
});
|
|
280
|
+
test('buildIssueDetailDto: entry last_error wins when non-null', () => {
|
|
281
|
+
const dto = buildIssueDetailDto('BUG-1', makeEntry({ last_error: 'entry-error' }), makeRetry());
|
|
282
|
+
assert.ok(dto);
|
|
283
|
+
assert.equal(dto.last_error, 'entry-error');
|
|
284
|
+
});
|
|
285
|
+
test('buildIssueDetailDto: failure retry → status retrying, ISO due_at from due_at_ms', () => {
|
|
286
|
+
const dto = buildIssueDetailDto('BUG-1', null, makeRetry({ kind: 'failure' }));
|
|
287
|
+
assert.ok(dto);
|
|
288
|
+
assert.equal(dto.status, 'retrying');
|
|
289
|
+
assert.equal(dto.running, null);
|
|
290
|
+
assert.ok(dto.retry);
|
|
291
|
+
assert.equal(dto.retry.kind, 'failure');
|
|
292
|
+
assert.equal(dto.retry.attempt, 2);
|
|
293
|
+
assert.equal(dto.retry.due_at, new Date(1_700_000_000_000).toISOString());
|
|
294
|
+
assert.equal(dto.retry.error, 'boom');
|
|
295
|
+
assert.equal(dto.last_error, 'boom');
|
|
296
|
+
assert.equal(dto.attempts.current_retry_attempt, 2);
|
|
297
|
+
});
|
|
298
|
+
test('buildIssueDetailDto: continuation retry → status resuming (the pill collapse case)', () => {
|
|
299
|
+
const dto = buildIssueDetailDto('BUG-1', null, makeRetry({ kind: 'continuation', error: null }));
|
|
300
|
+
assert.ok(dto);
|
|
301
|
+
assert.equal(dto.status, 'resuming');
|
|
302
|
+
assert.equal(dto.last_error, null);
|
|
303
|
+
});
|
|
304
|
+
test('buildIssueDetailDto: issue_id falls back to retry when no entry', () => {
|
|
305
|
+
const dto = buildIssueDetailDto('BUG-1', null, makeRetry({ issue_id: 'retry-id' }));
|
|
306
|
+
assert.ok(dto);
|
|
307
|
+
assert.equal(dto.issue_id, 'retry-id');
|
|
308
|
+
assert.deepEqual(dto.recent_events, []);
|
|
309
|
+
});
|
|
310
|
+
//# sourceMappingURL=decisions.test.js.map
|