tulingcode 0.1.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 +134 -0
- package/Dockerfile +18 -0
- package/README.md +15 -0
- package/bin/tuling +179 -0
- package/bunfig.toml +7 -0
- package/drizzle.config.ts +10 -0
- package/git +0 -0
- package/migration/20260127222353_familiar_lady_ursula/migration.sql +90 -0
- package/migration/20260127222353_familiar_lady_ursula/snapshot.json +796 -0
- package/migration/20260211171708_add_project_commands/migration.sql +1 -0
- package/migration/20260211171708_add_project_commands/snapshot.json +806 -0
- package/migration/20260213144116_wakeful_the_professor/migration.sql +11 -0
- package/migration/20260213144116_wakeful_the_professor/snapshot.json +897 -0
- package/migration/20260225215848_workspace/migration.sql +7 -0
- package/migration/20260225215848_workspace/snapshot.json +959 -0
- package/migration/20260227213759_add_session_workspace_id/migration.sql +2 -0
- package/migration/20260227213759_add_session_workspace_id/snapshot.json +983 -0
- package/migration/20260228203230_blue_harpoon/migration.sql +17 -0
- package/migration/20260228203230_blue_harpoon/snapshot.json +1102 -0
- package/migration/20260303231226_add_workspace_fields/migration.sql +5 -0
- package/migration/20260303231226_add_workspace_fields/snapshot.json +1013 -0
- package/migration/20260309230000_move_org_to_state/migration.sql +3 -0
- package/migration/20260309230000_move_org_to_state/snapshot.json +1156 -0
- package/migration/20260312043431_session_message_cursor/migration.sql +4 -0
- package/migration/20260312043431_session_message_cursor/snapshot.json +1168 -0
- package/migration/20260323234822_events/migration.sql +13 -0
- package/migration/20260323234822_events/snapshot.json +1271 -0
- package/migration/20260410174513_workspace-name/migration.sql +16 -0
- package/migration/20260410174513_workspace-name/snapshot.json +1271 -0
- package/migration/20260413175956_chief_energizer/migration.sql +13 -0
- package/migration/20260413175956_chief_energizer/snapshot.json +1399 -0
- package/migration/20260422160000_context_inheritance/migration.sql +3 -0
- package/migration/20260422170000_task_registry/migration.sql +18 -0
- package/migration/20260423145421_remove_session_entry/migration.sql +4 -0
- package/migration/20260515000000_actor_rename/migration.sql +7 -0
- package/migration/20260515010000_memory_fts/migration.sql +33 -0
- package/migration/20260515020000_user_task/migration.sql +29 -0
- package/migration/20260519000000_last_checkpoint_message_id/migration.sql +1 -0
- package/migration/20260521000000_message_agent_id/migration.sql +2 -0
- package/migration/20260521000100_actor_registry_v6/migration.sql +25 -0
- package/migration/20260521010000_memory_fts_v6/migration.sql +33 -0
- package/migration/20260521020000_memory_fts_triggers/migration.sql +17 -0
- package/migration/20260526000000_agent_id_main/migration.sql +14 -0
- package/migration/20260527000000_actor_lifecycle/migration.sql +8 -0
- package/migration/20260527000100_inbox/migration.sql +12 -0
- package/migration/20260529000000_task_todo_redesign/migration.sql +16 -0
- package/migration/20260603000000_task_in_progress_owner/migration.sql +1 -0
- package/migration/20260603000000_workflow_run/migration.sql +17 -0
- package/migration/20260604000000_workflow_script_sha/migration.sql +1 -0
- package/migration/20260608000000_claude_import/migration.sql +7 -0
- package/migration/20260608010000_claude_import_message_ids/migration.sql +1 -0
- package/migration/20260609000000_history_fts/migration.sql +29 -0
- package/migration/20260609230000_workflow_agent_timeout/migration.sql +1 -0
- package/package.json +196 -0
- package/parsers-config.ts +290 -0
- package/script/build.ts +267 -0
- package/script/check-migrations.ts +16 -0
- package/script/fix-node-pty.ts +28 -0
- package/script/generate.ts +23 -0
- package/script/postinstall.mjs +102 -0
- package/script/publish.ts +60 -0
- package/script/run-workspace-server +106 -0
- package/script/schema.ts +63 -0
- package/script/time.ts +6 -0
- package/script/trace-imports.ts +153 -0
- package/script/upgrade-opentui.ts +64 -0
- package/src/account/account.sql.ts +39 -0
- package/src/account/account.ts +456 -0
- package/src/account/repo.ts +166 -0
- package/src/account/schema.ts +99 -0
- package/src/account/url.ts +8 -0
- package/src/acp/README.md +174 -0
- package/src/acp/agent.ts +1783 -0
- package/src/acp/session.ts +116 -0
- package/src/acp/types.ts +24 -0
- package/src/actor/actor.sql.ts +38 -0
- package/src/actor/events.ts +67 -0
- package/src/actor/index.ts +2 -0
- package/src/actor/registry.ts +412 -0
- package/src/actor/return-header.ts +24 -0
- package/src/actor/schema.ts +47 -0
- package/src/actor/spawn-ref.ts +16 -0
- package/src/actor/spawn.ts +741 -0
- package/src/actor/turn.ts +49 -0
- package/src/actor/waiter.ts +166 -0
- package/src/agent/agent.ts +554 -0
- package/src/agent/config.ts +5 -0
- package/src/agent/generate.txt +75 -0
- package/src/agent/prompt/checkpoint-writer.txt +167 -0
- package/src/agent/prompt/compaction.txt +9 -0
- package/src/agent/prompt/distill.txt +199 -0
- package/src/agent/prompt/dream.txt +155 -0
- package/src/agent/prompt/explore.txt +18 -0
- package/src/agent/prompt/summary.txt +11 -0
- package/src/agent/prompt/title.txt +44 -0
- package/src/audio.d.ts +9 -0
- package/src/auth/index.ts +97 -0
- package/src/bus/bus-event.ts +33 -0
- package/src/bus/global.ts +12 -0
- package/src/bus/index.ts +193 -0
- package/src/cli/bootstrap.ts +33 -0
- package/src/cli/cmd/account.ts +258 -0
- package/src/cli/cmd/acp.ts +70 -0
- package/src/cli/cmd/agent.ts +248 -0
- package/src/cli/cmd/cmd.ts +7 -0
- package/src/cli/cmd/db.ts +120 -0
- package/src/cli/cmd/debug/agent.ts +192 -0
- package/src/cli/cmd/debug/config.ts +17 -0
- package/src/cli/cmd/debug/file.ts +100 -0
- package/src/cli/cmd/debug/index.ts +48 -0
- package/src/cli/cmd/debug/lsp.ts +61 -0
- package/src/cli/cmd/debug/ripgrep.ts +105 -0
- package/src/cli/cmd/debug/scrap.ts +16 -0
- package/src/cli/cmd/debug/skill.ts +23 -0
- package/src/cli/cmd/debug/snapshot.ts +53 -0
- package/src/cli/cmd/export.ts +306 -0
- package/src/cli/cmd/generate.ts +50 -0
- package/src/cli/cmd/github.ts +1647 -0
- package/src/cli/cmd/import.ts +208 -0
- package/src/cli/cmd/mcp.ts +812 -0
- package/src/cli/cmd/models.ts +88 -0
- package/src/cli/cmd/plug.ts +233 -0
- package/src/cli/cmd/pr.ts +138 -0
- package/src/cli/cmd/providers.ts +705 -0
- package/src/cli/cmd/run-completion.ts +77 -0
- package/src/cli/cmd/run.ts +694 -0
- package/src/cli/cmd/serve.ts +21 -0
- package/src/cli/cmd/session.ts +181 -0
- package/src/cli/cmd/stats.ts +413 -0
- package/src/cli/cmd/tui/app.tsx +1130 -0
- package/src/cli/cmd/tui/asset/TEN_VAD_LICENSE +12 -0
- package/src/cli/cmd/tui/asset/charge.wav +0 -0
- package/src/cli/cmd/tui/asset/pulse-a.wav +0 -0
- package/src/cli/cmd/tui/asset/pulse-b.wav +0 -0
- package/src/cli/cmd/tui/asset/pulse-c.wav +0 -0
- package/src/cli/cmd/tui/asset/ten_vad.wasm +0 -0
- package/src/cli/cmd/tui/asset/ten_vad_loader.js +30 -0
- package/src/cli/cmd/tui/attach.ts +83 -0
- package/src/cli/cmd/tui/component/background-image.tsx +150 -0
- package/src/cli/cmd/tui/component/bg-pulse.tsx +130 -0
- package/src/cli/cmd/tui/component/border.tsx +21 -0
- package/src/cli/cmd/tui/component/dialog-agent.tsx +31 -0
- package/src/cli/cmd/tui/component/dialog-command.tsx +208 -0
- package/src/cli/cmd/tui/component/dialog-console-org.tsx +103 -0
- package/src/cli/cmd/tui/component/dialog-go-upsell.tsx +157 -0
- package/src/cli/cmd/tui/component/dialog-image-list.tsx +111 -0
- package/src/cli/cmd/tui/component/dialog-logo-design.tsx +37 -0
- package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
- package/src/cli/cmd/tui/component/dialog-mimo-login.tsx +224 -0
- package/src/cli/cmd/tui/component/dialog-model.tsx +253 -0
- package/src/cli/cmd/tui/component/dialog-provider.tsx +490 -0
- package/src/cli/cmd/tui/component/dialog-session-delete-failed.tsx +101 -0
- package/src/cli/cmd/tui/component/dialog-session-list.tsx +269 -0
- package/src/cli/cmd/tui/component/dialog-session-rename.tsx +31 -0
- package/src/cli/cmd/tui/component/dialog-skill.tsx +42 -0
- package/src/cli/cmd/tui/component/dialog-stash.tsx +87 -0
- package/src/cli/cmd/tui/component/dialog-status.tsx +170 -0
- package/src/cli/cmd/tui/component/dialog-tag.tsx +44 -0
- package/src/cli/cmd/tui/component/dialog-theme-list.tsx +50 -0
- package/src/cli/cmd/tui/component/dialog-variant.tsx +39 -0
- package/src/cli/cmd/tui/component/dialog-workflows.tsx +62 -0
- package/src/cli/cmd/tui/component/dialog-workspace-create.tsx +289 -0
- package/src/cli/cmd/tui/component/dialog-workspace-unavailable.tsx +81 -0
- package/src/cli/cmd/tui/component/dialog-worktree.tsx +90 -0
- package/src/cli/cmd/tui/component/error-component.tsx +92 -0
- package/src/cli/cmd/tui/component/logo.tsx +961 -0
- package/src/cli/cmd/tui/component/plugin-route-missing.tsx +14 -0
- package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +684 -0
- package/src/cli/cmd/tui/component/prompt/cwd.ts +0 -0
- package/src/cli/cmd/tui/component/prompt/frecency.tsx +90 -0
- package/src/cli/cmd/tui/component/prompt/history.tsx +108 -0
- package/src/cli/cmd/tui/component/prompt/index.tsx +1812 -0
- package/src/cli/cmd/tui/component/prompt/part.ts +16 -0
- package/src/cli/cmd/tui/component/prompt/stash.tsx +101 -0
- package/src/cli/cmd/tui/component/spinner.tsx +24 -0
- package/src/cli/cmd/tui/component/starry-background.tsx +305 -0
- package/src/cli/cmd/tui/component/startup-loading.tsx +67 -0
- package/src/cli/cmd/tui/component/task-item.tsx +63 -0
- package/src/cli/cmd/tui/component/textarea-keybindings.ts +73 -0
- package/src/cli/cmd/tui/component/todo-item.tsx +32 -0
- package/src/cli/cmd/tui/config/cwd.ts +5 -0
- package/src/cli/cmd/tui/config/tui-migrate.ts +151 -0
- package/src/cli/cmd/tui/config/tui-schema.ts +38 -0
- package/src/cli/cmd/tui/config/tui.ts +219 -0
- package/src/cli/cmd/tui/context/args.tsx +16 -0
- package/src/cli/cmd/tui/context/directory.ts +15 -0
- package/src/cli/cmd/tui/context/event.ts +45 -0
- package/src/cli/cmd/tui/context/exit.tsx +65 -0
- package/src/cli/cmd/tui/context/helper.tsx +25 -0
- package/src/cli/cmd/tui/context/keybind.tsx +105 -0
- package/src/cli/cmd/tui/context/kv.tsx +76 -0
- package/src/cli/cmd/tui/context/language.tsx +91 -0
- package/src/cli/cmd/tui/context/local.tsx +455 -0
- package/src/cli/cmd/tui/context/plugin-keybinds.ts +41 -0
- package/src/cli/cmd/tui/context/project.tsx +109 -0
- package/src/cli/cmd/tui/context/prompt.tsx +18 -0
- package/src/cli/cmd/tui/context/route.tsx +61 -0
- package/src/cli/cmd/tui/context/sdk.tsx +150 -0
- package/src/cli/cmd/tui/context/sync.tsx +828 -0
- package/src/cli/cmd/tui/context/theme/aura.json +69 -0
- package/src/cli/cmd/tui/context/theme/ayu.json +80 -0
- package/src/cli/cmd/tui/context/theme/carbonfox.json +248 -0
- package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +230 -0
- package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +230 -0
- package/src/cli/cmd/tui/context/theme/catppuccin.json +112 -0
- package/src/cli/cmd/tui/context/theme/cobalt2.json +225 -0
- package/src/cli/cmd/tui/context/theme/cursor.json +249 -0
- package/src/cli/cmd/tui/context/theme/dracula.json +219 -0
- package/src/cli/cmd/tui/context/theme/everforest.json +241 -0
- package/src/cli/cmd/tui/context/theme/flexoki.json +237 -0
- package/src/cli/cmd/tui/context/theme/github.json +233 -0
- package/src/cli/cmd/tui/context/theme/gruvbox.json +242 -0
- package/src/cli/cmd/tui/context/theme/kanagawa.json +77 -0
- package/src/cli/cmd/tui/context/theme/lucent-orng.json +234 -0
- package/src/cli/cmd/tui/context/theme/material.json +235 -0
- package/src/cli/cmd/tui/context/theme/matrix.json +77 -0
- package/src/cli/cmd/tui/context/theme/mercury.json +252 -0
- package/src/cli/cmd/tui/context/theme/monokai.json +221 -0
- package/src/cli/cmd/tui/context/theme/nightowl.json +221 -0
- package/src/cli/cmd/tui/context/theme/nord.json +223 -0
- package/src/cli/cmd/tui/context/theme/one-dark.json +84 -0
- package/src/cli/cmd/tui/context/theme/orng.json +249 -0
- package/src/cli/cmd/tui/context/theme/osaka-jade.json +93 -0
- package/src/cli/cmd/tui/context/theme/palenight.json +222 -0
- package/src/cli/cmd/tui/context/theme/rosepine.json +234 -0
- package/src/cli/cmd/tui/context/theme/solarized.json +223 -0
- package/src/cli/cmd/tui/context/theme/synthwave84.json +226 -0
- package/src/cli/cmd/tui/context/theme/tokyonight.json +243 -0
- package/src/cli/cmd/tui/context/theme/tulingcode.json +245 -0
- package/src/cli/cmd/tui/context/theme/vercel.json +245 -0
- package/src/cli/cmd/tui/context/theme/vesper.json +218 -0
- package/src/cli/cmd/tui/context/theme/zenburn.json +223 -0
- package/src/cli/cmd/tui/context/theme.tsx +1298 -0
- package/src/cli/cmd/tui/context/thinking.ts +48 -0
- package/src/cli/cmd/tui/context/tui-config.tsx +9 -0
- package/src/cli/cmd/tui/event.ts +56 -0
- package/src/cli/cmd/tui/feature-plugins/home/footer.tsx +93 -0
- package/src/cli/cmd/tui/feature-plugins/home/tips-view.tsx +193 -0
- package/src/cli/cmd/tui/feature-plugins/home/tips.tsx +54 -0
- package/src/cli/cmd/tui/feature-plugins/sidebar/context.tsx +114 -0
- package/src/cli/cmd/tui/feature-plugins/sidebar/cwd.tsx +45 -0
- package/src/cli/cmd/tui/feature-plugins/sidebar/files.tsx +62 -0
- package/src/cli/cmd/tui/feature-plugins/sidebar/footer.tsx +93 -0
- package/src/cli/cmd/tui/feature-plugins/sidebar/goal.tsx +84 -0
- package/src/cli/cmd/tui/feature-plugins/sidebar/instructions.tsx +54 -0
- package/src/cli/cmd/tui/feature-plugins/sidebar/lsp.tsx +66 -0
- package/src/cli/cmd/tui/feature-plugins/sidebar/mcp.tsx +98 -0
- package/src/cli/cmd/tui/feature-plugins/sidebar/task.tsx +95 -0
- package/src/cli/cmd/tui/feature-plugins/sidebar/todo.tsx +51 -0
- package/src/cli/cmd/tui/feature-plugins/sidebar/tps.ts +31 -0
- package/src/cli/cmd/tui/feature-plugins/system/plugins.tsx +274 -0
- package/src/cli/cmd/tui/i18n/en.ts +397 -0
- package/src/cli/cmd/tui/i18n/es.ts +433 -0
- package/src/cli/cmd/tui/i18n/fr.ts +440 -0
- package/src/cli/cmd/tui/i18n/ja.ts +392 -0
- package/src/cli/cmd/tui/i18n/locales.ts +82 -0
- package/src/cli/cmd/tui/i18n/ru.ts +452 -0
- package/src/cli/cmd/tui/i18n/zh.ts +390 -0
- package/src/cli/cmd/tui/i18n/zht.ts +360 -0
- package/src/cli/cmd/tui/layer.ts +6 -0
- package/src/cli/cmd/tui/plugin/api.tsx +402 -0
- package/src/cli/cmd/tui/plugin/index.ts +3 -0
- package/src/cli/cmd/tui/plugin/internal.ts +35 -0
- package/src/cli/cmd/tui/plugin/runtime.ts +1030 -0
- package/src/cli/cmd/tui/plugin/slots.tsx +60 -0
- package/src/cli/cmd/tui/routes/home.tsx +165 -0
- package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +76 -0
- package/src/cli/cmd/tui/routes/session/dialog-message.tsx +116 -0
- package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +47 -0
- package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +47 -0
- package/src/cli/cmd/tui/routes/session/footer.tsx +91 -0
- package/src/cli/cmd/tui/routes/session/index.tsx +2532 -0
- package/src/cli/cmd/tui/routes/session/permission.tsx +691 -0
- package/src/cli/cmd/tui/routes/session/question.tsx +488 -0
- package/src/cli/cmd/tui/routes/session/sidebar.tsx +97 -0
- package/src/cli/cmd/tui/routes/session/subagent-footer.tsx +142 -0
- package/src/cli/cmd/tui/thread.ts +246 -0
- package/src/cli/cmd/tui/ui/dialog-alert.tsx +61 -0
- package/src/cli/cmd/tui/ui/dialog-confirm.tsx +95 -0
- package/src/cli/cmd/tui/ui/dialog-export-options.tsx +223 -0
- package/src/cli/cmd/tui/ui/dialog-help.tsx +42 -0
- package/src/cli/cmd/tui/ui/dialog-prompt.tsx +123 -0
- package/src/cli/cmd/tui/ui/dialog-select.tsx +452 -0
- package/src/cli/cmd/tui/ui/dialog.tsx +207 -0
- package/src/cli/cmd/tui/ui/link.tsx +28 -0
- package/src/cli/cmd/tui/ui/spinner.ts +378 -0
- package/src/cli/cmd/tui/ui/toast.tsx +102 -0
- package/src/cli/cmd/tui/util/clipboard.ts +203 -0
- package/src/cli/cmd/tui/util/editor.ts +35 -0
- package/src/cli/cmd/tui/util/image-protocol.ts +35 -0
- package/src/cli/cmd/tui/util/index.ts +6 -0
- package/src/cli/cmd/tui/util/model.ts +23 -0
- package/src/cli/cmd/tui/util/provider-origin.ts +7 -0
- package/src/cli/cmd/tui/util/revert-diff.ts +18 -0
- package/src/cli/cmd/tui/util/scroll.ts +23 -0
- package/src/cli/cmd/tui/util/selection.ts +23 -0
- package/src/cli/cmd/tui/util/signal.ts +41 -0
- package/src/cli/cmd/tui/util/sound.ts +154 -0
- package/src/cli/cmd/tui/util/system-locale.ts +209 -0
- package/src/cli/cmd/tui/util/terminal.ts +110 -0
- package/src/cli/cmd/tui/util/transcript.ts +112 -0
- package/src/cli/cmd/tui/util/vad.ts +229 -0
- package/src/cli/cmd/tui/util/voice.ts +360 -0
- package/src/cli/cmd/tui/win32.ts +130 -0
- package/src/cli/cmd/tui/worker.ts +104 -0
- package/src/cli/cmd/uninstall.ts +351 -0
- package/src/cli/cmd/upgrade.ts +79 -0
- package/src/cli/cmd/web.ts +81 -0
- package/src/cli/effect/prompt.ts +25 -0
- package/src/cli/error.ts +82 -0
- package/src/cli/heap.ts +59 -0
- package/src/cli/i18n.ts +15 -0
- package/src/cli/logo.ts +53 -0
- package/src/cli/network.ts +62 -0
- package/src/cli/ui.ts +133 -0
- package/src/cli/upgrade.ts +41 -0
- package/src/command/index.ts +276 -0
- package/src/command/template/initialize.txt +66 -0
- package/src/command/template/review.txt +101 -0
- package/src/config/agent.ts +197 -0
- package/src/config/command.ts +69 -0
- package/src/config/config.ts +1024 -0
- package/src/config/console-state.ts +16 -0
- package/src/config/entry-name.ts +16 -0
- package/src/config/error.ts +21 -0
- package/src/config/formatter.ts +17 -0
- package/src/config/history.ts +21 -0
- package/src/config/index.ts +16 -0
- package/src/config/keybinds.ts +127 -0
- package/src/config/layout.ts +10 -0
- package/src/config/lsp.ts +45 -0
- package/src/config/managed.ts +70 -0
- package/src/config/markdown.ts +97 -0
- package/src/config/mcp.ts +172 -0
- package/src/config/model-id.ts +14 -0
- package/src/config/parse.ts +44 -0
- package/src/config/paths.ts +73 -0
- package/src/config/permission.ts +76 -0
- package/src/config/plugin.ts +88 -0
- package/src/config/provider.ts +118 -0
- package/src/config/server.ts +20 -0
- package/src/config/skills.ts +16 -0
- package/src/config/variable.ts +90 -0
- package/src/control-plane/adaptors/index.ts +52 -0
- package/src/control-plane/adaptors/worktree.ts +47 -0
- package/src/control-plane/dev/debug-workspace-plugin.ts +73 -0
- package/src/control-plane/schema.ts +19 -0
- package/src/control-plane/sse.ts +66 -0
- package/src/control-plane/types.ts +34 -0
- package/src/control-plane/util.ts +37 -0
- package/src/control-plane/workspace-context.ts +26 -0
- package/src/control-plane/workspace.sql.ts +17 -0
- package/src/control-plane/workspace.ts +615 -0
- package/src/effect/app-runtime.ts +146 -0
- package/src/effect/bootstrap-runtime.ts +33 -0
- package/src/effect/bridge.ts +48 -0
- package/src/effect/cross-spawn-spawner.ts +514 -0
- package/src/effect/index.ts +5 -0
- package/src/effect/instance-ref.ts +11 -0
- package/src/effect/instance-registry.ts +12 -0
- package/src/effect/instance-state.ts +81 -0
- package/src/effect/logger.ts +73 -0
- package/src/effect/memo-map.ts +3 -0
- package/src/effect/observability.ts +107 -0
- package/src/effect/run-service.ts +52 -0
- package/src/effect/runner.ts +210 -0
- package/src/effect/runtime.ts +19 -0
- package/src/env/index.ts +37 -0
- package/src/file/ignore.ts +81 -0
- package/src/file/index.ts +664 -0
- package/src/file/protected.ts +59 -0
- package/src/file/ripgrep.ts +485 -0
- package/src/file/watcher.ts +163 -0
- package/src/flag/flag.ts +164 -0
- package/src/format/formatter.ts +403 -0
- package/src/format/index.ts +203 -0
- package/src/git/index.ts +260 -0
- package/src/global/index.ts +54 -0
- package/src/history/backfill.ts +162 -0
- package/src/history/extract.ts +67 -0
- package/src/history/fts-query.ts +15 -0
- package/src/history/fts.sql.ts +20 -0
- package/src/history/index.ts +10 -0
- package/src/history/resolve.ts +65 -0
- package/src/history/service.ts +258 -0
- package/src/history/writer.ts +112 -0
- package/src/id/id.ts +87 -0
- package/src/ide/index.ts +73 -0
- package/src/inbox/inbox-ref.ts +38 -0
- package/src/inbox/inbox.sql.ts +26 -0
- package/src/inbox/inbox.ts +223 -0
- package/src/inbox/index.ts +3 -0
- package/src/inbox/render.ts +40 -0
- package/src/index.ts +260 -0
- package/src/installation/index.ts +351 -0
- package/src/installation/version.ts +8 -0
- package/src/lsp/client.ts +249 -0
- package/src/lsp/diagnostic.ts +29 -0
- package/src/lsp/index.ts +3 -0
- package/src/lsp/language.ts +120 -0
- package/src/lsp/launch.ts +21 -0
- package/src/lsp/lsp.ts +519 -0
- package/src/lsp/server.ts +1956 -0
- package/src/mcp/auth.ts +144 -0
- package/src/mcp/index.ts +944 -0
- package/src/mcp/oauth-callback.ts +232 -0
- package/src/mcp/oauth-provider.ts +214 -0
- package/src/memory/fts-query.ts +37 -0
- package/src/memory/fts.sql.ts +19 -0
- package/src/memory/index.ts +1 -0
- package/src/memory/paths.ts +116 -0
- package/src/memory/reconcile.ts +144 -0
- package/src/memory/service.ts +144 -0
- package/src/metrics/client.ts +40 -0
- package/src/metrics/event.ts +43 -0
- package/src/metrics/index.ts +5 -0
- package/src/metrics/installation.ts +18 -0
- package/src/metrics/subscriber.ts +58 -0
- package/src/metrics/util.ts +9 -0
- package/src/node.ts +6 -0
- package/src/npm/config.ts +0 -0
- package/src/npm/index.ts +293 -0
- package/src/npmcli-config.d.ts +43 -0
- package/src/patch/index.ts +680 -0
- package/src/permission/arity.ts +163 -0
- package/src/permission/evaluate.ts +15 -0
- package/src/permission/index.ts +379 -0
- package/src/permission/schema.ts +17 -0
- package/src/plugin/checkpoint-splitover.ts +60 -0
- package/src/plugin/cloud-ai.ts +329 -0
- package/src/plugin/cloudflare.ts +76 -0
- package/src/plugin/codex.ts +607 -0
- package/src/plugin/github-copilot/copilot.ts +368 -0
- package/src/plugin/github-copilot/models.ts +153 -0
- package/src/plugin/index.ts +493 -0
- package/src/plugin/install.ts +439 -0
- package/src/plugin/loader.ts +216 -0
- package/src/plugin/matcher.ts +33 -0
- package/src/plugin/meta.ts +188 -0
- package/src/plugin/mimo-free.ts +153 -0
- package/src/plugin/mimo.ts +124 -0
- package/src/plugin/shared.ts +323 -0
- package/src/plugin/subagent-progress-checker.ts +147 -0
- package/src/project/bootstrap.ts +59 -0
- package/src/project/index.ts +2 -0
- package/src/project/instance.ts +190 -0
- package/src/project/project-id.ts +48 -0
- package/src/project/project.sql.ts +16 -0
- package/src/project/project.ts +501 -0
- package/src/project/schema.ts +15 -0
- package/src/project/vcs.ts +227 -0
- package/src/provider/auth.ts +234 -0
- package/src/provider/error.ts +216 -0
- package/src/provider/index.ts +5 -0
- package/src/provider/models.ts +180 -0
- package/src/provider/provider.ts +1782 -0
- package/src/provider/schema.ts +36 -0
- package/src/provider/sdk/copilot/README.md +5 -0
- package/src/provider/sdk/copilot/chat/convert-to-openai-compatible-chat-messages.ts +170 -0
- package/src/provider/sdk/copilot/chat/get-response-metadata.ts +15 -0
- package/src/provider/sdk/copilot/chat/map-openai-compatible-finish-reason.ts +19 -0
- package/src/provider/sdk/copilot/chat/openai-compatible-api-types.ts +64 -0
- package/src/provider/sdk/copilot/chat/openai-compatible-chat-language-model.ts +815 -0
- package/src/provider/sdk/copilot/chat/openai-compatible-chat-options.ts +28 -0
- package/src/provider/sdk/copilot/chat/openai-compatible-metadata-extractor.ts +44 -0
- package/src/provider/sdk/copilot/chat/openai-compatible-prepare-tools.ts +83 -0
- package/src/provider/sdk/copilot/copilot-provider.ts +100 -0
- package/src/provider/sdk/copilot/index.ts +2 -0
- package/src/provider/sdk/copilot/openai-compatible-error.ts +27 -0
- package/src/provider/sdk/copilot/responses/convert-to-openai-responses-input.ts +335 -0
- package/src/provider/sdk/copilot/responses/map-openai-responses-finish-reason.ts +22 -0
- package/src/provider/sdk/copilot/responses/openai-config.ts +18 -0
- package/src/provider/sdk/copilot/responses/openai-error.ts +22 -0
- package/src/provider/sdk/copilot/responses/openai-responses-api-types.ts +214 -0
- package/src/provider/sdk/copilot/responses/openai-responses-language-model.ts +1770 -0
- package/src/provider/sdk/copilot/responses/openai-responses-prepare-tools.ts +173 -0
- package/src/provider/sdk/copilot/responses/openai-responses-settings.ts +1 -0
- package/src/provider/sdk/copilot/responses/tool/code-interpreter.ts +87 -0
- package/src/provider/sdk/copilot/responses/tool/file-search.ts +127 -0
- package/src/provider/sdk/copilot/responses/tool/image-generation.ts +114 -0
- package/src/provider/sdk/copilot/responses/tool/local-shell.ts +64 -0
- package/src/provider/sdk/copilot/responses/tool/web-search-preview.ts +103 -0
- package/src/provider/sdk/copilot/responses/tool/web-search.ts +102 -0
- package/src/provider/transform.ts +1322 -0
- package/src/pty/index.ts +364 -0
- package/src/pty/pty.bun.ts +26 -0
- package/src/pty/pty.node.ts +27 -0
- package/src/pty/pty.ts +25 -0
- package/src/pty/schema.ts +17 -0
- package/src/question/index.ts +252 -0
- package/src/question/schema.ts +17 -0
- package/src/server/adapter.bun.ts +40 -0
- package/src/server/adapter.node.ts +66 -0
- package/src/server/adapter.ts +21 -0
- package/src/server/error.ts +53 -0
- package/src/server/event.ts +7 -0
- package/src/server/fence.ts +81 -0
- package/src/server/mdns.ts +60 -0
- package/src/server/middleware.ts +92 -0
- package/src/server/projectors.ts +28 -0
- package/src/server/proxy.ts +171 -0
- package/src/server/routes/control/index.ts +160 -0
- package/src/server/routes/control/workspace.ts +203 -0
- package/src/server/routes/global.ts +287 -0
- package/src/server/routes/instance/bash-interactive.ts +82 -0
- package/src/server/routes/instance/config.ts +89 -0
- package/src/server/routes/instance/event.ts +88 -0
- package/src/server/routes/instance/experimental.ts +408 -0
- package/src/server/routes/instance/file.ts +190 -0
- package/src/server/routes/instance/httpapi/config.ts +51 -0
- package/src/server/routes/instance/httpapi/permission.ts +72 -0
- package/src/server/routes/instance/httpapi/project.ts +62 -0
- package/src/server/routes/instance/httpapi/provider.ts +142 -0
- package/src/server/routes/instance/httpapi/question.ts +121 -0
- package/src/server/routes/instance/httpapi/server.ts +136 -0
- package/src/server/routes/instance/index.ts +301 -0
- package/src/server/routes/instance/mcp.ts +260 -0
- package/src/server/routes/instance/middleware.ts +35 -0
- package/src/server/routes/instance/permission.ts +73 -0
- package/src/server/routes/instance/project.ts +122 -0
- package/src/server/routes/instance/provider.ts +158 -0
- package/src/server/routes/instance/pty.ts +247 -0
- package/src/server/routes/instance/question.ts +162 -0
- package/src/server/routes/instance/session.ts +1296 -0
- package/src/server/routes/instance/sync.ts +143 -0
- package/src/server/routes/instance/trace.ts +59 -0
- package/src/server/routes/instance/tui.ts +384 -0
- package/src/server/routes/instance/workflows.ts +72 -0
- package/src/server/routes/ui.ts +55 -0
- package/src/server/server.ts +136 -0
- package/src/server/workspace.ts +122 -0
- package/src/session/auto-dream.ts +123 -0
- package/src/session/boundary.ts +77 -0
- package/src/session/budgeted-read.ts +118 -0
- package/src/session/checkpoint-align.ts +29 -0
- package/src/session/checkpoint-context.ts +36 -0
- package/src/session/checkpoint-paths.ts +86 -0
- package/src/session/checkpoint-progress-reconcile.ts +111 -0
- package/src/session/checkpoint-retry.ts +192 -0
- package/src/session/checkpoint-templates.ts +114 -0
- package/src/session/checkpoint-validator.ts +259 -0
- package/src/session/checkpoint.ts +1478 -0
- package/src/session/classify.ts +92 -0
- package/src/session/claude-import.sql.ts +13 -0
- package/src/session/claude-import.ts +379 -0
- package/src/session/compaction.ts +543 -0
- package/src/session/goal.ts +232 -0
- package/src/session/index.ts +1 -0
- package/src/session/instruction.ts +276 -0
- package/src/session/last-message-info.ts +32 -0
- package/src/session/llm-request-prefix.ts +82 -0
- package/src/session/llm.ts +735 -0
- package/src/session/max-mode.ts +397 -0
- package/src/session/message-v2.ts +1136 -0
- package/src/session/message.ts +191 -0
- package/src/session/overflow.ts +53 -0
- package/src/session/prefix-capture-ref.ts +48 -0
- package/src/session/processor.ts +962 -0
- package/src/session/projectors.ts +137 -0
- package/src/session/prompt/anthropic.txt +154 -0
- package/src/session/prompt/beast.txt +155 -0
- package/src/session/prompt/build-switch.txt +5 -0
- package/src/session/prompt/codex.txt +79 -0
- package/src/session/prompt/compose.txt +115 -0
- package/src/session/prompt/copilot-gpt-5.txt +143 -0
- package/src/session/prompt/default.txt +151 -0
- package/src/session/prompt/gemini.txt +155 -0
- package/src/session/prompt/gpt.txt +107 -0
- package/src/session/prompt/kimi.txt +95 -0
- package/src/session/prompt/max-steps.txt +16 -0
- package/src/session/prompt/trinity.txt +97 -0
- package/src/session/prompt.ts +3355 -0
- package/src/session/prune.ts +481 -0
- package/src/session/retry.ts +166 -0
- package/src/session/revert.ts +161 -0
- package/src/session/run-state.ts +135 -0
- package/src/session/schema.ts +36 -0
- package/src/session/session.sql.ts +110 -0
- package/src/session/session.ts +908 -0
- package/src/session/status.ts +89 -0
- package/src/session/summary.ts +163 -0
- package/src/session/system.ts +86 -0
- package/src/session/todo.ts +77 -0
- package/src/share/index.ts +2 -0
- package/src/share/session.ts +57 -0
- package/src/share/share-next.ts +381 -0
- package/src/share/share.sql.ts +13 -0
- package/src/shell/shell.ts +110 -0
- package/src/skill/compose/.bundle/ask/SKILL.md +58 -0
- package/src/skill/compose/.bundle/brainstorm/SKILL.md +220 -0
- package/src/skill/compose/.bundle/brainstorm/scripts/frame-template.html +214 -0
- package/src/skill/compose/.bundle/brainstorm/scripts/helper.js +88 -0
- package/src/skill/compose/.bundle/brainstorm/scripts/server.cjs +354 -0
- package/src/skill/compose/.bundle/brainstorm/scripts/start-server.sh +148 -0
- package/src/skill/compose/.bundle/brainstorm/scripts/stop-server.sh +56 -0
- package/src/skill/compose/.bundle/brainstorm/spec-document-reviewer-prompt.md +50 -0
- package/src/skill/compose/.bundle/brainstorm/visual-companion.md +287 -0
- package/src/skill/compose/.bundle/debug/CREATION-LOG.md +119 -0
- package/src/skill/compose/.bundle/debug/SKILL.md +297 -0
- package/src/skill/compose/.bundle/debug/condition-based-waiting-example.ts +158 -0
- package/src/skill/compose/.bundle/debug/condition-based-waiting.md +115 -0
- package/src/skill/compose/.bundle/debug/defense-in-depth.md +122 -0
- package/src/skill/compose/.bundle/debug/find-polluter.sh +63 -0
- package/src/skill/compose/.bundle/debug/root-cause-tracing.md +169 -0
- package/src/skill/compose/.bundle/debug/test-academic.md +14 -0
- package/src/skill/compose/.bundle/debug/test-pressure-1.md +58 -0
- package/src/skill/compose/.bundle/debug/test-pressure-2.md +68 -0
- package/src/skill/compose/.bundle/debug/test-pressure-3.md +69 -0
- package/src/skill/compose/.bundle/execute/SKILL.md +71 -0
- package/src/skill/compose/.bundle/feedback/SKILL.md +214 -0
- package/src/skill/compose/.bundle/merge/SKILL.md +252 -0
- package/src/skill/compose/.bundle/new-skill/SKILL.md +656 -0
- package/src/skill/compose/.bundle/new-skill/anthropic-best-practices.md +1150 -0
- package/src/skill/compose/.bundle/new-skill/examples/CLAUDE_MD_TESTING.md +189 -0
- package/src/skill/compose/.bundle/new-skill/graphviz-conventions.dot +172 -0
- package/src/skill/compose/.bundle/new-skill/persuasion-principles.md +187 -0
- package/src/skill/compose/.bundle/new-skill/render-graphs.js +168 -0
- package/src/skill/compose/.bundle/new-skill/testing-skills-with-subagents.md +384 -0
- package/src/skill/compose/.bundle/parallel/SKILL.md +182 -0
- package/src/skill/compose/.bundle/plan/SKILL.md +161 -0
- package/src/skill/compose/.bundle/plan/plan-document-reviewer-prompt.md +50 -0
- package/src/skill/compose/.bundle/report/SKILL.md +180 -0
- package/src/skill/compose/.bundle/review/SKILL.md +104 -0
- package/src/skill/compose/.bundle/review/code-reviewer.md +171 -0
- package/src/skill/compose/.bundle/subagent/SKILL.md +344 -0
- package/src/skill/compose/.bundle/subagent/code-quality-reviewer-prompt.md +24 -0
- package/src/skill/compose/.bundle/subagent/implementer-prompt.md +126 -0
- package/src/skill/compose/.bundle/subagent/spec-reviewer-prompt.md +112 -0
- package/src/skill/compose/.bundle/tdd/SKILL.md +372 -0
- package/src/skill/compose/.bundle/tdd/testing-anti-patterns.md +299 -0
- package/src/skill/compose/.bundle/verify/SKILL.md +140 -0
- package/src/skill/compose/.bundle/worktree/SKILL.md +234 -0
- package/src/skill/compose/LICENSE-karpathy +28 -0
- package/src/skill/compose/LICENSE-superpowers +26 -0
- package/src/skill/compose/bundle.macro.ts +30 -0
- package/src/skill/compose/extract.ts +85 -0
- package/src/skill/discovery.ts +116 -0
- package/src/skill/index.ts +311 -0
- package/src/snapshot/index.ts +777 -0
- package/src/sql.d.ts +4 -0
- package/src/storage/db.bun.ts +8 -0
- package/src/storage/db.node.ts +8 -0
- package/src/storage/db.ts +172 -0
- package/src/storage/index.ts +26 -0
- package/src/storage/json-migration.ts +426 -0
- package/src/storage/schema.sql.ts +10 -0
- package/src/storage/schema.ts +7 -0
- package/src/storage/storage.ts +331 -0
- package/src/sync/README.md +179 -0
- package/src/sync/event.sql.ts +16 -0
- package/src/sync/index.ts +278 -0
- package/src/sync/schema.ts +14 -0
- package/src/task/events.ts +28 -0
- package/src/task/gate-state.ts +54 -0
- package/src/task/gate.ts +116 -0
- package/src/task/index.ts +1 -0
- package/src/task/registry.ts +387 -0
- package/src/task/schema.ts +43 -0
- package/src/task/task.sql.ts +50 -0
- package/src/team/events.ts +22 -0
- package/src/team/index.ts +113 -0
- package/src/team/schema.ts +31 -0
- package/src/temporary.ts +33 -0
- package/src/tool/actor.shell.txt +72 -0
- package/src/tool/actor.ts +803 -0
- package/src/tool/actor.txt +103 -0
- package/src/tool/apply_patch.ts +308 -0
- package/src/tool/apply_patch.txt +33 -0
- package/src/tool/bash-interactive.ts +183 -0
- package/src/tool/bash.ts +696 -0
- package/src/tool/bash.txt +123 -0
- package/src/tool/change-directory.ts +91 -0
- package/src/tool/codesearch.ts +63 -0
- package/src/tool/codesearch.txt +12 -0
- package/src/tool/edit.ts +685 -0
- package/src/tool/edit.txt +10 -0
- package/src/tool/external-directory.ts +132 -0
- package/src/tool/glob.ts +100 -0
- package/src/tool/glob.txt +6 -0
- package/src/tool/grep.ts +145 -0
- package/src/tool/grep.txt +8 -0
- package/src/tool/history.ts +146 -0
- package/src/tool/history.txt +17 -0
- package/src/tool/index.ts +4 -0
- package/src/tool/invalid.ts +20 -0
- package/src/tool/invocation-style.ts +17 -0
- package/src/tool/lsp.ts +91 -0
- package/src/tool/lsp.txt +19 -0
- package/src/tool/mcp-exa.ts +78 -0
- package/src/tool/memory-path-guard.ts +162 -0
- package/src/tool/memory.ts +81 -0
- package/src/tool/memory.txt +69 -0
- package/src/tool/multiedit.ts +61 -0
- package/src/tool/multiedit.txt +41 -0
- package/src/tool/plan-enter.txt +14 -0
- package/src/tool/plan-exit.txt +13 -0
- package/src/tool/plan.ts +90 -0
- package/src/tool/question.ts +67 -0
- package/src/tool/question.txt +10 -0
- package/src/tool/read.ts +327 -0
- package/src/tool/read.txt +14 -0
- package/src/tool/registry.ts +413 -0
- package/src/tool/schema.ts +17 -0
- package/src/tool/session-cwd.ts +35 -0
- package/src/tool/shell-tokenize.ts +346 -0
- package/src/tool/shell-wrap.ts +190 -0
- package/src/tool/skill.ts +76 -0
- package/src/tool/skill.txt +5 -0
- package/src/tool/task.shell.txt +57 -0
- package/src/tool/task.ts +456 -0
- package/src/tool/task.txt +56 -0
- package/src/tool/tool.ts +153 -0
- package/src/tool/truncate.ts +201 -0
- package/src/tool/truncation-dir.ts +4 -0
- package/src/tool/webfetch.ts +199 -0
- package/src/tool/webfetch.txt +13 -0
- package/src/tool/websearch/index.ts +104 -0
- package/src/tool/websearch/mimo.ts +118 -0
- package/src/tool/websearch/websearch.txt +14 -0
- package/src/tool/workflow.ts +164 -0
- package/src/tool/workflow.txt +25 -0
- package/src/tool/write.ts +88 -0
- package/src/tool/write.txt +9 -0
- package/src/util/abort.ts +35 -0
- package/src/util/archive.ts +15 -0
- package/src/util/color.ts +17 -0
- package/src/util/data-url.ts +9 -0
- package/src/util/defer.ts +10 -0
- package/src/util/effect-http-client.ts +11 -0
- package/src/util/effect-zod.ts +367 -0
- package/src/util/error.ts +78 -0
- package/src/util/filesystem.ts +243 -0
- package/src/util/fn.ts +21 -0
- package/src/util/format.ts +20 -0
- package/src/util/iife.ts +3 -0
- package/src/util/index.ts +12 -0
- package/src/util/keybind.ts +101 -0
- package/src/util/lazy.ts +18 -0
- package/src/util/local-context.ts +23 -0
- package/src/util/locale.ts +79 -0
- package/src/util/lock.ts +96 -0
- package/src/util/log.ts +197 -0
- package/src/util/media.ts +26 -0
- package/src/util/mimo-process.ts +24 -0
- package/src/util/network.ts +9 -0
- package/src/util/process.ts +174 -0
- package/src/util/queue.ts +32 -0
- package/src/util/record.ts +3 -0
- package/src/util/rpc.ts +64 -0
- package/src/util/schema.ts +53 -0
- package/src/util/scrap.ts +10 -0
- package/src/util/signal.ts +12 -0
- package/src/util/timeout.ts +14 -0
- package/src/util/token.ts +5 -0
- package/src/util/update-schema.ts +13 -0
- package/src/util/which.ts +14 -0
- package/src/util/wildcard.ts +57 -0
- package/src/workflow/builtin/deep-research.js +391 -0
- package/src/workflow/builtin.ts +54 -0
- package/src/workflow/events.ts +72 -0
- package/src/workflow/meta.ts +335 -0
- package/src/workflow/persistence.ts +312 -0
- package/src/workflow/resolve.ts +45 -0
- package/src/workflow/runtime-ref.ts +18 -0
- package/src/workflow/runtime.ts +1234 -0
- package/src/workflow/sandbox.ts +280 -0
- package/src/workflow/workflow.sql.ts +31 -0
- package/src/workflow/workspace.ts +69 -0
- package/src/worktree/index.ts +614 -0
- package/sst-env.d.ts +10 -0
- package/test/AGENTS.md +133 -0
- package/test/account/repo.test.ts +352 -0
- package/test/account/service.test.ts +456 -0
- package/test/acp/agent-interface.test.ts +51 -0
- package/test/acp/event-subscription.test.ts +725 -0
- package/test/actor/cancel-cascade.test.ts +432 -0
- package/test/actor/no-completion-listener.test.ts +41 -0
- package/test/actor/poststop-progress-write-permission.repro.test.ts +414 -0
- package/test/actor/registry-render.test.ts +113 -0
- package/test/actor/registry-status.test.ts +111 -0
- package/test/actor/registry.test.ts +619 -0
- package/test/actor/return-header.test.ts +40 -0
- package/test/actor/spawn-lifecycle.test.ts +346 -0
- package/test/actor/spawn-no-deadlock.test.ts +340 -0
- package/test/actor/spawn-notification.test.ts +393 -0
- package/test/actor/spawn-task-autostart.test.ts +530 -0
- package/test/actor/spawn.test.ts +1072 -0
- package/test/actor/status-event-payload.test.ts +132 -0
- package/test/actor/terminology.test.ts +39 -0
- package/test/actor/turn.test.ts +125 -0
- package/test/actor/waiter.test.ts +246 -0
- package/test/agent/agent.test.ts +874 -0
- package/test/agent/allowlist.test.ts +45 -0
- package/test/auth/auth.test.ts +86 -0
- package/test/bus/bus-effect.test.ts +162 -0
- package/test/bus/bus-integration.test.ts +87 -0
- package/test/bus/bus.test.ts +219 -0
- package/test/cli/account.test.ts +26 -0
- package/test/cli/cmd/tui/prompt-part.test.ts +47 -0
- package/test/cli/error.test.ts +18 -0
- package/test/cli/github-action.test.ts +198 -0
- package/test/cli/github-remote.test.ts +80 -0
- package/test/cli/import.test.ts +54 -0
- package/test/cli/plugin-auth-picker.test.ts +120 -0
- package/test/cli/run-completion.test.ts +131 -0
- package/test/cli/tui/keybind-plugin.test.ts +90 -0
- package/test/cli/tui/plugin-add.test.ts +111 -0
- package/test/cli/tui/plugin-install.test.ts +87 -0
- package/test/cli/tui/plugin-lifecycle.test.ts +224 -0
- package/test/cli/tui/plugin-loader-entrypoint.test.ts +484 -0
- package/test/cli/tui/plugin-loader-pure.test.ts +71 -0
- package/test/cli/tui/plugin-loader.test.ts +816 -0
- package/test/cli/tui/plugin-toggle.test.ts +157 -0
- package/test/cli/tui/revert-diff.test.ts +35 -0
- package/test/cli/tui/route-agent-id.test.ts +26 -0
- package/test/cli/tui/sidebar-tps.test.ts +63 -0
- package/test/cli/tui/slot-replace.test.tsx +47 -0
- package/test/cli/tui/sync-bucket.test.ts +29 -0
- package/test/cli/tui/theme-store.test.ts +51 -0
- package/test/cli/tui/thread.test.ts +121 -0
- package/test/cli/tui/transcript.test.ts +426 -0
- package/test/cli/tui/use-event.test.tsx +175 -0
- package/test/cli/tui/voice.test.ts +269 -0
- package/test/command/deep-research-command.test.ts +16 -0
- package/test/config/agent-color.test.ts +77 -0
- package/test/config/checkpoint-fork.test.ts +21 -0
- package/test/config/config.test.ts +2577 -0
- package/test/config/fixtures/empty-frontmatter.md +4 -0
- package/test/config/fixtures/frontmatter.md +28 -0
- package/test/config/fixtures/markdown-header.md +11 -0
- package/test/config/fixtures/no-frontmatter.md +1 -0
- package/test/config/fixtures/weird-model-id.md +13 -0
- package/test/config/lsp.test.ts +87 -0
- package/test/config/markdown.test.ts +228 -0
- package/test/config/plugin.test.ts +0 -0
- package/test/config/tui.test.ts +627 -0
- package/test/control-plane/adaptors.test.ts +71 -0
- package/test/control-plane/sse.test.ts +56 -0
- package/test/effect/app-runtime-logger.test.ts +92 -0
- package/test/effect/cross-spawn-spawner.test.ts +411 -0
- package/test/effect/instance-state.test.ts +482 -0
- package/test/effect/observability.test.ts +46 -0
- package/test/effect/run-service.test.ts +46 -0
- package/test/effect/runner-warn-log.test.ts +111 -0
- package/test/effect/runner.test.ts +494 -0
- package/test/fake/provider.ts +90 -0
- package/test/file/fsmonitor.test.ts +68 -0
- package/test/file/ignore.test.ts +10 -0
- package/test/file/index.test.ts +956 -0
- package/test/file/path-traversal.test.ts +204 -0
- package/test/file/ripgrep.test.ts +214 -0
- package/test/file/watcher.test.ts +249 -0
- package/test/filesystem/filesystem.test.ts +319 -0
- package/test/fixture/db.ts +11 -0
- package/test/fixture/fixture.test.ts +58 -0
- package/test/fixture/fixture.ts +190 -0
- package/test/fixture/flock-worker.ts +72 -0
- package/test/fixture/lsp/fake-lsp-server.js +75 -0
- package/test/fixture/plug-worker.ts +93 -0
- package/test/fixture/plugin-meta-worker.ts +19 -0
- package/test/fixture/skills/agents-sdk/SKILL.md +152 -0
- package/test/fixture/skills/agents-sdk/references/callable.md +92 -0
- package/test/fixture/skills/cloudflare/SKILL.md +211 -0
- package/test/fixture/skills/index.json +6 -0
- package/test/fixture/tui-plugin.ts +328 -0
- package/test/fixture/tui-runtime.ts +31 -0
- package/test/format/format.test.ts +244 -0
- package/test/git/git.test.ts +128 -0
- package/test/global/fixture/global-paths-worker.ts +17 -0
- package/test/global/mimocode-home.test.ts +143 -0
- package/test/history/backfill.test.ts +149 -0
- package/test/history/extract.test.ts +106 -0
- package/test/history/fts-query.test.ts +30 -0
- package/test/history/resolve.test.ts +130 -0
- package/test/history/service.test.ts +210 -0
- package/test/history/writer.test.ts +163 -0
- package/test/ide/ide.test.ts +82 -0
- package/test/inbox/drain-in-loop.test.ts +230 -0
- package/test/inbox/fork-agent-compat.test.ts +387 -0
- package/test/inbox/gc-on-init.test.ts +167 -0
- package/test/inbox/send-no-block.test.ts +120 -0
- package/test/inbox/sender-cancel-independence.test.ts +160 -0
- package/test/inbox/wake-matrix.test.ts +141 -0
- package/test/installation/installation.test.ts +226 -0
- package/test/keybind.test.ts +421 -0
- package/test/lib/effect.ts +53 -0
- package/test/lib/filesystem.ts +10 -0
- package/test/lib/llm-server.ts +770 -0
- package/test/lib/scripted-llm-server.ts +245 -0
- package/test/lsp/client.test.ts +98 -0
- package/test/lsp/index.test.ts +109 -0
- package/test/lsp/launch.test.ts +22 -0
- package/test/lsp/lifecycle.test.ts +184 -0
- package/test/mcp/headers.test.ts +178 -0
- package/test/mcp/lifecycle.test.ts +824 -0
- package/test/mcp/oauth-auto-connect.test.ts +281 -0
- package/test/mcp/oauth-browser.test.ts +268 -0
- package/test/mcp/oauth-callback.test.ts +34 -0
- package/test/memory/abort-leak-webfetch.ts +49 -0
- package/test/memory/abort-leak.test.ts +127 -0
- package/test/memory/cc-frontmatter.test.ts +85 -0
- package/test/memory/cc-paths.test.ts +60 -0
- package/test/memory/cc-reconcile.test.ts +239 -0
- package/test/memory/cc-search.test.ts +151 -0
- package/test/memory/fts-query.test.ts +48 -0
- package/test/memory/fts-rowid-stability.test.ts +271 -0
- package/test/memory/paths.test.ts +210 -0
- package/test/memory/reconcile.test.ts +115 -0
- package/test/memory/service.test.ts +169 -0
- package/test/npm.test.ts +18 -0
- package/test/patch/patch.test.ts +348 -0
- package/test/permission/abort.test.ts +116 -0
- package/test/permission/arity.test.ts +33 -0
- package/test/permission/disabled.test.ts +51 -0
- package/test/permission/next.test.ts +1080 -0
- package/test/permission/non-interactive.test.ts +55 -0
- package/test/permission-task.test.ts +326 -0
- package/test/plugin/actor-hooks.test.ts +1471 -0
- package/test/plugin/auth-override.test.ts +79 -0
- package/test/plugin/checkpoint-splitover.test.ts +434 -0
- package/test/plugin/cloudflare.test.ts +68 -0
- package/test/plugin/codex.test.ts +123 -0
- package/test/plugin/github-copilot-models.test.ts +163 -0
- package/test/plugin/install-concurrency.test.ts +140 -0
- package/test/plugin/install.test.ts +570 -0
- package/test/plugin/loader-shared.test.ts +1169 -0
- package/test/plugin/matcher.test.ts +97 -0
- package/test/plugin/meta.test.ts +137 -0
- package/test/plugin/mimo.test.ts +257 -0
- package/test/plugin/shared.test.ts +88 -0
- package/test/plugin/subagent-progress-checker.test.ts +227 -0
- package/test/plugin/trigger.test.ts +116 -0
- package/test/plugin/workspace-adaptor.test.ts +109 -0
- package/test/preload.ts +102 -0
- package/test/project/migrate-global.test.ts +150 -0
- package/test/project/project-id.test.ts +64 -0
- package/test/project/project.test.ts +481 -0
- package/test/project/vcs.test.ts +286 -0
- package/test/project/worktree-remove.test.ts +126 -0
- package/test/project/worktree.test.ts +214 -0
- package/test/provider/amazon-bedrock.test.ts +462 -0
- package/test/provider/copilot/convert-to-copilot-messages.test.ts +523 -0
- package/test/provider/copilot/copilot-chat-model.test.ts +592 -0
- package/test/provider/error.test.ts +160 -0
- package/test/provider/gitlab-duo.test.ts +413 -0
- package/test/provider/model-groups.test.ts +389 -0
- package/test/provider/provider-chunk-timeout.test.ts +23 -0
- package/test/provider/provider.test.ts +2648 -0
- package/test/provider/transform.test.ts +3379 -0
- package/test/pty/pty-output-isolation.test.ts +146 -0
- package/test/pty/pty-session.test.ts +102 -0
- package/test/pty/pty-shell.test.ts +69 -0
- package/test/question/question.test.ts +464 -0
- package/test/server/global-session-list.test.ts +105 -0
- package/test/server/project-init-git.test.ts +122 -0
- package/test/server/session-actions.test.ts +49 -0
- package/test/server/session-list.test.ts +110 -0
- package/test/server/session-messages.test.ts +220 -0
- package/test/server/session-prompt-busy.test.ts +146 -0
- package/test/server/session-select.test.ts +100 -0
- package/test/server/session-task-route.test.ts +165 -0
- package/test/server/summarize-route-main-slice.test.ts +99 -0
- package/test/server/trace-attributes.test.ts +76 -0
- package/test/server/workflows-route.test.ts +279 -0
- package/test/session/bootstrap-skip-system.test.ts +121 -0
- package/test/session/boundary.test.ts +33 -0
- package/test/session/budgeted-read.test.ts +74 -0
- package/test/session/checkpoint-align.test.ts +58 -0
- package/test/session/checkpoint-boundary.test.ts +186 -0
- package/test/session/checkpoint-child-session.test.ts +508 -0
- package/test/session/checkpoint-context.test.ts +141 -0
- package/test/session/checkpoint-drain.test.ts +188 -0
- package/test/session/checkpoint-extract-titles.test.ts +58 -0
- package/test/session/checkpoint-fork-mode.test.ts +576 -0
- package/test/session/checkpoint-main-slice.test.ts +259 -0
- package/test/session/checkpoint-paths.test.ts +78 -0
- package/test/session/checkpoint-permission.test.ts +136 -0
- package/test/session/checkpoint-progress-reconcile.test.ts +219 -0
- package/test/session/checkpoint-rebuild-unify.test.ts +143 -0
- package/test/session/checkpoint-rebuild-v3.test.ts +248 -0
- package/test/session/checkpoint-render-verify.test.ts +512 -0
- package/test/session/checkpoint-retry.test.ts +150 -0
- package/test/session/checkpoint-splitover-integration.test.ts +533 -0
- package/test/session/checkpoint-templates.test.ts +51 -0
- package/test/session/checkpoint-thresholds.test.ts +120 -0
- package/test/session/checkpoint-validator.test.ts +189 -0
- package/test/session/classify-integration.test.ts +476 -0
- package/test/session/classify.test.ts +335 -0
- package/test/session/compaction-agent-scope.test.ts +164 -0
- package/test/session/context-inheritance.test.ts +46 -0
- package/test/session/fork-prefix-invariant.test.ts +116 -0
- package/test/session/goal.test.ts +106 -0
- package/test/session/instruction.test.ts +387 -0
- package/test/session/invalid-output-continuation.test.ts +150 -0
- package/test/session/last-message-info.test.ts +47 -0
- package/test/session/length-tool-safety.test.ts +121 -0
- package/test/session/llm-request-prefix.test.ts +197 -0
- package/test/session/llm-retry.test.ts +59 -0
- package/test/session/llm-system-prompt.test.ts +479 -0
- package/test/session/llm.test.ts +1272 -0
- package/test/session/main-lifecycle.test.ts +51 -0
- package/test/session/main-runloop-history-invariant.test.ts +182 -0
- package/test/session/max-mode-econnreset.test.ts +229 -0
- package/test/session/max-mode.test.ts +54 -0
- package/test/session/message-v2-filter.test.ts +197 -0
- package/test/session/message-v2.test.ts +1119 -0
- package/test/session/messages-default-main.test.ts +105 -0
- package/test/session/messages-pagination.test.ts +888 -0
- package/test/session/overflow.test.ts +576 -0
- package/test/session/processor-effect.test.ts +853 -0
- package/test/session/prompt-effect.test.ts +1574 -0
- package/test/session/prompt-rebuild-loop.test.ts +108 -0
- package/test/session/prompt-rebuild-reset.test.ts +67 -0
- package/test/session/prompt-sweep.test.ts +145 -0
- package/test/session/prompt-task-gate.test.ts +127 -0
- package/test/session/prompt.test.ts +703 -0
- package/test/session/prune-main-slice.test.ts +272 -0
- package/test/session/prune-skip-system.test.ts +346 -0
- package/test/session/prune.test.ts +419 -0
- package/test/session/rebuild-microcompact.test.ts +318 -0
- package/test/session/recall-reminder.test.ts +37 -0
- package/test/session/retry.test.ts +410 -0
- package/test/session/revert-compact.test.ts +639 -0
- package/test/session/run-state-tuple-key.test.ts +152 -0
- package/test/session/session-create-registers-main.test.ts +70 -0
- package/test/session/session.test.ts +181 -0
- package/test/session/snapshot-tool-race.test.ts +301 -0
- package/test/session/structured-output-integration.test.ts +264 -0
- package/test/session/structured-output-retry.test.ts +127 -0
- package/test/session/structured-output.test.ts +397 -0
- package/test/session/summary-main-slice.test.ts +170 -0
- package/test/session/system.test.ts +72 -0
- package/test/share/share-next.test.ts +332 -0
- package/test/shell/shell.test.ts +73 -0
- package/test/skill/compose-review.test.ts +141 -0
- package/test/skill/discovery.test.ts +116 -0
- package/test/skill/skill.test.ts +465 -0
- package/test/snapshot/snapshot.test.ts +1531 -0
- package/test/storage/db.test.ts +16 -0
- package/test/storage/json-migration.test.ts +831 -0
- package/test/storage/storage.test.ts +293 -0
- package/test/sync/index.test.ts +237 -0
- package/test/task/gate-state.test.ts +66 -0
- package/test/task/gate.test.ts +167 -0
- package/test/task/registry.test.ts +152 -0
- package/test/task/state-machine.test.ts +292 -0
- package/test/team/migrate-to-inbox.test.ts +124 -0
- package/test/team/team.test.ts +75 -0
- package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
- package/test/tool/actor-cancel.test.ts +206 -0
- package/test/tool/actor-recover.test.ts +50 -0
- package/test/tool/actor-send.test.ts +200 -0
- package/test/tool/actor-status.test.ts +296 -0
- package/test/tool/actor-wait.test.ts +193 -0
- package/test/tool/actor.shell.test.ts +250 -0
- package/test/tool/actor.test.ts +748 -0
- package/test/tool/apply_patch.test.ts +626 -0
- package/test/tool/bash.test.ts +1195 -0
- package/test/tool/describe-workflow.test.ts +12 -0
- package/test/tool/edit.test.ts +691 -0
- package/test/tool/external-directory.test.ts +207 -0
- package/test/tool/fixtures/large-image.png +0 -0
- package/test/tool/fixtures/models-api.json +65179 -0
- package/test/tool/glob.test.ts +81 -0
- package/test/tool/grep.test.ts +114 -0
- package/test/tool/history.test.ts +144 -0
- package/test/tool/invocation-style.test.ts +30 -0
- package/test/tool/memory-edit-ask-skip.test.ts +62 -0
- package/test/tool/memory-path-guard.test.ts +594 -0
- package/test/tool/memory.test.ts +71 -0
- package/test/tool/question.test.ts +167 -0
- package/test/tool/read.test.ts +483 -0
- package/test/tool/registry-invocation-style.test.ts +121 -0
- package/test/tool/registry.test.ts +164 -0
- package/test/tool/shell-tokenize.test.ts +273 -0
- package/test/tool/shell-wrap-missing-script.test.ts +128 -0
- package/test/tool/shell-wrap.test.ts +257 -0
- package/test/tool/skill.test.ts +99 -0
- package/test/tool/task-recover.test.ts +36 -0
- package/test/tool/task.shell.test.ts +234 -0
- package/test/tool/task.test.ts +296 -0
- package/test/tool/tool-def-shell-shape.test.ts +23 -0
- package/test/tool/tool-define.test.ts +59 -0
- package/test/tool/truncation.test.ts +253 -0
- package/test/tool/webfetch.test.ts +103 -0
- package/test/tool/whitelist.test.ts +373 -0
- package/test/tool/write.test.ts +244 -0
- package/test/util/data-url.test.ts +14 -0
- package/test/util/effect-zod.test.ts +869 -0
- package/test/util/error.test.ts +38 -0
- package/test/util/filesystem.test.ts +656 -0
- package/test/util/format.test.ts +59 -0
- package/test/util/glob.test.ts +164 -0
- package/test/util/iife.test.ts +36 -0
- package/test/util/lazy.test.ts +50 -0
- package/test/util/lock.test.ts +72 -0
- package/test/util/log.test.ts +44 -0
- package/test/util/module.test.ts +59 -0
- package/test/util/process.test.ts +128 -0
- package/test/util/timeout.test.ts +21 -0
- package/test/util/which.test.ts +100 -0
- package/test/util/wildcard.test.ts +90 -0
- package/test/workflow/builtin.test.ts +22 -0
- package/test/workflow/deep-research-cluster.test.ts +47 -0
- package/test/workflow/lib.ts +243 -0
- package/test/workflow/meta.test.ts +142 -0
- package/test/workflow/model-routing.test.ts +68 -0
- package/test/workflow/persistence.test.ts +229 -0
- package/test/workflow/resolve.test.ts +37 -0
- package/test/workflow/runtime-nested.test.ts +419 -0
- package/test/workflow/runtime-worktree.test.ts +261 -0
- package/test/workflow/runtime.test.ts +1078 -0
- package/test/workflow/sandbox.test.ts +259 -0
- package/test/workflow/tool.test.ts +473 -0
- package/test/workflow/verify-wow.test.ts +144 -0
- package/test/workflow/workspace.test.ts +88 -0
- package/test/workspace/workspace-restore.test.ts +281 -0
- package/test/worktree/index.test.ts +30 -0
- package/tsconfig.json +24 -0
|
@@ -0,0 +1,1078 @@
|
|
|
1
|
+
import { describe, expect, afterEach } from "bun:test"
|
|
2
|
+
import { Effect } from "effect"
|
|
3
|
+
import { Session } from "../../src/session"
|
|
4
|
+
import { Instance } from "../../src/project/instance"
|
|
5
|
+
import { provideTmpdirServer } from "../fixture/fixture"
|
|
6
|
+
import { testEffect } from "../lib/effect"
|
|
7
|
+
import { reply } from "../lib/llm-server"
|
|
8
|
+
import { WorkflowRuntime } from "../../src/workflow/runtime"
|
|
9
|
+
import { WorkflowAgentFailed } from "../../src/workflow/events"
|
|
10
|
+
import { WorkflowPersistence } from "../../src/workflow/persistence"
|
|
11
|
+
import { ActorRegistry } from "../../src/actor/registry"
|
|
12
|
+
import { Bus } from "../../src/bus"
|
|
13
|
+
import { makeLayer, ref, providerCfg } from "./lib"
|
|
14
|
+
|
|
15
|
+
afterEach(async () => {
|
|
16
|
+
await Instance.disposeAll()
|
|
17
|
+
})
|
|
18
|
+
|
|
19
|
+
const it = testEffect(makeLayer())
|
|
20
|
+
|
|
21
|
+
describe("WorkflowRuntime agent() fan-out", () => {
|
|
22
|
+
it.live("runs a script that fans out 3 agents and returns their results", () =>
|
|
23
|
+
provideTmpdirServer(
|
|
24
|
+
Effect.fnUntraced(function* ({ llm }) {
|
|
25
|
+
const runtime = yield* WorkflowRuntime.Service
|
|
26
|
+
const session = yield* Session.Service
|
|
27
|
+
const parent = yield* session.create({
|
|
28
|
+
title: "wf fanout",
|
|
29
|
+
permission: [{ permission: "*", pattern: "*", action: "allow" }],
|
|
30
|
+
})
|
|
31
|
+
yield* llm.text("done")
|
|
32
|
+
yield* llm.text("done")
|
|
33
|
+
yield* llm.text("done")
|
|
34
|
+
const script = [
|
|
35
|
+
`export const meta = { name: "t", description: "d" }`,
|
|
36
|
+
`const r = await parallel([() => agent("a"), () => agent("b"), () => agent("c")])`,
|
|
37
|
+
`return r`,
|
|
38
|
+
].join("\n")
|
|
39
|
+
const { runID } = yield* runtime.start({ script, sessionID: parent.id, parentActorID: "main", model: ref })
|
|
40
|
+
const outcome = yield* runtime.wait({ runID })
|
|
41
|
+
expect(outcome.status).toBe("completed")
|
|
42
|
+
expect((outcome as { result: string[] }).result.filter((x) => x === "done").length).toBe(3)
|
|
43
|
+
}),
|
|
44
|
+
{ git: true, config: providerCfg },
|
|
45
|
+
),
|
|
46
|
+
)
|
|
47
|
+
})
|
|
48
|
+
|
|
49
|
+
describe("WorkflowRuntime concurrency + resilience", () => {
|
|
50
|
+
it.live("8 agents under cap=2 all complete", () =>
|
|
51
|
+
provideTmpdirServer(
|
|
52
|
+
Effect.fnUntraced(function* ({ llm }) {
|
|
53
|
+
const runtime = yield* WorkflowRuntime.Service
|
|
54
|
+
const session = yield* Session.Service
|
|
55
|
+
const parent = yield* session.create({
|
|
56
|
+
title: "wf cap",
|
|
57
|
+
permission: [{ permission: "*", pattern: "*", action: "allow" }],
|
|
58
|
+
})
|
|
59
|
+
for (let i = 0; i < 8; i++) yield* llm.text("done")
|
|
60
|
+
const script = [
|
|
61
|
+
`export const meta = { name: "t", description: "d" }`,
|
|
62
|
+
`const ts = []`,
|
|
63
|
+
`for (let i = 0; i < 8; i++) ts.push(() => agent("x" + i))`,
|
|
64
|
+
`return (await parallel(ts)).length`,
|
|
65
|
+
].join("\n")
|
|
66
|
+
const { runID } = yield* runtime.start({
|
|
67
|
+
script,
|
|
68
|
+
sessionID: parent.id,
|
|
69
|
+
parentActorID: "main",
|
|
70
|
+
model: ref,
|
|
71
|
+
maxConcurrentAgents: 2,
|
|
72
|
+
})
|
|
73
|
+
const outcome = yield* runtime.wait({ runID })
|
|
74
|
+
expect(outcome.status).toBe("completed")
|
|
75
|
+
expect((outcome as { result: number }).result).toBe(8)
|
|
76
|
+
}),
|
|
77
|
+
{ git: true, config: providerCfg },
|
|
78
|
+
),
|
|
79
|
+
)
|
|
80
|
+
|
|
81
|
+
it.live("a failing child yields null; the run still completes", () =>
|
|
82
|
+
provideTmpdirServer(
|
|
83
|
+
Effect.fnUntraced(function* ({ llm }) {
|
|
84
|
+
const runtime = yield* WorkflowRuntime.Service
|
|
85
|
+
const session = yield* Session.Service
|
|
86
|
+
const parent = yield* session.create({
|
|
87
|
+
title: "wf null",
|
|
88
|
+
permission: [{ permission: "*", pattern: "*", action: "allow" }],
|
|
89
|
+
})
|
|
90
|
+
// A 400 is a non-retryable client error (see llm.retryable): the child
|
|
91
|
+
// that receives it finishes with no assistant text, so its AgentOutcome
|
|
92
|
+
// carries no finalText and agent() resolves to null. The other child gets
|
|
93
|
+
// "ok". agent() must NEVER throw the failure into the guest — both the run
|
|
94
|
+
// and the sibling agent must still complete. Assertions are
|
|
95
|
+
// order-independent, so the FIFO race over which child lands first is
|
|
96
|
+
// benign. (Each agent() is a subagent sharing the run's session, so
|
|
97
|
+
// result isolation rides on the agent-scoped lastAssistant fix — each
|
|
98
|
+
// child's outcome is extracted by its own agentID, so concurrent
|
|
99
|
+
// same-session children do not cross-contaminate — see runtime.ts.)
|
|
100
|
+
yield* llm.error(400, { error: { message: "bad request" } })
|
|
101
|
+
yield* llm.text("ok")
|
|
102
|
+
// A failed agent() resolves to a nullish value in the guest (the host
|
|
103
|
+
// returns null; the sandbox marshals host null → guest undefined). Treat
|
|
104
|
+
// both as the failure sentinel.
|
|
105
|
+
const script = [
|
|
106
|
+
`export const meta = { name: "t", description: "d" }`,
|
|
107
|
+
`const r = await parallel([() => agent("a"), () => agent("b")])`,
|
|
108
|
+
`return r.map((x) => (x === null || x === undefined) ? "null" : x)`,
|
|
109
|
+
].join("\n")
|
|
110
|
+
const { runID } = yield* runtime.start({ script, sessionID: parent.id, parentActorID: "main", model: ref })
|
|
111
|
+
const outcome = yield* runtime.wait({ runID })
|
|
112
|
+
expect(outcome.status).toBe("completed")
|
|
113
|
+
const r = (outcome as { result: string[] }).result
|
|
114
|
+
expect(r).toContain("null")
|
|
115
|
+
expect(r).toContain("ok")
|
|
116
|
+
}),
|
|
117
|
+
{ git: true, config: providerCfg },
|
|
118
|
+
),
|
|
119
|
+
)
|
|
120
|
+
})
|
|
121
|
+
|
|
122
|
+
describe("WorkflowRuntime convergence (scout drives fan-out)", () => {
|
|
123
|
+
// The scout is an ordinary agent() call with a schema → it returns structured
|
|
124
|
+
// data. The mock LLM answers its turn with a StructuredOutput tool call; the
|
|
125
|
+
// runtime spawns it under format:{type:"json_schema"} (runtime.ts:199) and
|
|
126
|
+
// agent() resolves to the validated object (runtime.ts:205).
|
|
127
|
+
const scoutSchema = {
|
|
128
|
+
type: "object",
|
|
129
|
+
additionalProperties: false,
|
|
130
|
+
properties: { todo: { type: "array", items: { type: "string" } } },
|
|
131
|
+
required: ["todo"],
|
|
132
|
+
}
|
|
133
|
+
|
|
134
|
+
// Drives the real runtime with a mock LLM: scout reports `todo`, the script
|
|
135
|
+
// fans out one worker per todo. Asserts the run result and the run's cumulative
|
|
136
|
+
// agent-spawn tally (agentCount = +1 per agent() call, never decremented —
|
|
137
|
+
// runtime.ts:301), the deterministic observable immune to LLM-queue races.
|
|
138
|
+
const runWithTodo = (todo: string[], expectedAgentCount: number) =>
|
|
139
|
+
provideTmpdirServer(
|
|
140
|
+
Effect.fnUntraced(function* ({ llm }) {
|
|
141
|
+
const runtime = yield* WorkflowRuntime.Service
|
|
142
|
+
const session = yield* Session.Service
|
|
143
|
+
const parent = yield* session.create({
|
|
144
|
+
title: "wf converge",
|
|
145
|
+
permission: [{ permission: "*", pattern: "*", action: "allow" }],
|
|
146
|
+
})
|
|
147
|
+
// Scout turn FIRST: it is awaited before the fan-out, so it consumes this
|
|
148
|
+
// StructuredOutput reply before any worker dequeues — ordering is stable.
|
|
149
|
+
yield* llm.push(reply().tool("StructuredOutput", { todo }))
|
|
150
|
+
// One plain text reply per worker (no schema → finalText, runtime.ts:205).
|
|
151
|
+
for (let i = 0; i < todo.length; i++) yield* llm.text("done")
|
|
152
|
+
const script = [
|
|
153
|
+
`export const meta = { name: "t", description: "d" }`,
|
|
154
|
+
`const s = await agent("scout", { schema: ${JSON.stringify(scoutSchema)} })`,
|
|
155
|
+
`const todo = (s && s.todo) || []`,
|
|
156
|
+
`await parallel(todo.map((u) => () => agent("work " + u)))`,
|
|
157
|
+
`return todo.length`,
|
|
158
|
+
].join("\n")
|
|
159
|
+
const { runID } = yield* runtime.start({ script, sessionID: parent.id, parentActorID: "main", model: ref })
|
|
160
|
+
const outcome = yield* runtime.wait({ runID })
|
|
161
|
+
expect(outcome.status).toBe("completed")
|
|
162
|
+
expect((outcome as { result: number }).result).toBe(todo.length)
|
|
163
|
+
// The load-bearing assertion: scout's reported todo count drives the
|
|
164
|
+
// worker fan-out → 1 scout + N workers spawned.
|
|
165
|
+
const snap = yield* runtime.status({ runID })
|
|
166
|
+
expect(snap.agentCount).toBe(expectedAgentCount)
|
|
167
|
+
}),
|
|
168
|
+
{ git: true, config: providerCfg },
|
|
169
|
+
)
|
|
170
|
+
|
|
171
|
+
it.live("4 todo → 1 scout + 4 workers spawned", () => runWithTodo(["a", "b", "c", "d"], 5))
|
|
172
|
+
|
|
173
|
+
it.live("2 todo → 1 scout + 2 workers spawned (a re-run with fewer undone units does less work)", () =>
|
|
174
|
+
runWithTodo(["a", "b"], 3),
|
|
175
|
+
)
|
|
176
|
+
})
|
|
177
|
+
|
|
178
|
+
describe("WorkflowRuntime schema contract (schema'd agent never returns prose)", () => {
|
|
179
|
+
// A schema'd agent() whose model NEVER calls StructuredOutput (answers with plain
|
|
180
|
+
// prose, exhausting the format.retryCount=2 retries) MUST resolve to `null`, NOT
|
|
181
|
+
// the prose finalText. Returning prose breaks scripts that do `r.fields.map(...)`
|
|
182
|
+
// (the prose is a truthy non-object) and our pipeline's catch then injects a bare
|
|
183
|
+
// null that bypasses the script's own `r ? … : []` guard — exactly the full-tree
|
|
184
|
+
// Phase-0 Verify crash. So: schema requested + structured-output failed ⇒ null.
|
|
185
|
+
const fieldsSchema = {
|
|
186
|
+
type: "object",
|
|
187
|
+
additionalProperties: false,
|
|
188
|
+
properties: { fields: { type: "array", items: { type: "string" } } },
|
|
189
|
+
required: ["fields"],
|
|
190
|
+
}
|
|
191
|
+
|
|
192
|
+
it.live("schema'd agent that only returns prose resolves to null, not the prose", () =>
|
|
193
|
+
provideTmpdirServer(
|
|
194
|
+
Effect.fnUntraced(function* ({ llm }) {
|
|
195
|
+
const runtime = yield* WorkflowRuntime.Service
|
|
196
|
+
const session = yield* Session.Service
|
|
197
|
+
const parent = yield* session.create({
|
|
198
|
+
title: "wf schema-contract",
|
|
199
|
+
permission: [{ permission: "*", pattern: "*", action: "allow" }],
|
|
200
|
+
})
|
|
201
|
+
// The model refuses structured output every turn: original + 2 retries
|
|
202
|
+
// (format.retryCount=2) all answer with prose, never a StructuredOutput call.
|
|
203
|
+
yield* llm.text("I cannot call StructuredOutput; here is prose instead.")
|
|
204
|
+
yield* llm.text("Still prose, no tool call.")
|
|
205
|
+
yield* llm.text("Prose again.")
|
|
206
|
+
// The script returns typeof the agent result. With the fix it must be
|
|
207
|
+
// "object" (null is typeof "object") and specifically === null — NOT "string".
|
|
208
|
+
const script = [
|
|
209
|
+
`export const meta = { name: "t", description: "d" }`,
|
|
210
|
+
`const r = await agent("classify", { schema: ${JSON.stringify(fieldsSchema)} })`,
|
|
211
|
+
`return { isFailure: r === null || r === undefined, isProse: typeof r === "string" }`,
|
|
212
|
+
].join("\n")
|
|
213
|
+
const { runID } = yield* runtime.start({ script, sessionID: parent.id, parentActorID: "main", model: ref })
|
|
214
|
+
const outcome = yield* runtime.wait({ runID })
|
|
215
|
+
expect(outcome.status).toBe("completed")
|
|
216
|
+
const result = (outcome as { result: { isFailure: boolean; isProse: boolean } }).result
|
|
217
|
+
expect(result.isProse).toBe(false) // never the prose string
|
|
218
|
+
expect(result.isFailure).toBe(true) // schema-fail ⇒ null/undefined sentinel
|
|
219
|
+
}),
|
|
220
|
+
{ git: true, config: providerCfg },
|
|
221
|
+
),
|
|
222
|
+
)
|
|
223
|
+
})
|
|
224
|
+
|
|
225
|
+
describe("WorkflowRuntime error visibility", () => {
|
|
226
|
+
// A script-logic error (a TypeError downstream of a never-throw agent()) must
|
|
227
|
+
// fail the run with a VISIBLE error — the real guest message — not the opaque
|
|
228
|
+
// Effect wrapper. Bare Effect.tryPromise wraps the sandbox rejection as an
|
|
229
|
+
// UnknownError whose .message is literally "An error occurred in
|
|
230
|
+
// Effect.tryPromise" (the real error is buried in .cause), so runtime.ts:547's
|
|
231
|
+
// `failure.message` extract emitted that useless string. The object-form
|
|
232
|
+
// tryPromise makes result.failure the raw sandbox Error, whose .message already
|
|
233
|
+
// carries the guest {name,message,stack} (vm.dump preserves it through the
|
|
234
|
+
// throw site). This is the test that would have caught the 3 opaque crashes.
|
|
235
|
+
it.live("a script-logic TypeError fails the run with a visible error (not the Effect wrapper)", () =>
|
|
236
|
+
provideTmpdirServer(
|
|
237
|
+
Effect.fnUntraced(function* ({ llm }) {
|
|
238
|
+
const runtime = yield* WorkflowRuntime.Service
|
|
239
|
+
const session = yield* Session.Service
|
|
240
|
+
const parent = yield* session.create({
|
|
241
|
+
title: "wf err-visible",
|
|
242
|
+
permission: [{ permission: "*", pattern: "*", action: "allow" }],
|
|
243
|
+
})
|
|
244
|
+
yield* llm.text("ok")
|
|
245
|
+
// agent("a") → "ok"; r = ["ok"]; r.map(x => x.nope.deeper) derefs a
|
|
246
|
+
// property of undefined downstream of the (now never-catching) parallel.
|
|
247
|
+
const script = [
|
|
248
|
+
`export const meta = { name: "t", description: "d" }`,
|
|
249
|
+
`const r = await parallel([() => agent("a")])`,
|
|
250
|
+
`return r.map((x) => x.nope.deeper)`,
|
|
251
|
+
].join("\n")
|
|
252
|
+
const { runID } = yield* runtime.start({ script, sessionID: parent.id, parentActorID: "main", model: ref })
|
|
253
|
+
const outcome = yield* runtime.wait({ runID })
|
|
254
|
+
expect(outcome.status).toBe("failed")
|
|
255
|
+
const error = (outcome as { error: string }).error
|
|
256
|
+
// NOT the opaque Effect wrapper, and carries a script-logic token.
|
|
257
|
+
expect(error).not.toBe("An error occurred in Effect.tryPromise")
|
|
258
|
+
expect(error).toMatch(/nope|deeper|undefined|not an object|TypeError/)
|
|
259
|
+
}),
|
|
260
|
+
{ git: true, config: providerCfg },
|
|
261
|
+
),
|
|
262
|
+
)
|
|
263
|
+
})
|
|
264
|
+
|
|
265
|
+
describe("WorkflowRuntime cancel cascade", () => {
|
|
266
|
+
it.live("cancel stops in-flight child agents and marks the run cancelled", () =>
|
|
267
|
+
provideTmpdirServer(
|
|
268
|
+
Effect.fnUntraced(function* ({ llm }) {
|
|
269
|
+
const runtime = yield* WorkflowRuntime.Service
|
|
270
|
+
const session = yield* Session.Service
|
|
271
|
+
const parent = yield* session.create({
|
|
272
|
+
title: "wf cancel",
|
|
273
|
+
permission: [{ permission: "*", pattern: "*", action: "allow" }],
|
|
274
|
+
})
|
|
275
|
+
yield* llm.hang // children hang so they're in-flight at cancel time
|
|
276
|
+
const script = [
|
|
277
|
+
`export const meta = { name: "t", description: "d" }`,
|
|
278
|
+
`return await parallel([() => agent("a"), () => agent("b"), () => agent("c")])`,
|
|
279
|
+
].join("\n")
|
|
280
|
+
const { runID } = yield* runtime.start({ script, sessionID: parent.id, parentActorID: "main", model: ref })
|
|
281
|
+
yield* Effect.sleep("250 millis") // let the fan-out spawn children
|
|
282
|
+
yield* runtime.cancel({ runID })
|
|
283
|
+
const s = yield* runtime.status({ runID })
|
|
284
|
+
expect(s.status).toBe("cancelled")
|
|
285
|
+
}),
|
|
286
|
+
{ git: true, config: providerCfg },
|
|
287
|
+
),
|
|
288
|
+
// Headroom over the default 5s: this cancel test can run concurrently with the
|
|
289
|
+
// heavyweight real-Instance worktree-isolation tests, where CI load occasionally
|
|
290
|
+
// pushed it past 5s. Generous margin keeps it deterministic without masking hangs.
|
|
291
|
+
15000,
|
|
292
|
+
)
|
|
293
|
+
|
|
294
|
+
// MR104 #2 — orphan-on-cancel race. The bug: spawnShared added the child's
|
|
295
|
+
// actorID to the run's reclaim set (entry.childActorIDs) only AFTER actor.spawn
|
|
296
|
+
// RESOLVED across the quickjs Promise bridge. A cancel landing during that gap
|
|
297
|
+
// reclaimed a STALE (empty) set, then interrupted the workflow fiber — but each
|
|
298
|
+
// child runs DETACHED in the actor scope (background:true + forkIn), so the
|
|
299
|
+
// interrupt never reaches it. Result: registered children that reclaim never
|
|
300
|
+
// cancels — orphans holding subscriptions/tokens/worktrees. Fix: register the id
|
|
301
|
+
// INSIDE the spawn Effect (onActorID), before the work fiber detaches.
|
|
302
|
+
//
|
|
303
|
+
// Observable: reclaim graceful-cancels every id in childActorIDs, and
|
|
304
|
+
// Actor.cancel writes lastOutcome="cancelled" on each (registry.test cancel
|
|
305
|
+
// cascade). So immediately after cancel returns, EVERY spawned child must carry
|
|
306
|
+
// lastOutcome="cancelled" — proof that reclaim saw it. Pre-fix the set is empty
|
|
307
|
+
// at reclaim time, so the children are never cancelled (lastOutcome stays unset).
|
|
308
|
+
// NOTE: we assert at the instant cancel returns, BEFORE any further sleep — a
|
|
309
|
+
// graceful-cancelled child can be re-driven by the auto-answering test LLM and
|
|
310
|
+
// bounce back to running:success later, which is a mock artifact unrelated to
|
|
311
|
+
// the orphan bug; the cancel-stamp at t0 is the stable signal.
|
|
312
|
+
it.live("cancel during an in-flight fan-out reclaims every child (no orphan)", () =>
|
|
313
|
+
provideTmpdirServer(
|
|
314
|
+
Effect.fnUntraced(function* ({ llm }) {
|
|
315
|
+
const runtime = yield* WorkflowRuntime.Service
|
|
316
|
+
const session = yield* Session.Service
|
|
317
|
+
const registry = yield* ActorRegistry.Service
|
|
318
|
+
const parent = yield* session.create({
|
|
319
|
+
title: "wf cancel no-orphan",
|
|
320
|
+
permission: [{ permission: "*", pattern: "*", action: "allow" }],
|
|
321
|
+
})
|
|
322
|
+
yield* llm.hang // every child hangs at the LLM → in-flight at cancel time
|
|
323
|
+
// A wide fan-out keeps spawns resolving across the bridge so the cancel
|
|
324
|
+
// lands while children are registered but the post-resolve add (the bug)
|
|
325
|
+
// has not run.
|
|
326
|
+
const script = [
|
|
327
|
+
`export const meta = { name: "t", description: "d" }`,
|
|
328
|
+
`const ts = []`,
|
|
329
|
+
`for (let i = 0; i < 8; i++) ts.push(() => agent("child" + i))`,
|
|
330
|
+
`return await parallel(ts)`,
|
|
331
|
+
].join("\n")
|
|
332
|
+
const { runID } = yield* runtime.start({
|
|
333
|
+
script,
|
|
334
|
+
sessionID: parent.id,
|
|
335
|
+
parentActorID: "main",
|
|
336
|
+
model: ref,
|
|
337
|
+
maxConcurrentAgents: 8,
|
|
338
|
+
})
|
|
339
|
+
// Let the fan-out register its children, then cancel mid-flight.
|
|
340
|
+
yield* Effect.sleep("150 millis")
|
|
341
|
+
yield* runtime.cancel({ runID })
|
|
342
|
+
|
|
343
|
+
const s = yield* runtime.status({ runID })
|
|
344
|
+
expect(s.status).toBe("cancelled")
|
|
345
|
+
|
|
346
|
+
// At least one child was actually spawned (else the test proves nothing).
|
|
347
|
+
const children = (yield* registry.listBySession(parent.id)).filter((a) => a.actorID !== "main")
|
|
348
|
+
expect(children.length).toBeGreaterThan(0)
|
|
349
|
+
// Every spawned child was reclaimed: cancel stamped lastOutcome="cancelled"
|
|
350
|
+
// on each. An orphan (never reclaimed) would have lastOutcome unset here.
|
|
351
|
+
expect(children.filter((a) => a.lastOutcome !== "cancelled")).toEqual([])
|
|
352
|
+
}),
|
|
353
|
+
{ git: true, config: providerCfg },
|
|
354
|
+
),
|
|
355
|
+
20000,
|
|
356
|
+
)
|
|
357
|
+
})
|
|
358
|
+
|
|
359
|
+
describe("WorkflowRuntime concurrency clamp", () => {
|
|
360
|
+
it.live("a request above 2x cores is clamped (run still completes)", () =>
|
|
361
|
+
provideTmpdirServer(
|
|
362
|
+
Effect.fnUntraced(function* ({ llm }) {
|
|
363
|
+
const runtime = yield* WorkflowRuntime.Service
|
|
364
|
+
const session = yield* Session.Service
|
|
365
|
+
const parent = yield* session.create({ title: "wf clamp", permission: [{ permission: "*", pattern: "*", action: "allow" }] })
|
|
366
|
+
yield* llm.text("done"); yield* llm.text("done")
|
|
367
|
+
const script = [
|
|
368
|
+
`export const meta = { name: "t", description: "d" }`,
|
|
369
|
+
`return (await parallel([() => agent("a"), () => agent("b")])).length`,
|
|
370
|
+
].join("\n")
|
|
371
|
+
const { runID } = yield* runtime.start({ script, sessionID: parent.id, parentActorID: "main", model: ref, maxConcurrentAgents: 100000 })
|
|
372
|
+
const o = yield* runtime.wait({ runID })
|
|
373
|
+
expect(o.status).toBe("completed")
|
|
374
|
+
expect((o as { result: number }).result).toBe(2)
|
|
375
|
+
}),
|
|
376
|
+
{ git: true, config: providerCfg },
|
|
377
|
+
),
|
|
378
|
+
)
|
|
379
|
+
})
|
|
380
|
+
|
|
381
|
+
describe("WorkflowRuntime per-agent timeout (straggler-abort)", () => {
|
|
382
|
+
// A single hung agent (e.g. a persistent mimo TTFT wall) must not stall the whole
|
|
383
|
+
// parallel/pipeline barrier indefinitely. With agentTimeoutMs set, the hung agent
|
|
384
|
+
// is gracefully cancelled and resolves to the never-throw null sentinel, so the
|
|
385
|
+
// sibling's "ok" and the run COMPLETE — bounded by the per-agent timeout, NOT the
|
|
386
|
+
// far-larger global scriptDeadline (a PASS proves the per-agent path fired).
|
|
387
|
+
it.live("a hung agent times out to null under agentTimeoutMs; the run completes", () =>
|
|
388
|
+
provideTmpdirServer(
|
|
389
|
+
Effect.fnUntraced(function* ({ llm }) {
|
|
390
|
+
const runtime = yield* WorkflowRuntime.Service
|
|
391
|
+
const session = yield* Session.Service
|
|
392
|
+
const parent = yield* session.create({
|
|
393
|
+
title: "wf agent-timeout",
|
|
394
|
+
permission: [{ permission: "*", pattern: "*", action: "allow" }],
|
|
395
|
+
})
|
|
396
|
+
// Queue ONE hang. The two agents race to dequeue it: whichever pulls it hangs
|
|
397
|
+
// forever; the other finds the queue empty and gets the server's auto-"ok".
|
|
398
|
+
// So exactly 1 hangs (→ times out → null) and 1 returns "ok", regardless of
|
|
399
|
+
// FIFO order — the assertion counts totals, so it's order-independent.
|
|
400
|
+
yield* llm.hang
|
|
401
|
+
const script = [
|
|
402
|
+
`export const meta = { name: "t", description: "d" }`,
|
|
403
|
+
`const r = await parallel([() => agent("a"), () => agent("b")])`,
|
|
404
|
+
`return r.map((x) => (x === null || x === undefined) ? "null" : "ok")`,
|
|
405
|
+
].join("\n")
|
|
406
|
+
const { runID } = yield* runtime.start({
|
|
407
|
+
script,
|
|
408
|
+
sessionID: parent.id,
|
|
409
|
+
parentActorID: "main",
|
|
410
|
+
model: ref,
|
|
411
|
+
agentTimeoutMs: 1500,
|
|
412
|
+
scriptDeadlineMs: 60000, // far above the per-agent timeout
|
|
413
|
+
})
|
|
414
|
+
const outcome = yield* runtime.wait({ runID })
|
|
415
|
+
expect(outcome.status).toBe("completed")
|
|
416
|
+
const r = (outcome as { result: string[] }).result
|
|
417
|
+
expect(r.filter((x) => x === "null").length).toBe(1)
|
|
418
|
+
expect(r.filter((x) => x === "ok").length).toBe(1)
|
|
419
|
+
}),
|
|
420
|
+
{ git: true, config: providerCfg },
|
|
421
|
+
),
|
|
422
|
+
20000, // budget >> the 1500ms per-agent timeout, well under any true hang
|
|
423
|
+
)
|
|
424
|
+
})
|
|
425
|
+
|
|
426
|
+
describe("WorkflowRuntime lifecycle cap", () => {
|
|
427
|
+
// Hitting the lifecycle agent cap is an EXPECTED steady-state for a large
|
|
428
|
+
// fan-out (lifetime-classify caps its own verify sample to stay under it), not
|
|
429
|
+
// a programming error. So an over-cap agent() must return the never-throw null
|
|
430
|
+
// sentinel (graceful degradation) — NOT throw, which post-NC-1 (combinators no
|
|
431
|
+
// longer catch) would reject the whole batch and waste every completed agent.
|
|
432
|
+
// maxLifecycleAgents parameterizes the cap so this is unit-testable at cap=2.
|
|
433
|
+
it.live("over-cap agent() returns null; the run still completes (cap=2, fan-out 3)", () =>
|
|
434
|
+
provideTmpdirServer(
|
|
435
|
+
Effect.fnUntraced(function* ({ llm }) {
|
|
436
|
+
const runtime = yield* WorkflowRuntime.Service
|
|
437
|
+
const session = yield* Session.Service
|
|
438
|
+
const parent = yield* session.create({
|
|
439
|
+
title: "wf cap-null",
|
|
440
|
+
permission: [{ permission: "*", pattern: "*", action: "allow" }],
|
|
441
|
+
})
|
|
442
|
+
// The check-then-increment is synchronous (no await between), so exactly 2
|
|
443
|
+
// of the 3 spawn (count 0→1, 1→2) and the 3rd sees count 2 ≥ cap → null.
|
|
444
|
+
// Two replies for the two that spawn; the 3rd never reaches the LLM.
|
|
445
|
+
yield* llm.text("done")
|
|
446
|
+
yield* llm.text("done")
|
|
447
|
+
const script = [
|
|
448
|
+
`export const meta = { name: "t", description: "d" }`,
|
|
449
|
+
`const r = await parallel([() => agent("a"), () => agent("b"), () => agent("c")])`,
|
|
450
|
+
`return r.map((x) => (x === null || x === undefined) ? "null" : "ok")`,
|
|
451
|
+
].join("\n")
|
|
452
|
+
const { runID } = yield* runtime.start({
|
|
453
|
+
script,
|
|
454
|
+
sessionID: parent.id,
|
|
455
|
+
parentActorID: "main",
|
|
456
|
+
model: ref,
|
|
457
|
+
maxLifecycleAgents: 2,
|
|
458
|
+
})
|
|
459
|
+
const outcome = yield* runtime.wait({ runID })
|
|
460
|
+
expect(outcome.status).toBe("completed")
|
|
461
|
+
const r = (outcome as { result: string[] }).result
|
|
462
|
+
expect(r.filter((x) => x === "ok").length).toBe(2)
|
|
463
|
+
expect(r.filter((x) => x === "null").length).toBe(1)
|
|
464
|
+
}),
|
|
465
|
+
{ git: true, config: providerCfg },
|
|
466
|
+
),
|
|
467
|
+
)
|
|
468
|
+
})
|
|
469
|
+
|
|
470
|
+
describe("WorkflowRuntime deadline", () => {
|
|
471
|
+
it.live("a script that exceeds scriptDeadlineMs is interrupted, not hung", () =>
|
|
472
|
+
provideTmpdirServer(
|
|
473
|
+
Effect.fnUntraced(function* ({ llm }) {
|
|
474
|
+
const runtime = yield* WorkflowRuntime.Service
|
|
475
|
+
const session = yield* Session.Service
|
|
476
|
+
const parent = yield* session.create({
|
|
477
|
+
title: "wf deadline",
|
|
478
|
+
permission: [{ permission: "*", pattern: "*", action: "allow" }],
|
|
479
|
+
})
|
|
480
|
+
yield* llm.hang // the single agent hangs forever
|
|
481
|
+
const script = [
|
|
482
|
+
`export const meta = { name: "t", description: "d" }`,
|
|
483
|
+
`return await agent("x")`,
|
|
484
|
+
].join("\n")
|
|
485
|
+
const { runID } = yield* runtime.start({
|
|
486
|
+
script,
|
|
487
|
+
sessionID: parent.id,
|
|
488
|
+
parentActorID: "main",
|
|
489
|
+
model: ref,
|
|
490
|
+
scriptDeadlineMs: 1500,
|
|
491
|
+
})
|
|
492
|
+
const outcome = yield* runtime.wait({ runID })
|
|
493
|
+
// The deadline fires inside the sandbox; the run resolves failed (not completed, not hung).
|
|
494
|
+
expect(outcome.status).toBe("failed")
|
|
495
|
+
}),
|
|
496
|
+
{ git: true, config: providerCfg },
|
|
497
|
+
),
|
|
498
|
+
)
|
|
499
|
+
})
|
|
500
|
+
|
|
501
|
+
describe("WorkflowRuntime counters", () => {
|
|
502
|
+
it.live("running/succeeded/failed tracked separately and persisted", () =>
|
|
503
|
+
provideTmpdirServer(
|
|
504
|
+
Effect.fnUntraced(function* ({ llm }) {
|
|
505
|
+
const runtime = yield* WorkflowRuntime.Service
|
|
506
|
+
const session = yield* Session.Service
|
|
507
|
+
const parent = yield* session.create({ title: "wf counters", permission: [{ permission: "*", pattern: "*", action: "allow" }] })
|
|
508
|
+
yield* llm.error(400, { error: { message: "bad" } })
|
|
509
|
+
yield* llm.text("ok")
|
|
510
|
+
const script = [
|
|
511
|
+
`export const meta = { name: "t", description: "d" }`,
|
|
512
|
+
`return await parallel([() => agent("a"), () => agent("b")])`,
|
|
513
|
+
].join("\n")
|
|
514
|
+
const { runID } = yield* runtime.start({ script, sessionID: parent.id, parentActorID: "main", model: ref })
|
|
515
|
+
yield* runtime.wait({ runID })
|
|
516
|
+
const row = yield* WorkflowPersistence.load(runID)
|
|
517
|
+
expect(row?.succeeded).toBe(1)
|
|
518
|
+
expect(row?.failed).toBe(1)
|
|
519
|
+
expect(row?.running).toBe(0)
|
|
520
|
+
expect(row?.status).toBe("completed")
|
|
521
|
+
}),
|
|
522
|
+
{ git: true, config: providerCfg },
|
|
523
|
+
),
|
|
524
|
+
)
|
|
525
|
+
})
|
|
526
|
+
|
|
527
|
+
describe("WorkflowRuntime list + resume", () => {
|
|
528
|
+
it.live("list returns persisted runs newest-first, filtered by session", () =>
|
|
529
|
+
provideTmpdirServer(
|
|
530
|
+
Effect.fnUntraced(function* ({ llm }) {
|
|
531
|
+
const runtime = yield* WorkflowRuntime.Service
|
|
532
|
+
const session = yield* Session.Service
|
|
533
|
+
const parent = yield* session.create({ title: "wf list", permission: [{ permission: "*", pattern: "*", action: "allow" }] })
|
|
534
|
+
yield* llm.text("done"); yield* llm.text("done")
|
|
535
|
+
const mk = () => [`export const meta = { name: "t", description: "d" }`, `return await agent("x")`].join("\n")
|
|
536
|
+
const r1 = yield* runtime.start({ script: mk(), sessionID: parent.id, parentActorID: "main", model: ref })
|
|
537
|
+
yield* runtime.wait({ runID: r1.runID })
|
|
538
|
+
const r2 = yield* runtime.start({ script: mk(), sessionID: parent.id, parentActorID: "main", model: ref })
|
|
539
|
+
yield* runtime.wait({ runID: r2.runID })
|
|
540
|
+
const rows = yield* runtime.list({ sessionID: parent.id })
|
|
541
|
+
expect(rows.length).toBe(2)
|
|
542
|
+
expect(rows[0].runID).toBe(r2.runID) // newest first
|
|
543
|
+
}),
|
|
544
|
+
{ git: true, config: providerCfg },
|
|
545
|
+
),
|
|
546
|
+
)
|
|
547
|
+
|
|
548
|
+
it.live("resume re-launches the persisted script under the same runID (cached replay, zero new spawns)", () =>
|
|
549
|
+
provideTmpdirServer(
|
|
550
|
+
Effect.fnUntraced(function* ({ llm }) {
|
|
551
|
+
const runtime = yield* WorkflowRuntime.Service
|
|
552
|
+
const session = yield* Session.Service
|
|
553
|
+
const parent = yield* session.create({ title: "wf resume", permission: [{ permission: "*", pattern: "*", action: "allow" }] })
|
|
554
|
+
yield* llm.text("done")
|
|
555
|
+
const script = [`export const meta = { name: "t", description: "d" }`, `return await agent("x")`].join("\n")
|
|
556
|
+
const { runID } = yield* runtime.start({ script, sessionID: parent.id, parentActorID: "main", model: ref })
|
|
557
|
+
const out1 = yield* runtime.wait({ runID })
|
|
558
|
+
expect(out1.status).toBe("completed")
|
|
559
|
+
// Resume: agent("x") is journaled, so it replays from cache. No reply is
|
|
560
|
+
// queued — and even if a stray spawn happened the test server would
|
|
561
|
+
// auto-"ok", so agentCount (not queue state) is the assertion that matters.
|
|
562
|
+
const resumed = yield* runtime.resume({ runID })
|
|
563
|
+
expect(resumed.runID).toBe(runID) // SAME runID
|
|
564
|
+
expect(resumed.resumed).toBe(true)
|
|
565
|
+
const out = yield* runtime.wait({ runID })
|
|
566
|
+
expect(out.status).toBe("completed")
|
|
567
|
+
expect((out as { result: unknown }).result).toBe("done") // replayed cached value
|
|
568
|
+
const st = yield* runtime.status({ runID })
|
|
569
|
+
expect(st.agentCount).toBe(0) // resume spawned nothing — pure replay
|
|
570
|
+
}),
|
|
571
|
+
{ git: true, config: providerCfg },
|
|
572
|
+
),
|
|
573
|
+
)
|
|
574
|
+
|
|
575
|
+
it.live("resume refuses a still-running run", () =>
|
|
576
|
+
provideTmpdirServer(
|
|
577
|
+
Effect.fnUntraced(function* ({ llm }) {
|
|
578
|
+
const runtime = yield* WorkflowRuntime.Service
|
|
579
|
+
const session = yield* Session.Service
|
|
580
|
+
const parent = yield* session.create({ title: "wf resume live", permission: [{ permission: "*", pattern: "*", action: "allow" }] })
|
|
581
|
+
yield* llm.hang
|
|
582
|
+
const script = [`export const meta = { name: "t", description: "d" }`, `return await agent("x")`].join("\n")
|
|
583
|
+
const { runID } = yield* runtime.start({ script, sessionID: parent.id, parentActorID: "main", model: ref })
|
|
584
|
+
yield* Effect.sleep("200 millis") // let it be running
|
|
585
|
+
const r = yield* runtime.resume({ runID })
|
|
586
|
+
expect(r.resumed).toBe(false)
|
|
587
|
+
yield* runtime.cancel({ runID })
|
|
588
|
+
}),
|
|
589
|
+
{ git: true, config: providerCfg },
|
|
590
|
+
),
|
|
591
|
+
)
|
|
592
|
+
|
|
593
|
+
// MR104 P2-1 — in-process double-resume race. The bug: resume()'s live-guard
|
|
594
|
+
// (runs.get(runID).status === "running") is a check-then-act, not atomic. Two
|
|
595
|
+
// concurrent resume(sameRunID) of a COMPLETED run BOTH read status "completed",
|
|
596
|
+
// BOTH pass the guard, BOTH launch() — and launch does runs.set(runID, entry),
|
|
597
|
+
// so the second clobbers the first (orphaned fiber, raced counter flush) and
|
|
598
|
+
// both fibers append to the SAME .jsonl journal (interleaved).
|
|
599
|
+
//
|
|
600
|
+
// The fix serializes the resume critical section (live-guard THROUGH launch's
|
|
601
|
+
// runs.set) with an in-process Lock.write keyed on the runID. The first waiter
|
|
602
|
+
// launches and flips the entry to "running" before releasing; the second waiter
|
|
603
|
+
// then sees status "running" at the guard and bails with resumed:false.
|
|
604
|
+
//
|
|
605
|
+
// DISCRIMINATING ASSERTION: with the lock, EXACTLY ONE of the two concurrent
|
|
606
|
+
// resumes returns resumed:true (the other resumed:false). Unlocked, BOTH pass
|
|
607
|
+
// the guard and BOTH re-launch → resumed:true twice. This is the direct, clean
|
|
608
|
+
// signal that the launch ran once, not twice, and it is DETERMINISTIC: Effect.all
|
|
609
|
+
// forks both resumes; both reach resume's first yield before either reaches
|
|
610
|
+
// launch, so unlocked they both observe status "completed" and both relaunch.
|
|
611
|
+
// (Verified empirically: against the unlocked code this assertion fails with
|
|
612
|
+
// Received: 2, fast and repeatably — not flaky.)
|
|
613
|
+
it.live("two concurrent resumes of the same completed run launch exactly once (no double-launch)", () =>
|
|
614
|
+
provideTmpdirServer(
|
|
615
|
+
Effect.fnUntraced(function* ({ llm }) {
|
|
616
|
+
const runtime = yield* WorkflowRuntime.Service
|
|
617
|
+
const session = yield* Session.Service
|
|
618
|
+
const parent = yield* session.create({
|
|
619
|
+
title: "wf resume race",
|
|
620
|
+
permission: [{ permission: "*", pattern: "*", action: "allow" }],
|
|
621
|
+
})
|
|
622
|
+
// First run: one real spawn for agent("x"). It is journaled, so every resume
|
|
623
|
+
// replays from cache (zero new spawns).
|
|
624
|
+
yield* llm.text("done")
|
|
625
|
+
const script = [`export const meta = { name: "t", description: "d" }`, `return await agent("x")`].join("\n")
|
|
626
|
+
const { runID } = yield* runtime.start({ script, sessionID: parent.id, parentActorID: "main", model: ref })
|
|
627
|
+
const first = yield* runtime.wait({ runID })
|
|
628
|
+
expect(first.status).toBe("completed")
|
|
629
|
+
|
|
630
|
+
// Fire two resume(sameRunID) concurrently. Without the lock both pass the
|
|
631
|
+
// live-guard (status is "completed") and both re-launch. With the lock the
|
|
632
|
+
// second serializes behind the first and sees status "running" → resumed:false.
|
|
633
|
+
const results = yield* Effect.all([runtime.resume({ runID }), runtime.resume({ runID })], {
|
|
634
|
+
concurrency: "unbounded",
|
|
635
|
+
})
|
|
636
|
+
|
|
637
|
+
// PRIMARY (and discriminating): exactly one re-launch happened — locked: 1
|
|
638
|
+
// true + 1 false; unlocked: 2 true (double-launch).
|
|
639
|
+
expect(results.filter((r) => r.resumed).length).toBe(1)
|
|
640
|
+
expect(results.filter((r) => !r.resumed).length).toBe(1)
|
|
641
|
+
|
|
642
|
+
// The single relaunched pass replays cleanly to completion.
|
|
643
|
+
const out = yield* runtime.wait({ runID })
|
|
644
|
+
expect(out.status).toBe("completed")
|
|
645
|
+
expect((out as { result: unknown }).result).toBe("done") // replayed cached value
|
|
646
|
+
|
|
647
|
+
// SANITY: the surviving entry is a pure cache replay (no spawn) — a double-
|
|
648
|
+
// launch that re-spawned would show agentCount > 0.
|
|
649
|
+
const st = yield* runtime.status({ runID })
|
|
650
|
+
expect(st.agentCount).toBe(0)
|
|
651
|
+
|
|
652
|
+
// Settle the relaunched fiber's terminal tail (bus.publish + inbox.send run
|
|
653
|
+
// AFTER Deferred.succeed, so wait() returns before the fiber is fully done)
|
|
654
|
+
// before the tmpdir fixture's Instance.disposeAll() tears the layer scope
|
|
655
|
+
// down. Without this drain, disposeAll can interrupt the fiber mid-tail and
|
|
656
|
+
// hang teardown — a PRE-EXISTING resume teardown flake (the upstream single-
|
|
657
|
+
// resume test exhibits it in isolation too), independent of the P2-1 lock.
|
|
658
|
+
yield* Effect.sleep("300 millis")
|
|
659
|
+
}),
|
|
660
|
+
{ git: true, config: providerCfg },
|
|
661
|
+
),
|
|
662
|
+
15000,
|
|
663
|
+
)
|
|
664
|
+
})
|
|
665
|
+
|
|
666
|
+
describe("WorkflowRuntime agent() label + phase opts", () => {
|
|
667
|
+
// label + phase are pure observability metadata. They land in the spawn
|
|
668
|
+
// `description` ONLY (no schema / currentPhase / counter change). The actor
|
|
669
|
+
// registry stores `description` (notNull, falls back to agentType otherwise),
|
|
670
|
+
// so the most direct observable is the child actor's registry row after the
|
|
671
|
+
// run: an agent() with {label,phase} must carry "[Implement] impl:foo".
|
|
672
|
+
it.live("label + phase thread into the spawned agent description", () =>
|
|
673
|
+
provideTmpdirServer(
|
|
674
|
+
Effect.fnUntraced(function* ({ llm }) {
|
|
675
|
+
const runtime = yield* WorkflowRuntime.Service
|
|
676
|
+
const session = yield* Session.Service
|
|
677
|
+
const registry = yield* ActorRegistry.Service
|
|
678
|
+
const parent = yield* session.create({
|
|
679
|
+
title: "wf label",
|
|
680
|
+
permission: [{ permission: "*", pattern: "*", action: "allow" }],
|
|
681
|
+
})
|
|
682
|
+
yield* llm.text("done")
|
|
683
|
+
const script = [
|
|
684
|
+
`export const meta = { name: "t", description: "d" }`,
|
|
685
|
+
`return await agent("port it", { label: "impl:foo", phase: "Implement" })`,
|
|
686
|
+
].join("\n")
|
|
687
|
+
const { runID } = yield* runtime.start({ script, sessionID: parent.id, parentActorID: "main", model: ref })
|
|
688
|
+
yield* runtime.wait({ runID })
|
|
689
|
+
// The spawned subagent's registry row carries the [phase] label description.
|
|
690
|
+
// agent() defaults agentType to "general" and shares the parent session, so
|
|
691
|
+
// the child is the lone non-"main" actor in that session.
|
|
692
|
+
const actors = yield* registry.listBySession(parent.id)
|
|
693
|
+
const child = actors.find((a) => a.actorID !== "main")
|
|
694
|
+
expect(child?.description).toBe("[Implement] impl:foo")
|
|
695
|
+
}),
|
|
696
|
+
{ git: true, config: providerCfg },
|
|
697
|
+
),
|
|
698
|
+
)
|
|
699
|
+
})
|
|
700
|
+
|
|
701
|
+
describe("WorkflowRuntime replay journal", () => {
|
|
702
|
+
it.live("re-running the SAME runID replays cached results with zero new spawns", () =>
|
|
703
|
+
provideTmpdirServer(
|
|
704
|
+
Effect.fnUntraced(function* ({ llm }) {
|
|
705
|
+
const runtime = yield* WorkflowRuntime.Service
|
|
706
|
+
const session = yield* Session.Service
|
|
707
|
+
const parent = yield* session.create({
|
|
708
|
+
title: "wf replay",
|
|
709
|
+
permission: [{ permission: "*", pattern: "*", action: "allow" }],
|
|
710
|
+
})
|
|
711
|
+
const script = [
|
|
712
|
+
`export const meta = { name: "t", description: "d" }`,
|
|
713
|
+
`const r = await parallel([() => agent("a"), () => agent("b"), () => agent("c")])`,
|
|
714
|
+
`return r`,
|
|
715
|
+
].join("\n")
|
|
716
|
+
// First run: 3 real spawns.
|
|
717
|
+
yield* llm.text("done")
|
|
718
|
+
yield* llm.text("done")
|
|
719
|
+
yield* llm.text("done")
|
|
720
|
+
const first = yield* runtime.start({ script, sessionID: parent.id, parentActorID: "main", model: ref })
|
|
721
|
+
const out1 = yield* runtime.wait({ runID: first.runID })
|
|
722
|
+
expect(out1.status).toBe("completed")
|
|
723
|
+
const st1 = yield* runtime.status({ runID: first.runID })
|
|
724
|
+
expect(st1.agentCount).toBe(3)
|
|
725
|
+
|
|
726
|
+
// Resume the SAME runID. The journal must replay all 3 — assert via
|
|
727
|
+
// agentCount (the spawn counter), NOT via queue starvation (the test
|
|
728
|
+
// server auto-"ok"s an unqueued request, so a stray spawn would silently
|
|
729
|
+
// succeed). We queue nothing to keep intent clear.
|
|
730
|
+
const r = yield* runtime.resume({ runID: first.runID })
|
|
731
|
+
expect(r.resumed).toBe(true)
|
|
732
|
+
const out2 = yield* runtime.wait({ runID: first.runID })
|
|
733
|
+
expect(out2.status).toBe("completed")
|
|
734
|
+
expect((out2 as { result: string[] }).result.filter((x) => x === "done").length).toBe(3)
|
|
735
|
+
const st2 = yield* runtime.status({ runID: first.runID })
|
|
736
|
+
expect(st2.agentCount).toBe(0) // ZERO new spawns on the resumed pass
|
|
737
|
+
}),
|
|
738
|
+
{ git: true, config: providerCfg },
|
|
739
|
+
),
|
|
740
|
+
)
|
|
741
|
+
|
|
742
|
+
it.live("resume re-spawns ONLY the uncached unit (O(remaining), not O(all))", () =>
|
|
743
|
+
provideTmpdirServer(
|
|
744
|
+
Effect.fnUntraced(function* ({ llm }) {
|
|
745
|
+
const runtime = yield* WorkflowRuntime.Service
|
|
746
|
+
const session = yield* Session.Service
|
|
747
|
+
const parent = yield* session.create({
|
|
748
|
+
title: "wf partial",
|
|
749
|
+
permission: [{ permission: "*", pattern: "*", action: "allow" }],
|
|
750
|
+
})
|
|
751
|
+
const script = [
|
|
752
|
+
`export const meta = { name: "t", description: "d" }`,
|
|
753
|
+
`const r = await parallel([() => agent("u0"), () => agent("u1"), () => agent("u2")])`,
|
|
754
|
+
`return r`,
|
|
755
|
+
].join("\n")
|
|
756
|
+
yield* llm.text("done")
|
|
757
|
+
yield* llm.text("done")
|
|
758
|
+
yield* llm.text("done")
|
|
759
|
+
const first = yield* runtime.start({ script, sessionID: parent.id, parentActorID: "main", model: ref })
|
|
760
|
+
yield* runtime.wait({ runID: first.runID })
|
|
761
|
+
|
|
762
|
+
// Simulate a partial: drop ONE agent line from the journal so one unit is
|
|
763
|
+
// "unfinished". Distinct prompts => distinct hashes => occ 0 each, so we
|
|
764
|
+
// drop by line content (the last agent line).
|
|
765
|
+
const fs = yield* Effect.promise(() => import("fs/promises"))
|
|
766
|
+
const { Global } = yield* Effect.promise(() => import("../../src/global"))
|
|
767
|
+
const p = `${Global.Path.data}/workflow/${first.runID}.jsonl`
|
|
768
|
+
const lines = (yield* Effect.promise(() => Bun.file(p).text())).split("\n").filter(Boolean)
|
|
769
|
+
const agentLines = lines.filter((l) => l.includes('"t":"agent"'))
|
|
770
|
+
const dropKey = JSON.parse(agentLines[agentLines.length - 1]).key
|
|
771
|
+
const kept = lines.filter((l) => !l.includes(`"${dropKey}"`))
|
|
772
|
+
yield* Effect.promise(() => fs.writeFile(p, kept.join("\n") + "\n"))
|
|
773
|
+
|
|
774
|
+
// Resume: queue exactly ONE response (for the one dropped unit). The other
|
|
775
|
+
// two replay from cache. agentCount===1 is the proof.
|
|
776
|
+
yield* llm.text("done")
|
|
777
|
+
yield* runtime.resume({ runID: first.runID })
|
|
778
|
+
const out = yield* runtime.wait({ runID: first.runID })
|
|
779
|
+
expect(out.status).toBe("completed")
|
|
780
|
+
expect((out as { result: string[] }).result.filter((x) => x === "done").length).toBe(3)
|
|
781
|
+
const st = yield* runtime.status({ runID: first.runID })
|
|
782
|
+
expect(st.agentCount).toBe(1) // exactly the one uncached unit re-spawned
|
|
783
|
+
}),
|
|
784
|
+
{ git: true, config: providerCfg },
|
|
785
|
+
),
|
|
786
|
+
)
|
|
787
|
+
|
|
788
|
+
// MR104 P1-2 — script-change invalidation. The journal keys results by
|
|
789
|
+
// {prompt,agentType,model,schema,phase}+occ but NOT by the script body. If the
|
|
790
|
+
// user edits the workflow script between resume cycles, replaying the OLD journal
|
|
791
|
+
// onto NEW code is silent divergence. Fix: recordStart stamps sha256(script body)
|
|
792
|
+
// on the run row; resume compares the stored sha to the current script's sha. On
|
|
793
|
+
// MISMATCH the stale journal is cleared and the run re-spawns fresh (re-stamping
|
|
794
|
+
// the new sha so a SUBSEQUENT resume of the now-current script replays correctly).
|
|
795
|
+
//
|
|
796
|
+
// DISCRIMINATOR vs the same-script resume test above: same body → sha matches →
|
|
797
|
+
// replay → agentCount 0. Changed body → sha differs → fresh → agentCount > 0, and
|
|
798
|
+
// the old journal lines are gone (cleared, not interleaved with the new pass).
|
|
799
|
+
it.live("resume with an EDITED script discards the stale journal and re-spawns fresh", () =>
|
|
800
|
+
provideTmpdirServer(
|
|
801
|
+
Effect.fnUntraced(function* ({ llm }) {
|
|
802
|
+
const runtime = yield* WorkflowRuntime.Service
|
|
803
|
+
const session = yield* Session.Service
|
|
804
|
+
const parent = yield* session.create({
|
|
805
|
+
title: "wf script change",
|
|
806
|
+
permission: [{ permission: "*", pattern: "*", action: "allow" }],
|
|
807
|
+
})
|
|
808
|
+
const scriptA = [
|
|
809
|
+
`export const meta = { name: "t", description: "d" }`,
|
|
810
|
+
`const r = await parallel([() => agent("a"), () => agent("b")])`,
|
|
811
|
+
`return r`,
|
|
812
|
+
].join("\n")
|
|
813
|
+
// First run under script A: 2 real spawns, both journaled.
|
|
814
|
+
yield* llm.text("done")
|
|
815
|
+
yield* llm.text("done")
|
|
816
|
+
const first = yield* runtime.start({ script: scriptA, sessionID: parent.id, parentActorID: "main", model: ref })
|
|
817
|
+
const out1 = yield* runtime.wait({ runID: first.runID })
|
|
818
|
+
expect(out1.status).toBe("completed")
|
|
819
|
+
const st1 = yield* runtime.status({ runID: first.runID })
|
|
820
|
+
expect(st1.agentCount).toBe(2)
|
|
821
|
+
|
|
822
|
+
// Edit the persisted script: overwrite <runID>.js with a DIFFERENT body
|
|
823
|
+
// (resume reads its script from this file). Same prompts but a changed body
|
|
824
|
+
// → a different sha → the stored journal must be discarded.
|
|
825
|
+
const scriptB = [
|
|
826
|
+
`export const meta = { name: "t", description: "d" }`,
|
|
827
|
+
`// edited between resume cycles — a different body changes the sha`,
|
|
828
|
+
`const r = await parallel([() => agent("a"), () => agent("b")])`,
|
|
829
|
+
`return r`,
|
|
830
|
+
].join("\n")
|
|
831
|
+
yield* WorkflowPersistence.writeScript(first.runID, scriptB)
|
|
832
|
+
|
|
833
|
+
// Resume: the sha mismatch must force a fresh run. Queue 2 fresh replies; if
|
|
834
|
+
// the journal were (wrongly) replayed these would go unused and agentCount
|
|
835
|
+
// would be 0. agentCount === 2 is the proof of a fresh re-spawn.
|
|
836
|
+
yield* llm.text("done")
|
|
837
|
+
yield* llm.text("done")
|
|
838
|
+
const r = yield* runtime.resume({ runID: first.runID })
|
|
839
|
+
expect(r.resumed).toBe(true)
|
|
840
|
+
const out2 = yield* runtime.wait({ runID: first.runID })
|
|
841
|
+
expect(out2.status).toBe("completed")
|
|
842
|
+
expect((out2 as { result: string[] }).result.filter((x) => x === "done").length).toBe(2)
|
|
843
|
+
const st2 = yield* runtime.status({ runID: first.runID })
|
|
844
|
+
expect(st2.agentCount).toBe(2) // fresh re-spawn, NOT a 0-spawn replay
|
|
845
|
+
|
|
846
|
+
// The new sha was re-stamped: a SECOND resume of the now-current script B
|
|
847
|
+
// replays from the freshly-written journal (zero new spawns).
|
|
848
|
+
const r2 = yield* runtime.resume({ runID: first.runID })
|
|
849
|
+
expect(r2.resumed).toBe(true)
|
|
850
|
+
const out3 = yield* runtime.wait({ runID: first.runID })
|
|
851
|
+
expect(out3.status).toBe("completed")
|
|
852
|
+
const st3 = yield* runtime.status({ runID: first.runID })
|
|
853
|
+
expect(st3.agentCount).toBe(0) // sha now matches → pure replay
|
|
854
|
+
}),
|
|
855
|
+
{ git: true, config: providerCfg },
|
|
856
|
+
),
|
|
857
|
+
20000,
|
|
858
|
+
)
|
|
859
|
+
})
|
|
860
|
+
|
|
861
|
+
// agent() collapses every failure path to bare null, but operators need to know
|
|
862
|
+
// the REASON to triage (mimo TTFT timeout vs spawn-reject vs over-cap). The
|
|
863
|
+
// WorkflowAgentFailed bus event carries the reason without changing agent()'s
|
|
864
|
+
// null contract — these tests pin both invariants: the script still sees null,
|
|
865
|
+
// AND the bus carries one event per failed agent with the right reason.
|
|
866
|
+
describe("WorkflowRuntime agent failure event (Gap 3)", () => {
|
|
867
|
+
it.live("a 400 client error → reason='no-deliverable'; success sibling → no event", () =>
|
|
868
|
+
// The actor outcome is status:"success" (agent finished its turn cleanly),
|
|
869
|
+
// but the failed-LLM call produced no assistant text → no finalText/structured
|
|
870
|
+
// to extract → deliverable is null → reason="no-deliverable". This matches the
|
|
871
|
+
// existing "a failing child yields null" test's mechanism (line 79).
|
|
872
|
+
provideTmpdirServer(
|
|
873
|
+
Effect.fnUntraced(function* ({ llm }) {
|
|
874
|
+
const runtime = yield* WorkflowRuntime.Service
|
|
875
|
+
const session = yield* Session.Service
|
|
876
|
+
const bus = yield* Bus.Service
|
|
877
|
+
const events: Array<{ reason: string; label?: string; phase?: string }> = []
|
|
878
|
+
yield* bus.subscribeCallback(WorkflowAgentFailed, (e) => {
|
|
879
|
+
events.push({ reason: e.properties.reason, label: e.properties.label, phase: e.properties.phase })
|
|
880
|
+
})
|
|
881
|
+
const parent = yield* session.create({
|
|
882
|
+
title: "wf failreason",
|
|
883
|
+
permission: [{ permission: "*", pattern: "*", action: "allow" }],
|
|
884
|
+
})
|
|
885
|
+
yield* llm.error(400, { error: { message: "bad request" } })
|
|
886
|
+
yield* llm.text("ok")
|
|
887
|
+
const script = [
|
|
888
|
+
`export const meta = { name: "t", description: "d" }`,
|
|
889
|
+
`await parallel([`,
|
|
890
|
+
` () => agent("a", { label: "fail-one", phase: "Test" }),`,
|
|
891
|
+
` () => agent("b", { label: "ok-one" })`,
|
|
892
|
+
`])`,
|
|
893
|
+
].join("\n")
|
|
894
|
+
const { runID } = yield* runtime.start({ script, sessionID: parent.id, parentActorID: "main", model: ref })
|
|
895
|
+
const outcome = yield* runtime.wait({ runID })
|
|
896
|
+
expect(outcome.status).toBe("completed")
|
|
897
|
+
// Bus is async; let the publish settle before asserting.
|
|
898
|
+
yield* Effect.sleep("100 millis")
|
|
899
|
+
expect(events.length).toBe(1)
|
|
900
|
+
expect(events[0].reason).toBe("no-deliverable")
|
|
901
|
+
expect(events[0].label).toBe("fail-one")
|
|
902
|
+
expect(events[0].phase).toBe("Test")
|
|
903
|
+
}),
|
|
904
|
+
{ git: true, config: providerCfg },
|
|
905
|
+
),
|
|
906
|
+
)
|
|
907
|
+
|
|
908
|
+
it.live("a hung agent under timeoutMs → reason='timeout'", () =>
|
|
909
|
+
provideTmpdirServer(
|
|
910
|
+
Effect.fnUntraced(function* ({ llm }) {
|
|
911
|
+
const runtime = yield* WorkflowRuntime.Service
|
|
912
|
+
const session = yield* Session.Service
|
|
913
|
+
const bus = yield* Bus.Service
|
|
914
|
+
const events: string[] = []
|
|
915
|
+
yield* bus.subscribeCallback(WorkflowAgentFailed, (e) => {
|
|
916
|
+
events.push(e.properties.reason)
|
|
917
|
+
})
|
|
918
|
+
const parent = yield* session.create({
|
|
919
|
+
title: "wf timeout",
|
|
920
|
+
permission: [{ permission: "*", pattern: "*", action: "allow" }],
|
|
921
|
+
})
|
|
922
|
+
yield* llm.hang
|
|
923
|
+
const script = [
|
|
924
|
+
`export const meta = { name: "t", description: "d" }`,
|
|
925
|
+
`return await agent("hangs", { timeoutMs: 50 })`,
|
|
926
|
+
].join("\n")
|
|
927
|
+
const { runID } = yield* runtime.start({ script, sessionID: parent.id, parentActorID: "main", model: ref })
|
|
928
|
+
const outcome = yield* runtime.wait({ runID })
|
|
929
|
+
expect(outcome.status).toBe("completed")
|
|
930
|
+
// The script returned the null deliverable directly: it must be null
|
|
931
|
+
// (sandbox marshals host null → guest undefined; equate the two).
|
|
932
|
+
const v = (outcome as { result: unknown }).result
|
|
933
|
+
expect(v === null || v === undefined).toBe(true)
|
|
934
|
+
yield* Effect.sleep("100 millis")
|
|
935
|
+
expect(events.length).toBe(1)
|
|
936
|
+
expect(events[0]).toBe("timeout")
|
|
937
|
+
}),
|
|
938
|
+
{ git: true, config: providerCfg },
|
|
939
|
+
),
|
|
940
|
+
)
|
|
941
|
+
|
|
942
|
+
it.live("over-cap rejections emit reason='over-cap' and don't increment agentCount", () =>
|
|
943
|
+
provideTmpdirServer(
|
|
944
|
+
Effect.fnUntraced(function* ({ llm }) {
|
|
945
|
+
const runtime = yield* WorkflowRuntime.Service
|
|
946
|
+
const session = yield* Session.Service
|
|
947
|
+
const bus = yield* Bus.Service
|
|
948
|
+
const events: string[] = []
|
|
949
|
+
yield* bus.subscribeCallback(WorkflowAgentFailed, (e) => {
|
|
950
|
+
events.push(e.properties.reason)
|
|
951
|
+
})
|
|
952
|
+
const parent = yield* session.create({
|
|
953
|
+
title: "wf overcap",
|
|
954
|
+
permission: [{ permission: "*", pattern: "*", action: "allow" }],
|
|
955
|
+
})
|
|
956
|
+
// 4 agents try to spawn, cap is 2 → 2 succeed, 2 hit over-cap → null.
|
|
957
|
+
for (let i = 0; i < 2; i++) yield* llm.text("done")
|
|
958
|
+
const script = [
|
|
959
|
+
`export const meta = { name: "t", description: "d" }`,
|
|
960
|
+
`const ts = []`,
|
|
961
|
+
`for (let i = 0; i < 4; i++) ts.push(() => agent("x" + i))`,
|
|
962
|
+
`return await parallel(ts)`,
|
|
963
|
+
].join("\n")
|
|
964
|
+
const { runID } = yield* runtime.start({
|
|
965
|
+
script,
|
|
966
|
+
sessionID: parent.id,
|
|
967
|
+
parentActorID: "main",
|
|
968
|
+
model: ref,
|
|
969
|
+
maxLifecycleAgents: 2,
|
|
970
|
+
})
|
|
971
|
+
const outcome = yield* runtime.wait({ runID })
|
|
972
|
+
expect(outcome.status).toBe("completed")
|
|
973
|
+
yield* Effect.sleep("100 millis")
|
|
974
|
+
// Exactly 2 over-cap events (the cap-exceeding spawn attempts); zero
|
|
975
|
+
// other-reason events since the 2 in-cap agents both succeeded.
|
|
976
|
+
const overCap = events.filter((r) => r === "over-cap")
|
|
977
|
+
expect(overCap.length).toBe(2)
|
|
978
|
+
expect(events.length).toBe(2) // no other reasons leaked in
|
|
979
|
+
}),
|
|
980
|
+
{ git: true, config: providerCfg },
|
|
981
|
+
),
|
|
982
|
+
)
|
|
983
|
+
})
|
|
984
|
+
|
|
985
|
+
// Math.random is seeded per-run from a hash of runID, so two unrelated runs of
|
|
986
|
+
// the same script get DIFFERENT sequences (sampling-style scripts get fresh
|
|
987
|
+
// coverage); a resume of the same run gets the SAME sequence (replay invariant
|
|
988
|
+
// covered by other tests). This pins the cross-run divergence.
|
|
989
|
+
describe("WorkflowRuntime PRNG seeding (cross-run divergence)", () => {
|
|
990
|
+
it.live("two unrelated runs of the same Math.random script get different sequences", () =>
|
|
991
|
+
provideTmpdirServer(
|
|
992
|
+
Effect.fnUntraced(function* () {
|
|
993
|
+
const runtime = yield* WorkflowRuntime.Service
|
|
994
|
+
const session = yield* Session.Service
|
|
995
|
+
const parent = yield* session.create({
|
|
996
|
+
title: "wf prng",
|
|
997
|
+
permission: [{ permission: "*", pattern: "*", action: "allow" }],
|
|
998
|
+
})
|
|
999
|
+
const script = [
|
|
1000
|
+
`export const meta = { name: "t", description: "d" }`,
|
|
1001
|
+
`return [Math.random(), Math.random(), Math.random()]`,
|
|
1002
|
+
].join("\n")
|
|
1003
|
+
const a = yield* runtime.start({ script, sessionID: parent.id, parentActorID: "main", model: ref })
|
|
1004
|
+
const ao = yield* runtime.wait({ runID: a.runID })
|
|
1005
|
+
const b = yield* runtime.start({ script, sessionID: parent.id, parentActorID: "main", model: ref })
|
|
1006
|
+
const bo = yield* runtime.wait({ runID: b.runID })
|
|
1007
|
+
expect(ao.status).toBe("completed")
|
|
1008
|
+
expect(bo.status).toBe("completed")
|
|
1009
|
+
const av = (ao as { result: number[] }).result
|
|
1010
|
+
const bv = (bo as { result: number[] }).result
|
|
1011
|
+
expect(av).not.toEqual(bv) // different runIDs → different seeds → different sequences
|
|
1012
|
+
expect(av[0]).toBeGreaterThanOrEqual(0)
|
|
1013
|
+
expect(av[0]).toBeLessThan(1)
|
|
1014
|
+
}),
|
|
1015
|
+
{ git: true, config: providerCfg },
|
|
1016
|
+
),
|
|
1017
|
+
)
|
|
1018
|
+
})
|
|
1019
|
+
|
|
1020
|
+
// agent_timeout_ms is persisted on the workflow_run row at start time so a
|
|
1021
|
+
// resume that doesn't supply its own override (e.g. the TUI's /workflows resume
|
|
1022
|
+
// command, which currently passes only runID) inherits the original timeout
|
|
1023
|
+
// instead of silently dropping to unbounded — which used to let a wedged mimo
|
|
1024
|
+
// TTFT stall the resumed run forever. This test pins the "implicit-resume
|
|
1025
|
+
// inherits the persisted timeout" contract end-to-end via the persistence layer.
|
|
1026
|
+
describe("WorkflowRuntime persists agentTimeoutMs across resume (TUI-style)", () => {
|
|
1027
|
+
it.live("a row started with agentTimeoutMs is readable via persistence.load", () =>
|
|
1028
|
+
provideTmpdirServer(
|
|
1029
|
+
Effect.fnUntraced(function* () {
|
|
1030
|
+
const runtime = yield* WorkflowRuntime.Service
|
|
1031
|
+
const session = yield* Session.Service
|
|
1032
|
+
const parent = yield* session.create({
|
|
1033
|
+
title: "wf timeout-persist",
|
|
1034
|
+
permission: [{ permission: "*", pattern: "*", action: "allow" }],
|
|
1035
|
+
})
|
|
1036
|
+
const script = `export const meta = { name: "t", description: "d" }`
|
|
1037
|
+
const r = yield* runtime.start({
|
|
1038
|
+
script,
|
|
1039
|
+
sessionID: parent.id,
|
|
1040
|
+
parentActorID: "main",
|
|
1041
|
+
model: ref,
|
|
1042
|
+
agentTimeoutMs: 12345,
|
|
1043
|
+
})
|
|
1044
|
+
yield* runtime.wait({ runID: r.runID })
|
|
1045
|
+
// Row is persisted with the timeout; a no-arg resume() would read this back
|
|
1046
|
+
// (resume uses input.agentTimeoutMs ?? row.agentTimeoutMs as the effective
|
|
1047
|
+
// value passed to launch()). Fresh-start with a value smaller than the
|
|
1048
|
+
// smallest sleep would be a clean integration test, but it requires real
|
|
1049
|
+
// mock timing; the persistence path is the load-bearing check so we pin
|
|
1050
|
+
// that directly to keep the test cheap and deterministic.
|
|
1051
|
+
const row = yield* WorkflowPersistence.load(r.runID)
|
|
1052
|
+
expect(row).toBeDefined()
|
|
1053
|
+
expect(row!.agentTimeoutMs).toBe(12345)
|
|
1054
|
+
}),
|
|
1055
|
+
{ git: true, config: providerCfg },
|
|
1056
|
+
),
|
|
1057
|
+
)
|
|
1058
|
+
|
|
1059
|
+
it.live("a row started with NO timeout reads back undefined (no silent default)", () =>
|
|
1060
|
+
provideTmpdirServer(
|
|
1061
|
+
Effect.fnUntraced(function* () {
|
|
1062
|
+
const runtime = yield* WorkflowRuntime.Service
|
|
1063
|
+
const session = yield* Session.Service
|
|
1064
|
+
const parent = yield* session.create({
|
|
1065
|
+
title: "wf timeout-undef",
|
|
1066
|
+
permission: [{ permission: "*", pattern: "*", action: "allow" }],
|
|
1067
|
+
})
|
|
1068
|
+
const script = `export const meta = { name: "t", description: "d" }`
|
|
1069
|
+
const r = yield* runtime.start({ script, sessionID: parent.id, parentActorID: "main", model: ref })
|
|
1070
|
+
yield* runtime.wait({ runID: r.runID })
|
|
1071
|
+
const row = yield* WorkflowPersistence.load(r.runID)
|
|
1072
|
+
expect(row).toBeDefined()
|
|
1073
|
+
expect(row!.agentTimeoutMs).toBeUndefined()
|
|
1074
|
+
}),
|
|
1075
|
+
{ git: true, config: providerCfg },
|
|
1076
|
+
),
|
|
1077
|
+
)
|
|
1078
|
+
})
|