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,139 @@
|
|
|
1
|
+
// Pure unit tests for src/core/credential/extract.ts — no mocks (every export is
|
|
2
|
+
// pure). Covers the main decision branches: claude shapes, codex ChatGPT-OAuth vs
|
|
3
|
+
// API-key (and the host-only-refresh invariant: refresh_token is never read), JWT
|
|
4
|
+
// exp decoding (incl. a multibyte UTF-8 payload through the atob path), and expiry
|
|
5
|
+
// coercion.
|
|
6
|
+
import { test } from 'node:test';
|
|
7
|
+
import assert from 'node:assert/strict';
|
|
8
|
+
import { extractClaudeToken, extractCodexToken, codexEnvFallback, decodeJwtExpMs, coerceExpiresAtMs, } from '../../src/core/credential/extract.js';
|
|
9
|
+
// Fixtures generated from a `{alg:"none"}` header + the noted payload.
|
|
10
|
+
const JWT_VALID = 'eyJhbGciOiJub25lIiwidHlwIjoiSldUIn0.eyJleHAiOjE3MDAwMDAwMDAsInN1YiI6IngifQ.sig'; // exp=1700000000
|
|
11
|
+
const JWT_NOEXP = 'eyJhbGciOiJub25lIiwidHlwIjoiSldUIn0.eyJzdWIiOiJ4In0.sig';
|
|
12
|
+
const JWT_STREXP = 'eyJhbGciOiJub25lIiwidHlwIjoiSldUIn0.eyJleHAiOiJzb29uIn0.sig'; // exp="soon"
|
|
13
|
+
const JWT_UTF8 = 'eyJhbGciOiJub25lIiwidHlwIjoiSldUIn0.eyJleHAiOjE3MDAwMDAwMDAsIm5hbWUiOiJjYWbDqeKAlOKckyJ9.sig'; // multibyte payload, exp=1700000000
|
|
14
|
+
// ── extractClaudeToken ───────────────────────────────────────────────────────
|
|
15
|
+
test('extractClaudeToken: reads the nested claudeAiOauth bundle', () => {
|
|
16
|
+
const info = extractClaudeToken({
|
|
17
|
+
claudeAiOauth: { accessToken: 'sk-live', expiresAt: 1234 },
|
|
18
|
+
});
|
|
19
|
+
assert.deepEqual(info, { accessToken: 'sk-live', expiresAtMs: 1234 });
|
|
20
|
+
});
|
|
21
|
+
test('extractClaudeToken: tolerates flat top-level fields', () => {
|
|
22
|
+
const info = extractClaudeToken({ accessToken: 'flat-tok', expiresAt: 9999 });
|
|
23
|
+
assert.deepEqual(info, { accessToken: 'flat-tok', expiresAtMs: 9999 });
|
|
24
|
+
});
|
|
25
|
+
test('extractClaudeToken: missing/empty token -> null', () => {
|
|
26
|
+
assert.equal(extractClaudeToken({ claudeAiOauth: { accessToken: '' } }), null);
|
|
27
|
+
assert.equal(extractClaudeToken({ claudeAiOauth: {} }), null);
|
|
28
|
+
assert.equal(extractClaudeToken({}), null);
|
|
29
|
+
assert.equal(extractClaudeToken(null), null);
|
|
30
|
+
assert.equal(extractClaudeToken('nope'), null);
|
|
31
|
+
});
|
|
32
|
+
test('extractClaudeToken: unparseable expiresAt -> expiresAtMs null but token kept', () => {
|
|
33
|
+
const info = extractClaudeToken({ accessToken: 't', expiresAt: { not: 'a date' } });
|
|
34
|
+
assert.deepEqual(info, { accessToken: 't', expiresAtMs: null });
|
|
35
|
+
});
|
|
36
|
+
// ── extractCodexToken ────────────────────────────────────────────────────────
|
|
37
|
+
test('extractCodexToken: ChatGPT-OAuth token -> chatgptOAuth=true + JWT exp + account id', () => {
|
|
38
|
+
const info = extractCodexToken({
|
|
39
|
+
tokens: { access_token: JWT_VALID, account_id: 'acct-1', refresh_token: 'SECRET' },
|
|
40
|
+
});
|
|
41
|
+
assert.equal(info?.accessToken, JWT_VALID);
|
|
42
|
+
assert.equal(info?.chatgptOAuth, true);
|
|
43
|
+
assert.equal(info?.chatgptAccountId, 'acct-1');
|
|
44
|
+
assert.equal(info?.expiresAtMs, 1700000000 * 1000);
|
|
45
|
+
});
|
|
46
|
+
test('extractCodexToken: NEVER reads refresh_token (host-only-refresh invariant)', () => {
|
|
47
|
+
// The refresh token is present but must never surface in the returned TokenInfo.
|
|
48
|
+
const info = extractCodexToken({
|
|
49
|
+
tokens: { access_token: JWT_VALID, refresh_token: 'rt_SECRET_must_not_leak' },
|
|
50
|
+
});
|
|
51
|
+
assert.equal(info?.accessToken, JWT_VALID);
|
|
52
|
+
assert.equal(JSON.stringify(info).includes('rt_SECRET_must_not_leak'), false);
|
|
53
|
+
});
|
|
54
|
+
test('extractCodexToken: non-JWT OAuth access token -> expiresAtMs null (degrades safely)', () => {
|
|
55
|
+
const info = extractCodexToken({ tokens: { access_token: 'opaque-not-a-jwt' } });
|
|
56
|
+
assert.deepEqual(info, {
|
|
57
|
+
accessToken: 'opaque-not-a-jwt',
|
|
58
|
+
expiresAtMs: null,
|
|
59
|
+
chatgptOAuth: true,
|
|
60
|
+
chatgptAccountId: null,
|
|
61
|
+
});
|
|
62
|
+
});
|
|
63
|
+
test('extractCodexToken: no OAuth token -> falls back to OPENAI_API_KEY (API-key route)', () => {
|
|
64
|
+
const info = extractCodexToken({ OPENAI_API_KEY: 'sk-apikey' });
|
|
65
|
+
// API key: no chatgptOAuth flag, expiry null (API keys do not expire).
|
|
66
|
+
assert.deepEqual(info, { accessToken: 'sk-apikey', expiresAtMs: null });
|
|
67
|
+
});
|
|
68
|
+
test('extractCodexToken: empty tokens.access_token falls through to API key', () => {
|
|
69
|
+
const info = extractCodexToken({
|
|
70
|
+
tokens: { access_token: '' },
|
|
71
|
+
OPENAI_API_KEY: 'sk-fallback',
|
|
72
|
+
});
|
|
73
|
+
assert.deepEqual(info, { accessToken: 'sk-fallback', expiresAtMs: null });
|
|
74
|
+
});
|
|
75
|
+
test('extractCodexToken: nothing usable -> null', () => {
|
|
76
|
+
assert.equal(extractCodexToken({}), null);
|
|
77
|
+
assert.equal(extractCodexToken({ tokens: {} }), null);
|
|
78
|
+
assert.equal(extractCodexToken({ tokens: 'bad', OPENAI_API_KEY: '' }), null);
|
|
79
|
+
assert.equal(extractCodexToken(null), null);
|
|
80
|
+
assert.equal(extractCodexToken([1, 2, 3]), null);
|
|
81
|
+
});
|
|
82
|
+
// ── codexEnvFallback ─────────────────────────────────────────────────────────
|
|
83
|
+
test('codexEnvFallback: OPENAI_API_KEY in env -> API-key TokenInfo', () => {
|
|
84
|
+
assert.deepEqual(codexEnvFallback({ OPENAI_API_KEY: 'envkey' }), {
|
|
85
|
+
accessToken: 'envkey',
|
|
86
|
+
expiresAtMs: null,
|
|
87
|
+
});
|
|
88
|
+
});
|
|
89
|
+
test('codexEnvFallback: absent/empty -> null', () => {
|
|
90
|
+
assert.equal(codexEnvFallback({}), null);
|
|
91
|
+
assert.equal(codexEnvFallback({ OPENAI_API_KEY: '' }), null);
|
|
92
|
+
assert.equal(codexEnvFallback({ OPENAI_API_KEY: undefined }), null);
|
|
93
|
+
});
|
|
94
|
+
// ── decodeJwtExpMs ───────────────────────────────────────────────────────────
|
|
95
|
+
test('decodeJwtExpMs: well-formed JWT -> exp seconds * 1000', () => {
|
|
96
|
+
assert.equal(decodeJwtExpMs(JWT_VALID), 1700000000 * 1000);
|
|
97
|
+
});
|
|
98
|
+
test('decodeJwtExpMs: payload with multibyte UTF-8 still decodes the exp', () => {
|
|
99
|
+
// Exercises the base64url -> atob -> UTF-8 reinterpretation path on a payload
|
|
100
|
+
// containing non-ASCII bytes (é, em-dash, check mark).
|
|
101
|
+
assert.equal(decodeJwtExpMs(JWT_UTF8), 1700000000 * 1000);
|
|
102
|
+
});
|
|
103
|
+
test('decodeJwtExpMs: wrong segment count -> null', () => {
|
|
104
|
+
assert.equal(decodeJwtExpMs('only.two'), null);
|
|
105
|
+
assert.equal(decodeJwtExpMs('a.b.c.d'), null);
|
|
106
|
+
assert.equal(decodeJwtExpMs('singlestring'), null);
|
|
107
|
+
});
|
|
108
|
+
test('decodeJwtExpMs: empty payload segment -> null', () => {
|
|
109
|
+
assert.equal(decodeJwtExpMs('header..sig'), null);
|
|
110
|
+
});
|
|
111
|
+
test('decodeJwtExpMs: non-numeric / missing exp -> null', () => {
|
|
112
|
+
assert.equal(decodeJwtExpMs(JWT_NOEXP), null);
|
|
113
|
+
assert.equal(decodeJwtExpMs(JWT_STREXP), null);
|
|
114
|
+
});
|
|
115
|
+
test('decodeJwtExpMs: payload not valid base64/JSON -> null', () => {
|
|
116
|
+
// '@@@' is not valid base64 content; must degrade to null, not throw.
|
|
117
|
+
assert.equal(decodeJwtExpMs('h.@@@.s'), null);
|
|
118
|
+
// Valid base64 of a non-object ("5") -> null.
|
|
119
|
+
assert.equal(decodeJwtExpMs('h.NQ.s'), null);
|
|
120
|
+
});
|
|
121
|
+
// ── coerceExpiresAtMs ────────────────────────────────────────────────────────
|
|
122
|
+
test('coerceExpiresAtMs: finite number passes through', () => {
|
|
123
|
+
assert.equal(coerceExpiresAtMs(1717000000000), 1717000000000);
|
|
124
|
+
assert.equal(coerceExpiresAtMs(0), 0);
|
|
125
|
+
});
|
|
126
|
+
test('coerceExpiresAtMs: numeric string parsed as a number', () => {
|
|
127
|
+
assert.equal(coerceExpiresAtMs('1717000000000'), 1717000000000);
|
|
128
|
+
});
|
|
129
|
+
test('coerceExpiresAtMs: ISO date string parsed via Date.parse', () => {
|
|
130
|
+
assert.equal(coerceExpiresAtMs('2024-01-01T00:00:00Z'), Date.parse('2024-01-01T00:00:00Z'));
|
|
131
|
+
});
|
|
132
|
+
test('coerceExpiresAtMs: non-finite / non-coercible -> null', () => {
|
|
133
|
+
assert.equal(coerceExpiresAtMs(Infinity), null);
|
|
134
|
+
assert.equal(coerceExpiresAtMs(NaN), null);
|
|
135
|
+
assert.equal(coerceExpiresAtMs('not-a-date'), null);
|
|
136
|
+
assert.equal(coerceExpiresAtMs(null), null);
|
|
137
|
+
assert.equal(coerceExpiresAtMs({}), null);
|
|
138
|
+
});
|
|
139
|
+
//# sourceMappingURL=extract.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"extract.test.js","sourceRoot":"","sources":["../../../tests/core/extract.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,134 @@
|
|
|
1
|
+
// Pure unit tests for src/core/credential/fake-creds.ts.
|
|
2
|
+
//
|
|
3
|
+
// The functions under test are pure (data -> data), so no mocks: we pass plain
|
|
4
|
+
// data and assert on the returned GondolinFakeCreds / GuestCredFile[] / config
|
|
5
|
+
// JSON. Covers the main decision branches:
|
|
6
|
+
// - the adapter switch in buildGondolinFakeCreds (claude / codex);
|
|
7
|
+
// - the placeholder staged VERBATIM as the bearer (exact-match egress);
|
|
8
|
+
// - the env map always { [secretName]: placeholder };
|
|
9
|
+
// - claude identity present vs. absent (1 vs. 2 files);
|
|
10
|
+
// - codex completeness fields + the defense-in-depth UUID/auth_mode/ISO guards
|
|
11
|
+
// (a token-shaped accountId / bad authMode / bad lastRefresh are OMITTED);
|
|
12
|
+
// - file mode is 0600 on every staged file.
|
|
13
|
+
import { test } from 'node:test';
|
|
14
|
+
import assert from 'node:assert/strict';
|
|
15
|
+
import { buildGondolinFakeCreds, buildClaudeFiles, buildCodexFiles, validAccountId, } from '../../src/core/credential/fake-creds.js';
|
|
16
|
+
const CLAUDE_PLACEHOLDER = 'PLACEHOLDER-claude-bearer-0123456789';
|
|
17
|
+
const CODEX_PLACEHOLDER = 'eyJhbGciOiJSUzI1NiJ9.PLACEHOLDER.codex';
|
|
18
|
+
const VALID_ACCOUNT_ID = '11111111-2222-3333-4444-555555555555';
|
|
19
|
+
function byPath(files, guestPath) {
|
|
20
|
+
const f = files.find((x) => x.guestPath === guestPath);
|
|
21
|
+
assert.ok(f, `expected a staged file at ${guestPath}`);
|
|
22
|
+
return f;
|
|
23
|
+
}
|
|
24
|
+
function baseInput(over) {
|
|
25
|
+
return {
|
|
26
|
+
adapter: 'claude',
|
|
27
|
+
placeholder: CLAUDE_PLACEHOLDER,
|
|
28
|
+
secretName: 'ANTHROPIC_AUTH_TOKEN',
|
|
29
|
+
claudeIdentity: null,
|
|
30
|
+
codexIdentity: null,
|
|
31
|
+
...over,
|
|
32
|
+
};
|
|
33
|
+
}
|
|
34
|
+
// ---------------------------------------------------------------------------
|
|
35
|
+
// env: always single-entry { [secretName]: placeholder }.
|
|
36
|
+
// ---------------------------------------------------------------------------
|
|
37
|
+
test('env is always { [secretName]: placeholder } for every adapter', () => {
|
|
38
|
+
const claude = buildGondolinFakeCreds(baseInput({ adapter: 'claude', secretName: 'ANTHROPIC_AUTH_TOKEN', placeholder: CLAUDE_PLACEHOLDER }));
|
|
39
|
+
assert.deepEqual(claude.env, { ANTHROPIC_AUTH_TOKEN: CLAUDE_PLACEHOLDER });
|
|
40
|
+
const codex = buildGondolinFakeCreds(baseInput({ adapter: 'codex', secretName: 'CODEX_AUTH', placeholder: CODEX_PLACEHOLDER }));
|
|
41
|
+
assert.deepEqual(codex.env, { CODEX_AUTH: CODEX_PLACEHOLDER });
|
|
42
|
+
});
|
|
43
|
+
// ---------------------------------------------------------------------------
|
|
44
|
+
// claude branch.
|
|
45
|
+
// ---------------------------------------------------------------------------
|
|
46
|
+
test('claude: no identity -> only .credentials.json, placeholder verbatim, junk refresh, far-future expiry', () => {
|
|
47
|
+
const { files } = buildGondolinFakeCreds(baseInput({ adapter: 'claude' }));
|
|
48
|
+
assert.equal(files.length, 1);
|
|
49
|
+
const creds = byPath(files, '/root/.claude/.credentials.json');
|
|
50
|
+
assert.equal(creds.mode, 0o600);
|
|
51
|
+
const parsed = JSON.parse(creds.content);
|
|
52
|
+
// Placeholder staged VERBATIM (exact-match egress substitution).
|
|
53
|
+
assert.equal(parsed.claudeAiOauth.accessToken, CLAUDE_PLACEHOLDER);
|
|
54
|
+
assert.equal(parsed.claudeAiOauth.refreshToken, 'JUNK-PLACEHOLDER-REFRESH-not-a-real-token');
|
|
55
|
+
assert.equal(parsed.claudeAiOauth.expiresAt, 4_102_444_800_000);
|
|
56
|
+
});
|
|
57
|
+
test('claude: identity present -> also stages scrubbed ~/.claude.json with oauthAccount UUIDs only', () => {
|
|
58
|
+
const files = buildClaudeFiles(CLAUDE_PLACEHOLDER, {
|
|
59
|
+
accountUuid: 'acct-uuid',
|
|
60
|
+
organizationUuid: 'org-uuid',
|
|
61
|
+
});
|
|
62
|
+
assert.equal(files.length, 2);
|
|
63
|
+
const config = byPath(files, '/root/.claude.json');
|
|
64
|
+
assert.equal(config.mode, 0o600);
|
|
65
|
+
const parsed = JSON.parse(config.content);
|
|
66
|
+
assert.equal(parsed.hasCompletedOnboarding, true);
|
|
67
|
+
assert.deepEqual(parsed.oauthAccount, { accountUuid: 'acct-uuid', organizationUuid: 'org-uuid' });
|
|
68
|
+
assert.deepEqual(parsed.projects, {});
|
|
69
|
+
// No token anywhere in the identity file.
|
|
70
|
+
assert.ok(!config.content.includes(CLAUDE_PLACEHOLDER));
|
|
71
|
+
});
|
|
72
|
+
// ---------------------------------------------------------------------------
|
|
73
|
+
// codex branch.
|
|
74
|
+
// ---------------------------------------------------------------------------
|
|
75
|
+
test('codex: complete identity -> all completeness fields present, placeholder as access+id token', () => {
|
|
76
|
+
const files = buildCodexFiles(CODEX_PLACEHOLDER, {
|
|
77
|
+
accountId: VALID_ACCOUNT_ID,
|
|
78
|
+
authMode: 'chatgpt',
|
|
79
|
+
lastRefresh: '2026-06-01T00:00:00Z',
|
|
80
|
+
});
|
|
81
|
+
assert.equal(files.length, 1);
|
|
82
|
+
const auth = byPath(files, '/root/.codex/auth.json');
|
|
83
|
+
assert.equal(auth.mode, 0o600);
|
|
84
|
+
const parsed = JSON.parse(auth.content);
|
|
85
|
+
assert.equal(parsed.OPENAI_API_KEY, null);
|
|
86
|
+
assert.equal(parsed.auth_mode, 'chatgpt');
|
|
87
|
+
assert.equal(parsed.last_refresh, '2026-06-01T00:00:00Z');
|
|
88
|
+
assert.equal(parsed.tokens.access_token, CODEX_PLACEHOLDER);
|
|
89
|
+
assert.equal(parsed.tokens.id_token, CODEX_PLACEHOLDER);
|
|
90
|
+
assert.equal(parsed.tokens.refresh_token, 'JUNK-PLACEHOLDER-REFRESH-not-a-real-token');
|
|
91
|
+
assert.equal(parsed.tokens.account_id, VALID_ACCOUNT_ID);
|
|
92
|
+
});
|
|
93
|
+
test('codex: null identity -> OPENAI_API_KEY+tokens only, no account_id/auth_mode/last_refresh', () => {
|
|
94
|
+
const files = buildCodexFiles(CODEX_PLACEHOLDER, null);
|
|
95
|
+
const auth = byPath(files, '/root/.codex/auth.json');
|
|
96
|
+
const parsed = JSON.parse(auth.content);
|
|
97
|
+
assert.equal(parsed.OPENAI_API_KEY, null);
|
|
98
|
+
assert.ok(!('auth_mode' in parsed));
|
|
99
|
+
assert.ok(!('last_refresh' in parsed));
|
|
100
|
+
assert.ok(!('account_id' in parsed.tokens));
|
|
101
|
+
// Bearer placeholder still staged verbatim.
|
|
102
|
+
assert.equal(parsed.tokens.access_token, CODEX_PLACEHOLDER);
|
|
103
|
+
});
|
|
104
|
+
test('codex defense-in-depth: token-shaped accountId / bad authMode / bad lastRefresh are OMITTED', () => {
|
|
105
|
+
const files = buildCodexFiles(CODEX_PLACEHOLDER, {
|
|
106
|
+
accountId: 'sk-not-a-uuid-but-token-shaped', // not a UUID -> omitted
|
|
107
|
+
authMode: 'bogus-mode', // not in the closed set -> omitted
|
|
108
|
+
lastRefresh: 'definitely not an iso timestamp with spaces', // fails ISO shape -> omitted
|
|
109
|
+
});
|
|
110
|
+
const auth = byPath(files, '/root/.codex/auth.json');
|
|
111
|
+
const parsed = JSON.parse(auth.content);
|
|
112
|
+
assert.ok(!('account_id' in parsed.tokens), 'token-shaped accountId must be omitted');
|
|
113
|
+
assert.ok(!('auth_mode' in parsed), 'unknown authMode must be omitted');
|
|
114
|
+
assert.ok(!('last_refresh' in parsed), 'malformed lastRefresh must be omitted');
|
|
115
|
+
});
|
|
116
|
+
test('validAccountId: UUID passes, token-shaped values are rejected', () => {
|
|
117
|
+
assert.equal(validAccountId(VALID_ACCOUNT_ID), VALID_ACCOUNT_ID);
|
|
118
|
+
assert.equal(validAccountId('sk-abc123'), null);
|
|
119
|
+
assert.equal(validAccountId('eyJhbGciOiJ.jwt.token'), null);
|
|
120
|
+
assert.equal(validAccountId(null), null);
|
|
121
|
+
});
|
|
122
|
+
// ---------------------------------------------------------------------------
|
|
123
|
+
// dispatch: buildGondolinFakeCreds routes to the right builder.
|
|
124
|
+
// ---------------------------------------------------------------------------
|
|
125
|
+
test('buildGondolinFakeCreds claude routes with identity from the input struct', () => {
|
|
126
|
+
const { files } = buildGondolinFakeCreds(baseInput({
|
|
127
|
+
adapter: 'claude',
|
|
128
|
+
claudeIdentity: { accountUuid: 'a', organizationUuid: 'o' },
|
|
129
|
+
}));
|
|
130
|
+
assert.equal(files.length, 2);
|
|
131
|
+
byPath(files, '/root/.claude/.credentials.json');
|
|
132
|
+
byPath(files, '/root/.claude.json');
|
|
133
|
+
});
|
|
134
|
+
//# sourceMappingURL=fake-creds.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"fake-creds.test.js","sourceRoot":"","sources":["../../../tests/core/fake-creds.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,197 @@
|
|
|
1
|
+
// Pure unit tests for the issue-file serializer core. No mocks — the function
|
|
2
|
+
// is total + synchronous; we feed plain data and assert on the returned value.
|
|
3
|
+
import { test } from 'node:test';
|
|
4
|
+
import assert from 'node:assert/strict';
|
|
5
|
+
import { composeIssueFile, } from '../../src/core/issue/file.js';
|
|
6
|
+
const NOW = '2026-06-02T12:00:00.000Z';
|
|
7
|
+
function baseInput(overrides = {}) {
|
|
8
|
+
return {
|
|
9
|
+
trackerRoot: '/tmp/tracker',
|
|
10
|
+
state: 'Todo',
|
|
11
|
+
title: 'Fix the thing',
|
|
12
|
+
existingIdentifiers: new Set(),
|
|
13
|
+
nowIso: NOW,
|
|
14
|
+
...overrides,
|
|
15
|
+
};
|
|
16
|
+
}
|
|
17
|
+
function expectOk(r) {
|
|
18
|
+
assert.equal(r.ok, true, 'expected ok result');
|
|
19
|
+
if (!r.ok)
|
|
20
|
+
throw new Error('unreachable');
|
|
21
|
+
return r.result;
|
|
22
|
+
}
|
|
23
|
+
function expectErr(r) {
|
|
24
|
+
assert.equal(r.ok, false, 'expected error result');
|
|
25
|
+
if (r.ok)
|
|
26
|
+
throw new Error('unreachable');
|
|
27
|
+
return r.error;
|
|
28
|
+
}
|
|
29
|
+
// ── numeric next-id picking (no explicit identifier) ────────────────────────
|
|
30
|
+
test('no identifier + empty set → picks "1"', () => {
|
|
31
|
+
const res = expectOk(composeIssueFile(baseInput()));
|
|
32
|
+
assert.equal(res.identifier, '1');
|
|
33
|
+
assert.equal(res.state, 'Todo');
|
|
34
|
+
assert.equal(res.path, '/tmp/tracker/Todo/1.md');
|
|
35
|
+
});
|
|
36
|
+
test('no identifier + set {1,2} → picks first free "3"', () => {
|
|
37
|
+
const res = expectOk(composeIssueFile(baseInput({ existingIdentifiers: new Set(['1', '2']) })));
|
|
38
|
+
assert.equal(res.identifier, '3');
|
|
39
|
+
assert.equal(res.path, '/tmp/tracker/Todo/3.md');
|
|
40
|
+
});
|
|
41
|
+
test('no identifier ignores non-numeric (slug) ids when picking', () => {
|
|
42
|
+
const res = expectOk(composeIssueFile(baseInput({ existingIdentifiers: new Set(['add-changelog', 'some-slug']) })));
|
|
43
|
+
// Slug files are inert — "1" is still free.
|
|
44
|
+
assert.equal(res.identifier, '1');
|
|
45
|
+
});
|
|
46
|
+
test('no identifier + gap {1,3} → picks "2"', () => {
|
|
47
|
+
const res = expectOk(composeIssueFile(baseInput({ existingIdentifiers: new Set(['1', '3']) })));
|
|
48
|
+
assert.equal(res.identifier, '2');
|
|
49
|
+
});
|
|
50
|
+
// ── explicit identifier sanitize + collision ───────────────────────────────
|
|
51
|
+
test('explicit identifier is sanitized (illegal chars → "_")', () => {
|
|
52
|
+
const res = expectOk(composeIssueFile(baseInput({ identifier: 'my issue/name!' })));
|
|
53
|
+
assert.equal(res.identifier, 'my_issue_name_');
|
|
54
|
+
assert.equal(res.path, '/tmp/tracker/Todo/my_issue_name_.md');
|
|
55
|
+
});
|
|
56
|
+
test('explicit identifier preserving allowed chars [A-Za-z0-9._-]', () => {
|
|
57
|
+
const res = expectOk(composeIssueFile(baseInput({ identifier: 'Feature_42.v-1' })));
|
|
58
|
+
assert.equal(res.identifier, 'Feature_42.v-1');
|
|
59
|
+
});
|
|
60
|
+
test('explicit identifier trims surrounding whitespace', () => {
|
|
61
|
+
const res = expectOk(composeIssueFile(baseInput({ identifier: ' bug-7 ' })));
|
|
62
|
+
assert.equal(res.identifier, 'bug-7');
|
|
63
|
+
});
|
|
64
|
+
test('all-illegal explicit identifier sanitizes to underscores (NOT empty_identifier)', () => {
|
|
65
|
+
// Every illegal char becomes "_", so sanitize can never yield "" for a
|
|
66
|
+
// non-blank trimmed input. The empty_identifier branch is preserved from the
|
|
67
|
+
// original for fidelity but is effectively unreachable via this path: '@@@'
|
|
68
|
+
// → '___', a valid (non-empty) identifier.
|
|
69
|
+
const res = expectOk(composeIssueFile(baseInput({ identifier: '@@@' })));
|
|
70
|
+
assert.equal(res.identifier, '___');
|
|
71
|
+
assert.equal(res.path, '/tmp/tracker/Todo/___.md');
|
|
72
|
+
});
|
|
73
|
+
test('blank/whitespace identifier falls through to numeric pick (not error)', () => {
|
|
74
|
+
const res = expectOk(composeIssueFile(baseInput({ identifier: ' ' })));
|
|
75
|
+
assert.equal(res.identifier, '1');
|
|
76
|
+
});
|
|
77
|
+
test('explicit identifier already present → identifier_exists error', () => {
|
|
78
|
+
const err = expectErr(composeIssueFile(baseInput({
|
|
79
|
+
identifier: 'bug-7',
|
|
80
|
+
existingIdentifiers: new Set(['bug-7']),
|
|
81
|
+
})));
|
|
82
|
+
assert.equal(err.code, 'identifier_exists');
|
|
83
|
+
if (err.code === 'identifier_exists') {
|
|
84
|
+
assert.equal(err.identifier, 'bug-7');
|
|
85
|
+
assert.equal(err.path, '/tmp/tracker/Todo/bug-7.md');
|
|
86
|
+
assert.match(err.message, /already exists/);
|
|
87
|
+
}
|
|
88
|
+
});
|
|
89
|
+
test('explicit identifier collision compares the SANITIZED form', () => {
|
|
90
|
+
// "bug 7" sanitizes to "bug_7"; a pre-existing "bug_7" must collide.
|
|
91
|
+
const err = expectErr(composeIssueFile(baseInput({
|
|
92
|
+
identifier: 'bug 7',
|
|
93
|
+
existingIdentifiers: new Set(['bug_7']),
|
|
94
|
+
})));
|
|
95
|
+
assert.equal(err.code, 'identifier_exists');
|
|
96
|
+
});
|
|
97
|
+
// ── front matter rendering ──────────────────────────────────────────────────
|
|
98
|
+
test('renders standard front matter in declaration order with JSON-quoted strings', () => {
|
|
99
|
+
const res = expectOk(composeIssueFile(baseInput({ identifier: '1' })));
|
|
100
|
+
const expectedFm = [
|
|
101
|
+
'---',
|
|
102
|
+
'id: "1"',
|
|
103
|
+
'identifier: "1"',
|
|
104
|
+
'title: "Fix the thing"',
|
|
105
|
+
`created_at: ${JSON.stringify(NOW)}`,
|
|
106
|
+
`updated_at: ${JSON.stringify(NOW)}`,
|
|
107
|
+
'---',
|
|
108
|
+
'',
|
|
109
|
+
].join('\n');
|
|
110
|
+
assert.equal(res.content, expectedFm);
|
|
111
|
+
});
|
|
112
|
+
test('priority (number) is rendered unquoted; null/undefined omitted', () => {
|
|
113
|
+
const withPrio = expectOk(composeIssueFile(baseInput({ identifier: '1', priority: 3 })));
|
|
114
|
+
assert.match(withPrio.content, /\npriority: 3\n/);
|
|
115
|
+
const nullPrio = expectOk(composeIssueFile(baseInput({ identifier: '1', priority: null })));
|
|
116
|
+
assert.doesNotMatch(nullPrio.content, /priority:/);
|
|
117
|
+
const noPrio = expectOk(composeIssueFile(baseInput({ identifier: '1' })));
|
|
118
|
+
assert.doesNotMatch(noPrio.content, /priority:/);
|
|
119
|
+
});
|
|
120
|
+
test('labels and blocked_by render as JSON arrays only when non-empty', () => {
|
|
121
|
+
const res = expectOk(composeIssueFile(baseInput({ identifier: '1', labels: ['a', 'b'], blocked_by: ['5'] })));
|
|
122
|
+
assert.match(res.content, /\nlabels: \["a", "b"\]\n/);
|
|
123
|
+
assert.match(res.content, /\nblocked_by: \["5"\]\n/);
|
|
124
|
+
const empty = expectOk(composeIssueFile(baseInput({ identifier: '1', labels: [], blocked_by: [] })));
|
|
125
|
+
assert.doesNotMatch(empty.content, /labels:/);
|
|
126
|
+
assert.doesNotMatch(empty.content, /blocked_by:/);
|
|
127
|
+
});
|
|
128
|
+
test('extra_front_matter (propose_issue provenance) is appended after standard set', () => {
|
|
129
|
+
const res = expectOk(composeIssueFile(baseInput({
|
|
130
|
+
identifier: '1',
|
|
131
|
+
extra_front_matter: { proposed_by: '42', proposed_at: NOW, dup: true },
|
|
132
|
+
})));
|
|
133
|
+
assert.match(res.content, /\nproposed_by: "42"\n/);
|
|
134
|
+
assert.match(res.content, new RegExp(`\\nproposed_at: ${JSON.stringify(NOW)}\\n`));
|
|
135
|
+
assert.match(res.content, /\ndup: true\n/);
|
|
136
|
+
// Appended AFTER updated_at.
|
|
137
|
+
const idxUpdated = res.content.indexOf('updated_at:');
|
|
138
|
+
const idxProposed = res.content.indexOf('proposed_by:');
|
|
139
|
+
assert.ok(idxUpdated < idxProposed, 'extras must come after standard set');
|
|
140
|
+
});
|
|
141
|
+
test('extra_front_matter can override a standard key', () => {
|
|
142
|
+
const res = expectOk(composeIssueFile(baseInput({ identifier: '1', title: 'orig', extra_front_matter: { title: 'override' } })));
|
|
143
|
+
// Single title line, with the override value (Object key reuse preserves position).
|
|
144
|
+
const titleLines = res.content.split('\n').filter((l) => l.startsWith('title:'));
|
|
145
|
+
assert.equal(titleLines.length, 1);
|
|
146
|
+
assert.equal(titleLines[0], 'title: "override"');
|
|
147
|
+
});
|
|
148
|
+
// ── body composition ─────────────────────────────────────────────────────────
|
|
149
|
+
test('description body is trimmed and appended with a trailing newline', () => {
|
|
150
|
+
const res = expectOk(composeIssueFile(baseInput({ identifier: '1', description: ' hello world ' })));
|
|
151
|
+
// Front matter ends `...---\n` (the trailing '' element adds one newline after
|
|
152
|
+
// the closing fence), then the trimmed body + a final newline — no blank line
|
|
153
|
+
// between the fence and the body, matching the original serializer.
|
|
154
|
+
assert.ok(res.content.endsWith('---\nhello world\n'), res.content);
|
|
155
|
+
});
|
|
156
|
+
test('blank description produces no body (front matter only)', () => {
|
|
157
|
+
const res = expectOk(composeIssueFile(baseInput({ identifier: '1', description: ' ' })));
|
|
158
|
+
assert.ok(res.content.endsWith('---\n'), res.content);
|
|
159
|
+
// Nothing after the closing fence.
|
|
160
|
+
const afterFence = res.content.slice(res.content.lastIndexOf('---\n') + 4);
|
|
161
|
+
assert.equal(afterFence, '');
|
|
162
|
+
});
|
|
163
|
+
// ── emitted effect ───────────────────────────────────────────────────────────
|
|
164
|
+
test('returns exactly one tracker.write_issue Effect carrying the inputs', () => {
|
|
165
|
+
const res = expectOk(composeIssueFile(baseInput({
|
|
166
|
+
identifier: 'bug 7',
|
|
167
|
+
description: 'body',
|
|
168
|
+
priority: 2,
|
|
169
|
+
labels: ['x'],
|
|
170
|
+
blocked_by: ['9'],
|
|
171
|
+
extra_front_matter: { proposed_by: '1' },
|
|
172
|
+
})));
|
|
173
|
+
assert.equal(res.effects.length, 1);
|
|
174
|
+
const eff = res.effects[0];
|
|
175
|
+
assert.equal(eff?.family, 'tracker');
|
|
176
|
+
if (eff?.family === 'tracker' && eff.kind === 'write_issue') {
|
|
177
|
+
assert.equal(eff.trackerRoot, '/tmp/tracker');
|
|
178
|
+
assert.equal(eff.state, 'Todo');
|
|
179
|
+
assert.equal(eff.title, 'Fix the thing');
|
|
180
|
+
// Effect carries the SANITIZED identifier (matches the chosen id).
|
|
181
|
+
assert.equal(eff.identifier, 'bug_7');
|
|
182
|
+
assert.equal(eff.identifier, res.identifier);
|
|
183
|
+
assert.equal(eff.description, 'body');
|
|
184
|
+
assert.equal(eff.priority, 2);
|
|
185
|
+
assert.deepEqual(eff.labels, ['x']);
|
|
186
|
+
assert.deepEqual(eff.blocked_by, ['9']);
|
|
187
|
+
assert.deepEqual(eff.extraFrontMatter, { proposed_by: '1' });
|
|
188
|
+
}
|
|
189
|
+
else {
|
|
190
|
+
assert.fail('expected tracker.write_issue effect');
|
|
191
|
+
}
|
|
192
|
+
});
|
|
193
|
+
test('path is POSIX-joined and collapses a trailing slash on the root', () => {
|
|
194
|
+
const res = expectOk(composeIssueFile(baseInput({ trackerRoot: '/tmp/tracker/', identifier: '1' })));
|
|
195
|
+
assert.equal(res.path, '/tmp/tracker/Todo/1.md');
|
|
196
|
+
});
|
|
197
|
+
//# sourceMappingURL=file.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"file.test.js","sourceRoot":"","sources":["../../../tests/core/file.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,113 @@
|
|
|
1
|
+
// Pure unit tests for the git push / gh PR-create result classifiers (§4a). These
|
|
2
|
+
// were stranded inline in the IO effect-interpreter; pinned here so the rejection
|
|
3
|
+
// regex + PR-number parse can't silently drift from the behavior they replaced.
|
|
4
|
+
import { test } from 'node:test';
|
|
5
|
+
import assert from 'node:assert/strict';
|
|
6
|
+
import { classifyPushResult, parsePrNumberFromUrl, isWorkflowScopeDecline, describeHandoffFailure, } from '../../src/core/git/result.js';
|
|
7
|
+
function result(over) {
|
|
8
|
+
return { ran: true, exit_code: 0, signal: null, timed_out: false, stdout: '', stderr: '', ...over };
|
|
9
|
+
}
|
|
10
|
+
// ─── classifyPushResult ──────────────────────────────────────────────────────
|
|
11
|
+
test('classifyPushResult: exit 0 is clean and never non-fast-forward', () => {
|
|
12
|
+
assert.deepEqual(classifyPushResult(result({ exit_code: 0 })), { clean: true, nonFastForward: false });
|
|
13
|
+
// Even if the (successful) output happens to contain the word "rejected", a
|
|
14
|
+
// clean push is never flagged NFF.
|
|
15
|
+
assert.deepEqual(classifyPushResult(result({ exit_code: 0, stderr: 'remote: rejected nothing' })), { clean: true, nonFastForward: false });
|
|
16
|
+
});
|
|
17
|
+
test('classifyPushResult: a non-fast-forward rejection is flagged from stderr', () => {
|
|
18
|
+
const r = result({
|
|
19
|
+
exit_code: 1,
|
|
20
|
+
stderr: ' ! [rejected] main -> main (non-fast-forward)\nerror: failed to push some refs',
|
|
21
|
+
});
|
|
22
|
+
assert.deepEqual(classifyPushResult(r), { clean: false, nonFastForward: true });
|
|
23
|
+
});
|
|
24
|
+
test('classifyPushResult: each git phrasing of the rejection matches', () => {
|
|
25
|
+
for (const phrase of ['non-fast-forward', 'fetch first', '(non-fast-forward)', 'Updates were rejected']) {
|
|
26
|
+
const r = classifyPushResult(result({ exit_code: 1, stderr: phrase }));
|
|
27
|
+
assert.equal(r.nonFastForward, true, `phrase "${phrase}" should be NFF`);
|
|
28
|
+
}
|
|
29
|
+
});
|
|
30
|
+
test('classifyPushResult: a non-NFF failure is not flagged (clean=false, NFF=false)', () => {
|
|
31
|
+
const r = result({ exit_code: 128, stderr: 'fatal: repository not found' });
|
|
32
|
+
assert.deepEqual(classifyPushResult(r), { clean: false, nonFastForward: false });
|
|
33
|
+
});
|
|
34
|
+
test('classifyPushResult: a server-side [remote rejected] decline is NOT non-fast-forward (227)', () => {
|
|
35
|
+
// The 227 case: GitHub declines a `.github/workflows/` edit because the PAT
|
|
36
|
+
// lacks `workflow` scope. The message contains the bare word "rejected", which
|
|
37
|
+
// the old regex false-positived into NFF recovery (its fetch then masked this
|
|
38
|
+
// real reason with "couldn't find remote ref"). It must now classify NOT NFF so
|
|
39
|
+
// the push fails fast with this actionable stderr.
|
|
40
|
+
const r = result({
|
|
41
|
+
exit_code: 1,
|
|
42
|
+
stderr: ' ! [remote rejected] agent/224 -> agent/224 (refusing to allow a Personal Access Token ' +
|
|
43
|
+
'to create or update workflow `.github/workflows/release.yml` without `workflow` scope)\n' +
|
|
44
|
+
'error: failed to push some refs',
|
|
45
|
+
});
|
|
46
|
+
assert.deepEqual(classifyPushResult(r), { clean: false, nonFastForward: false });
|
|
47
|
+
});
|
|
48
|
+
test('classifyPushResult: a (push declined …) pre-receive decline is NOT non-fast-forward', () => {
|
|
49
|
+
const r = result({
|
|
50
|
+
exit_code: 1,
|
|
51
|
+
stderr: ' ! [remote rejected] agent/224 -> agent/224 (push declined due to repository rule violations)',
|
|
52
|
+
});
|
|
53
|
+
assert.deepEqual(classifyPushResult(r), { clean: false, nonFastForward: false });
|
|
54
|
+
});
|
|
55
|
+
test('classifyPushResult: matches across the combined stdout+stderr', () => {
|
|
56
|
+
// The original tested `${stdout}\n${stderr}`; a phrase split or living on stdout
|
|
57
|
+
// still matches.
|
|
58
|
+
const r = classifyPushResult(result({ exit_code: 1, stdout: 'hint: fetch first', stderr: '' }));
|
|
59
|
+
assert.equal(r.nonFastForward, true);
|
|
60
|
+
});
|
|
61
|
+
// ─── parsePrNumberFromUrl ────────────────────────────────────────────────────
|
|
62
|
+
test('parsePrNumberFromUrl: takes the last line and parses /pull/<n>', () => {
|
|
63
|
+
// gh prints a couple of progress lines then the URL as the final line.
|
|
64
|
+
const stdout = '\nCreating pull request for feat into main\nhttps://github.com/o/r/pull/4321\n';
|
|
65
|
+
assert.deepEqual(parsePrNumberFromUrl(stdout), { number: 4321, url: 'https://github.com/o/r/pull/4321' });
|
|
66
|
+
});
|
|
67
|
+
test('parsePrNumberFromUrl: a single-line URL parses', () => {
|
|
68
|
+
assert.deepEqual(parsePrNumberFromUrl('https://github.com/o/r/pull/7'), {
|
|
69
|
+
number: 7,
|
|
70
|
+
url: 'https://github.com/o/r/pull/7',
|
|
71
|
+
});
|
|
72
|
+
});
|
|
73
|
+
test('parsePrNumberFromUrl: a URL with no /pull/<n> surfaces the URL with number 0', () => {
|
|
74
|
+
assert.deepEqual(parsePrNumberFromUrl('https://github.com/o/r'), { number: 0, url: 'https://github.com/o/r' });
|
|
75
|
+
});
|
|
76
|
+
test('parsePrNumberFromUrl: empty stdout yields number 0 and empty url', () => {
|
|
77
|
+
assert.deepEqual(parsePrNumberFromUrl(''), { number: 0, url: '' });
|
|
78
|
+
assert.deepEqual(parsePrNumberFromUrl(' \n \n'), { number: 0, url: '' });
|
|
79
|
+
});
|
|
80
|
+
// ─── isWorkflowScopeDecline / describeHandoffFailure (issue 235) ──────────────
|
|
81
|
+
const WORKFLOW_DECLINE = ' ! [remote rejected] agent/235 -> agent/235 (refusing to allow a Personal Access Token ' +
|
|
82
|
+
'to create or update workflow `.github/workflows/ci.yml` without `workflow` scope)\n' +
|
|
83
|
+
'error: failed to push some refs';
|
|
84
|
+
test('isWorkflowScopeDecline: the GitHub workflow-scope server-side decline matches', () => {
|
|
85
|
+
assert.equal(isWorkflowScopeDecline(WORKFLOW_DECLINE), true);
|
|
86
|
+
// The trailing "without `workflow` scope" clause alone is enough.
|
|
87
|
+
assert.equal(isWorkflowScopeDecline('push declined: missing `workflow` scope'), true);
|
|
88
|
+
// An OAuth-app variant of the same refusal.
|
|
89
|
+
assert.equal(isWorkflowScopeDecline('refusing to allow an OAuth App to create or update workflow `x.yml`'), true);
|
|
90
|
+
});
|
|
91
|
+
test('isWorkflowScopeDecline: a non-fast-forward / generic failure does NOT match', () => {
|
|
92
|
+
assert.equal(isWorkflowScopeDecline(' ! [rejected] main -> main (non-fast-forward)'), false);
|
|
93
|
+
assert.equal(isWorkflowScopeDecline('fatal: unable to access ... Connection timed out'), false);
|
|
94
|
+
// A bare branch-protection decline (no workflow scope) is a different handoff
|
|
95
|
+
// failure and is left to pass through unchanged.
|
|
96
|
+
assert.equal(isWorkflowScopeDecline('push declined due to repository rule violations'), false);
|
|
97
|
+
});
|
|
98
|
+
test('describeHandoffFailure: a workflow-scope decline is rewritten to "manual SSH push required", keeping the raw error', () => {
|
|
99
|
+
const out = describeHandoffFailure(WORKFLOW_DECLINE);
|
|
100
|
+
assert.ok(out.startsWith('manual SSH push required'), 'leads with the actionable instruction');
|
|
101
|
+
assert.ok(out.includes('Workflows:write'), 'names the missing permission');
|
|
102
|
+
assert.ok(out.includes('.github/workflows/'), 'names the triggering path');
|
|
103
|
+
assert.ok(out.includes(WORKFLOW_DECLINE), 'appends the original git error verbatim');
|
|
104
|
+
});
|
|
105
|
+
test('describeHandoffFailure: a non-workflow reason passes through unchanged', () => {
|
|
106
|
+
const nff = ' ! [rejected] main -> main (non-fast-forward)';
|
|
107
|
+
assert.equal(describeHandoffFailure(nff), nff);
|
|
108
|
+
});
|
|
109
|
+
test('describeHandoffFailure: idempotent — an already-rewritten reason is not double-wrapped', () => {
|
|
110
|
+
const once = describeHandoffFailure(WORKFLOW_DECLINE);
|
|
111
|
+
assert.equal(describeHandoffFailure(once), once);
|
|
112
|
+
});
|
|
113
|
+
//# sourceMappingURL=git-result.test.js.map
|