@desplega.ai/agent-swarm 1.20.0 → 1.51.2
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/README.md +271 -169
- package/openapi.json +5015 -0
- package/package.json +40 -7
- package/plugin/commands/close-issue.md +7 -3
- package/plugin/commands/create-pr.md +18 -12
- package/plugin/commands/implement-issue.md +7 -3
- package/plugin/commands/respond-github.md +8 -4
- package/plugin/commands/review-pr.md +44 -10
- package/plugin/commands/start-leader.md +1 -3
- package/plugin/commands/start-worker.md +1 -3
- package/plugin/commands/work-on-task.md +22 -3
- package/plugin/pi-skills/close-issue/SKILL.md +90 -0
- package/plugin/pi-skills/create-pr/SKILL.md +99 -0
- package/plugin/pi-skills/implement-issue/SKILL.md +135 -0
- package/plugin/pi-skills/investigate-sentry-issue/SKILL.md +138 -0
- package/plugin/pi-skills/respond-github/SKILL.md +98 -0
- package/plugin/pi-skills/review-offered-task/SKILL.md +45 -0
- package/plugin/pi-skills/review-pr/SKILL.md +261 -0
- package/plugin/pi-skills/start-leader/SKILL.md +121 -0
- package/plugin/pi-skills/start-worker/SKILL.md +60 -0
- package/plugin/pi-skills/swarm-chat/SKILL.md +82 -0
- package/plugin/pi-skills/todos/SKILL.md +66 -0
- package/plugin/pi-skills/work-on-task/SKILL.md +65 -0
- package/plugin/skills/artifacts/examples/approval-flow.ts +34 -0
- package/plugin/skills/artifacts/examples/hono-dashboard.ts +31 -0
- package/plugin/skills/artifacts/examples/multi-artifact.ts +20 -0
- package/plugin/skills/artifacts/examples/static-report.sh +17 -0
- package/plugin/skills/artifacts/skill.md +71 -0
- package/src/agentmail/app.ts +65 -0
- package/src/agentmail/handlers.ts +262 -0
- package/src/agentmail/index.ts +9 -0
- package/src/agentmail/templates.ts +111 -0
- package/src/agentmail/types.ts +51 -0
- package/src/artifact-sdk/browser-sdk.ts +30 -0
- package/src/artifact-sdk/index.ts +2 -0
- package/src/artifact-sdk/localtunnel.d.ts +20 -0
- package/src/artifact-sdk/port.ts +12 -0
- package/src/artifact-sdk/server.ts +156 -0
- package/src/artifact-sdk/tunnel.ts +19 -0
- package/src/be/chunking.ts +193 -0
- package/src/be/db-queries/oauth.ts +90 -0
- package/src/be/db-queries/tracker.ts +182 -0
- package/src/be/db.ts +3327 -784
- package/src/be/embedding.ts +80 -0
- package/src/be/migrations/001_initial.sql +409 -0
- package/src/be/migrations/002_one_time_schedules.sql +59 -0
- package/src/be/migrations/003_workflows.sql +51 -0
- package/src/be/migrations/004_workflow_source.sql +81 -0
- package/src/be/migrations/005_epic_next_steps.sql +2 -0
- package/src/be/migrations/006_vcs_provider.sql +94 -0
- package/src/be/migrations/007_task_dir.sql +2 -0
- package/src/be/migrations/008_workflow_redesign.sql +85 -0
- package/src/be/migrations/009_tracker_integration.sql +144 -0
- package/src/be/migrations/010_step_diagnostics.sql +1 -0
- package/src/be/migrations/011_step_next_port.sql +1 -0
- package/src/be/migrations/012_trigger_schema.sql +1 -0
- package/src/be/migrations/013_task_output_schema.sql +2 -0
- package/src/be/migrations/014_prompt_templates.sql +33 -0
- package/src/be/migrations/015_workflow_workspace.sql +3 -0
- package/src/be/migrations/016_active_session_runner_session.sql +4 -0
- package/src/be/migrations/017_channel_activity_cursors.sql +6 -0
- package/src/be/migrations/018_fix_seed_double_version.sql +30 -0
- package/src/be/migrations/runner.ts +188 -0
- package/src/be/seed.ts +62 -0
- package/src/cli.tsx +231 -299
- package/src/commands/artifact.ts +241 -0
- package/src/commands/onboard/compose-generator.ts +169 -0
- package/src/commands/onboard/env-generator.ts +79 -0
- package/src/commands/onboard/manifest.ts +37 -0
- package/src/commands/onboard/presets.ts +85 -0
- package/src/commands/onboard/service-names.ts +47 -0
- package/src/commands/onboard/steps/core-credentials.tsx +111 -0
- package/src/commands/onboard/steps/custom-templates.tsx +168 -0
- package/src/commands/onboard/steps/generate.tsx +154 -0
- package/src/commands/onboard/steps/harness-credentials.tsx +195 -0
- package/src/commands/onboard/steps/harness.tsx +21 -0
- package/src/commands/onboard/steps/health-check.tsx +171 -0
- package/src/commands/onboard/steps/integration-github.tsx +105 -0
- package/src/commands/onboard/steps/integration-gitlab.tsx +79 -0
- package/src/commands/onboard/steps/integration-menu.tsx +58 -0
- package/src/commands/onboard/steps/integration-sentry.tsx +79 -0
- package/src/commands/onboard/steps/integration-slack.tsx +165 -0
- package/src/commands/onboard/steps/post-connect.tsx +145 -0
- package/src/commands/onboard/steps/post-dashboard.tsx +34 -0
- package/src/commands/onboard/steps/post-task.tsx +103 -0
- package/src/commands/onboard/steps/prereq-check.tsx +178 -0
- package/src/commands/onboard/steps/review.tsx +82 -0
- package/src/commands/onboard/steps/start.tsx +97 -0
- package/src/commands/onboard/templates.ts +34 -0
- package/src/commands/onboard/types.ts +259 -0
- package/src/commands/onboard.tsx +425 -0
- package/src/commands/runner.ts +1540 -630
- package/src/commands/setup.tsx +23 -38
- package/src/commands/shared/client-config.ts +41 -0
- package/src/commands/templates.ts +172 -0
- package/src/github/app.ts +8 -0
- package/src/github/handlers.ts +384 -151
- package/src/github/index.ts +1 -0
- package/src/github/mentions-aliases.test.ts +73 -0
- package/src/github/mentions.test.ts +3 -3
- package/src/github/mentions.ts +32 -6
- package/src/github/templates.ts +398 -0
- package/src/github/types.ts +1 -0
- package/src/gitlab/auth.ts +63 -0
- package/src/gitlab/handlers.ts +368 -0
- package/src/gitlab/index.ts +19 -0
- package/src/gitlab/reactions.ts +104 -0
- package/src/gitlab/templates.ts +140 -0
- package/src/gitlab/types.ts +130 -0
- package/src/heartbeat/heartbeat.ts +434 -0
- package/src/heartbeat/index.ts +1 -0
- package/src/heartbeat/templates.ts +30 -0
- package/src/hooks/hook.ts +555 -4
- package/src/hooks/tool-loop-detection.test.ts +158 -0
- package/src/hooks/tool-loop-detection.ts +167 -0
- package/src/http/active-sessions.ts +199 -0
- package/src/http/agents.ts +328 -0
- package/src/http/config.ts +191 -0
- package/src/http/core.ts +309 -0
- package/src/http/db-query.ts +91 -0
- package/src/http/ecosystem.ts +63 -0
- package/src/http/epics.ts +460 -0
- package/src/http/index.ts +216 -0
- package/src/http/mcp.ts +77 -0
- package/src/http/memory.ts +168 -0
- package/src/http/openapi.ts +109 -0
- package/src/http/poll.ts +299 -0
- package/src/http/prompt-templates.ts +412 -0
- package/src/http/repos.ts +195 -0
- package/src/http/route-def.ts +123 -0
- package/src/http/schedules.ts +426 -0
- package/src/http/session-data.ts +241 -0
- package/src/http/stats.ts +174 -0
- package/src/http/tasks.ts +468 -0
- package/src/http/trackers/index.ts +10 -0
- package/src/http/trackers/linear.ts +187 -0
- package/src/http/types.ts +12 -0
- package/src/http/utils.ts +87 -0
- package/src/http/webhooks.ts +432 -0
- package/src/http/workflows.ts +530 -0
- package/src/http.ts +1 -1890
- package/src/linear/README.md +65 -0
- package/src/linear/app.ts +48 -0
- package/src/linear/client.ts +18 -0
- package/src/linear/index.ts +1 -0
- package/src/linear/oauth.ts +35 -0
- package/src/linear/outbound.ts +212 -0
- package/src/linear/sync.ts +567 -0
- package/src/linear/templates.ts +47 -0
- package/src/linear/types.ts +7 -0
- package/src/linear/webhook.ts +104 -0
- package/src/oauth/README.md +66 -0
- package/src/oauth/index.ts +6 -0
- package/src/oauth/wrapper.ts +204 -0
- package/src/prompts/base-prompt.ts +150 -265
- package/src/prompts/defaults.ts +196 -0
- package/src/prompts/registry.ts +57 -0
- package/src/prompts/resolver.ts +296 -0
- package/src/prompts/session-templates.ts +604 -0
- package/src/providers/claude-adapter.ts +442 -0
- package/src/providers/index.ts +24 -0
- package/src/providers/pi-mono-adapter.ts +442 -0
- package/src/providers/pi-mono-extension.ts +624 -0
- package/src/providers/pi-mono-mcp-client.ts +124 -0
- package/src/providers/types.ts +75 -0
- package/src/scheduler/scheduler.test.ts +2 -0
- package/src/scheduler/scheduler.ts +231 -40
- package/src/server.ts +97 -6
- package/src/slack/HEURISTICS.md +105 -0
- package/src/slack/actions.ts +133 -0
- package/src/slack/app.ts +7 -0
- package/src/slack/assistant.ts +118 -0
- package/src/slack/blocks.ts +233 -0
- package/src/slack/channel-activity.ts +177 -0
- package/src/slack/commands.ts +31 -17
- package/src/slack/files.ts +1 -1
- package/src/slack/handlers.test.ts +114 -1
- package/src/slack/handlers.ts +230 -55
- package/src/slack/responses.ts +120 -67
- package/src/slack/router.ts +17 -99
- package/src/slack/templates.ts +55 -0
- package/src/slack/thread-buffer.ts +213 -0
- package/src/slack/watcher.ts +119 -4
- package/src/tests/agent-activity.test.ts +247 -0
- package/src/tests/agentmail-filters.test.ts +97 -0
- package/src/tests/artifact-sdk.test.ts +800 -0
- package/src/tests/base-prompt.test.ts +264 -0
- package/src/tests/build-pi-skills.test.ts +127 -0
- package/src/tests/channel-activity.test.ts +363 -0
- package/src/tests/claude-adapter.test.ts +126 -0
- package/src/tests/context-versioning.test.ts +425 -0
- package/src/tests/db-queries-oauth.test.ts +197 -0
- package/src/tests/db-queries-tracker.test.ts +230 -0
- package/src/tests/epics.test.ts +3 -3
- package/src/tests/error-tracker.test.ts +368 -0
- package/src/tests/fetch-resolved-env.test.ts +167 -0
- package/src/tests/generate-default-claude-md.test.ts +9 -1
- package/src/tests/generate-identity-templates.test.ts +124 -0
- package/src/tests/gitlab-auth.test.ts +109 -0
- package/src/tests/gitlab-handlers.test.ts +691 -0
- package/src/tests/gitlab-vcs-db.test.ts +177 -0
- package/src/tests/heartbeat.test.ts +364 -0
- package/src/tests/http-api-integration.test.ts +1698 -0
- package/src/tests/linear-outbound-sync.test.ts +200 -0
- package/src/tests/linear-webhook.test.ts +406 -0
- package/src/tests/match-route.test.ts +187 -0
- package/src/tests/memory.test.ts +737 -0
- package/src/tests/migration-runner-regressions.test.ts +86 -0
- package/src/tests/model-control.test.ts +338 -0
- package/src/tests/oauth-wrapper.test.ts +147 -0
- package/src/tests/onboard-compose.test.ts +138 -0
- package/src/tests/onboard-env.test.ts +174 -0
- package/src/tests/onboard-manifest.test.ts +137 -0
- package/src/tests/pi-mono-adapter.test.ts +234 -0
- package/src/tests/pool-session-logs.test.ts +199 -0
- package/src/tests/progress-dedup.test.ts +98 -0
- package/src/tests/prompt-template-github.test.ts +682 -0
- package/src/tests/prompt-template-remaining.test.ts +504 -0
- package/src/tests/prompt-template-resolver.test.ts +621 -0
- package/src/tests/prompt-template-session.test.ts +363 -0
- package/src/tests/prompt-templates-db.test.ts +616 -0
- package/src/tests/provider-adapter.test.ts +122 -0
- package/src/tests/provider-command-format.test.ts +98 -0
- package/src/tests/reload-config.test.ts +170 -0
- package/src/tests/runner-polling-api.test.ts +25 -20
- package/src/tests/scheduled-tasks.test.ts +104 -0
- package/src/tests/scheduler-backoff.test.ts +166 -0
- package/src/tests/self-improvement.test.ts +541 -0
- package/src/tests/session-attach.test.ts +536 -0
- package/src/tests/session-costs.test.ts +267 -1
- package/src/tests/slack-actions.test.ts +133 -0
- package/src/tests/slack-assistant.test.ts +136 -0
- package/src/tests/slack-blocks.test.ts +246 -0
- package/src/tests/slack-metadata-inheritance.test.ts +243 -0
- package/src/tests/slack-queue-offline.test.ts +174 -0
- package/src/tests/slack-router.test.ts +181 -0
- package/src/tests/slack-thread-buffer.test.ts +305 -0
- package/src/tests/slack-thread-followups.test.ts +298 -0
- package/src/tests/slack-watcher.test.ts +101 -0
- package/src/tests/structured-output.test.ts +307 -0
- package/src/tests/swarm-repos.test.ts +198 -0
- package/src/tests/task-cancellation.test.ts +6 -4
- package/src/tests/task-working-dir.test.ts +176 -0
- package/src/tests/template-fetch.test.ts +490 -0
- package/src/tests/tool-annotations.test.ts +371 -0
- package/src/tests/tracker-tools.test.ts +184 -0
- package/src/tests/update-profile-agentid.test.ts +248 -0
- package/src/tests/update-profile-api.test.ts +143 -3
- package/src/tests/update-profile-auth.test.ts +195 -0
- package/src/tests/validation-adapters.test.ts +86 -0
- package/src/tests/vcs-provider.test.ts +27 -0
- package/src/tests/workflow-agent-task.test.ts +196 -0
- package/src/tests/workflow-async-v2.test.ts +508 -0
- package/src/tests/workflow-convergence.test.ts +541 -0
- package/src/tests/workflow-definition-validation.test.ts +366 -0
- package/src/tests/workflow-engine-v2.test.ts +691 -0
- package/src/tests/workflow-executors.test.ts +736 -0
- package/src/tests/workflow-http-v2.test.ts +599 -0
- package/src/tests/workflow-integration-io.test.ts +902 -0
- package/src/tests/workflow-io-schemas.test.ts +624 -0
- package/src/tests/workflow-registry.test.ts +592 -0
- package/src/tests/workflow-retry-v2.test.ts +401 -0
- package/src/tests/workflow-retry-validation.test.ts +282 -0
- package/src/tests/workflow-schedule-trigger.test.ts +104 -0
- package/src/tests/workflow-template.test.ts +288 -0
- package/src/tests/workflow-trigger-schema.test.ts +359 -0
- package/src/tests/workflow-triggers-v2.test.ts +264 -0
- package/src/tests/workflow-versions.test.ts +208 -0
- package/src/tests/workflow-workspace.test.ts +272 -0
- package/src/tests/x402-client.test.ts +117 -0
- package/src/tests/x402-config.test.ts +182 -0
- package/src/tests/x402-spending-tracker.test.ts +185 -0
- package/src/tools/cancel-task.ts +2 -0
- package/src/tools/context-diff.ts +171 -0
- package/src/tools/context-history.ts +138 -0
- package/src/tools/create-channel.ts +1 -0
- package/src/tools/db-query.ts +78 -0
- package/src/tools/delete-channel.ts +132 -0
- package/src/tools/epics/assign-task-to-epic.ts +1 -0
- package/src/tools/epics/create-epic.ts +3 -2
- package/src/tools/epics/delete-epic.ts +2 -0
- package/src/tools/epics/get-epic-details.ts +2 -0
- package/src/tools/epics/list-epics.ts +2 -0
- package/src/tools/epics/unassign-task-from-epic.ts +1 -0
- package/src/tools/epics/update-epic.ts +7 -4
- package/src/tools/get-swarm.ts +2 -0
- package/src/tools/get-task-details.ts +2 -0
- package/src/tools/get-tasks.ts +27 -1
- package/src/tools/inject-learning.ts +106 -0
- package/src/tools/join-swarm.ts +17 -7
- package/src/tools/list-channels.ts +2 -0
- package/src/tools/list-services.ts +2 -0
- package/src/tools/memory-get.ts +56 -0
- package/src/tools/memory-search.ts +131 -0
- package/src/tools/my-agent-info.ts +2 -0
- package/src/tools/poll-task.ts +2 -20
- package/src/tools/post-message.ts +1 -0
- package/src/tools/prompt-templates/delete.ts +86 -0
- package/src/tools/prompt-templates/get.ts +89 -0
- package/src/tools/prompt-templates/index.ts +5 -0
- package/src/tools/prompt-templates/list.ts +95 -0
- package/src/tools/prompt-templates/preview.ts +84 -0
- package/src/tools/prompt-templates/set.ts +117 -0
- package/src/tools/read-messages.ts +2 -0
- package/src/tools/register-agentmail-inbox.ts +166 -0
- package/src/tools/register-service.ts +2 -0
- package/src/tools/schedules/create-schedule.ts +134 -24
- package/src/tools/schedules/delete-schedule.ts +2 -0
- package/src/tools/schedules/list-schedules.ts +20 -4
- package/src/tools/schedules/run-schedule-now.ts +1 -0
- package/src/tools/schedules/update-schedule.ts +49 -17
- package/src/tools/send-task.ts +132 -10
- package/src/tools/slack-download-file.ts +4 -2
- package/src/tools/slack-list-channels.ts +2 -0
- package/src/tools/slack-post.ts +2 -0
- package/src/tools/slack-read.ts +2 -0
- package/src/tools/slack-reply.ts +2 -0
- package/src/tools/slack-upload-file.ts +2 -0
- package/src/tools/store-progress.ts +205 -4
- package/src/tools/swarm-config/delete-config.ts +87 -0
- package/src/tools/swarm-config/get-config.ts +108 -0
- package/src/tools/swarm-config/index.ts +4 -0
- package/src/tools/swarm-config/list-config.ts +99 -0
- package/src/tools/swarm-config/set-config.ts +118 -0
- package/src/tools/task-action.ts +50 -5
- package/src/tools/task-dedup.ts +97 -0
- package/src/tools/templates.ts +53 -0
- package/src/tools/tool-config.ts +124 -0
- package/src/tools/tracker/index.ts +6 -0
- package/src/tools/tracker/tracker-link-epic.ts +64 -0
- package/src/tools/tracker/tracker-link-task.ts +64 -0
- package/src/tools/tracker/tracker-map-agent.ts +57 -0
- package/src/tools/tracker/tracker-status.ts +56 -0
- package/src/tools/tracker/tracker-sync-status.ts +42 -0
- package/src/tools/tracker/tracker-unlink.ts +41 -0
- package/src/tools/unregister-service.ts +2 -0
- package/src/tools/update-profile.ts +172 -17
- package/src/tools/update-service-status.ts +2 -0
- package/src/tools/utils.ts +10 -1
- package/src/tools/workflows/create-workflow.ts +129 -0
- package/src/tools/workflows/delete-workflow.ts +42 -0
- package/src/tools/workflows/get-workflow-run.ts +59 -0
- package/src/tools/workflows/get-workflow.ts +53 -0
- package/src/tools/workflows/index.ts +9 -0
- package/src/tools/workflows/list-workflow-runs.ts +48 -0
- package/src/tools/workflows/list-workflows.ts +42 -0
- package/src/tools/workflows/retry-workflow-run.ts +40 -0
- package/src/tools/workflows/trigger-workflow.ts +96 -0
- package/src/tools/workflows/update-workflow.ts +133 -0
- package/src/tracker/types.ts +51 -0
- package/src/types.ts +530 -14
- package/src/utils/credentials.test.ts +156 -0
- package/src/utils/credentials.ts +50 -0
- package/src/utils/error-tracker.ts +190 -0
- package/src/vcs/index.ts +15 -0
- package/src/vcs/types.ts +5 -0
- package/src/workflows/checkpoint.ts +121 -0
- package/src/workflows/cooldown.ts +28 -0
- package/src/workflows/definition.ts +235 -0
- package/src/workflows/engine.ts +580 -0
- package/src/workflows/event-bus.ts +29 -0
- package/src/workflows/executors/agent-task.ts +103 -0
- package/src/workflows/executors/base.ts +86 -0
- package/src/workflows/executors/code-match.ts +88 -0
- package/src/workflows/executors/index.ts +16 -0
- package/src/workflows/executors/notify.ts +93 -0
- package/src/workflows/executors/property-match.ts +104 -0
- package/src/workflows/executors/raw-llm.ts +83 -0
- package/src/workflows/executors/registry.ts +76 -0
- package/src/workflows/executors/script.ts +103 -0
- package/src/workflows/executors/validate.ts +215 -0
- package/src/workflows/executors/vcs.ts +58 -0
- package/src/workflows/index.ts +61 -0
- package/src/workflows/input.ts +46 -0
- package/src/workflows/json-schema-validator.ts +118 -0
- package/src/workflows/recovery.ts +139 -0
- package/src/workflows/resume.ts +229 -0
- package/src/workflows/retry-poller.ts +216 -0
- package/src/workflows/template.ts +74 -0
- package/src/workflows/templates.ts +86 -0
- package/src/workflows/triggers.ts +124 -0
- package/src/workflows/validation.ts +104 -0
- package/src/workflows/version.ts +44 -0
- package/src/x402/cli.ts +140 -0
- package/src/x402/client.ts +192 -0
- package/src/x402/config.ts +131 -0
- package/src/x402/index.ts +37 -0
- package/src/x402/openfort-signer.ts +83 -0
- package/src/x402/spending-tracker.ts +109 -0
- package/templates/official/coder/CLAUDE.md +49 -0
- package/templates/official/coder/IDENTITY.md +28 -0
- package/templates/official/coder/SOUL.md +43 -0
- package/templates/official/coder/TOOLS.md +40 -0
- package/templates/official/coder/config.json +23 -0
- package/templates/official/coder/start-up.sh +23 -0
- package/templates/official/content-reviewer/CLAUDE.md +68 -0
- package/templates/official/content-reviewer/IDENTITY.md +28 -0
- package/templates/official/content-reviewer/SOUL.md +44 -0
- package/templates/official/content-reviewer/TOOLS.md +37 -0
- package/templates/official/content-reviewer/config.json +23 -0
- package/templates/official/content-reviewer/start-up.sh +23 -0
- package/templates/official/content-strategist/CLAUDE.md +63 -0
- package/templates/official/content-strategist/IDENTITY.md +33 -0
- package/templates/official/content-strategist/SOUL.md +48 -0
- package/templates/official/content-strategist/TOOLS.md +47 -0
- package/templates/official/content-strategist/config.json +23 -0
- package/templates/official/content-strategist/start-up.sh +23 -0
- package/templates/official/content-writer/CLAUDE.md +72 -0
- package/templates/official/content-writer/IDENTITY.md +30 -0
- package/templates/official/content-writer/SOUL.md +46 -0
- package/templates/official/content-writer/TOOLS.md +44 -0
- package/templates/official/content-writer/config.json +23 -0
- package/templates/official/content-writer/start-up.sh +23 -0
- package/templates/official/forward-deployed-engineer/CLAUDE.md +54 -0
- package/templates/official/forward-deployed-engineer/IDENTITY.md +37 -0
- package/templates/official/forward-deployed-engineer/SOUL.md +55 -0
- package/templates/official/forward-deployed-engineer/config.json +21 -0
- package/templates/official/lead/CLAUDE.md +33 -0
- package/templates/official/lead/IDENTITY.md +36 -0
- package/templates/official/lead/SOUL.md +51 -0
- package/templates/official/lead/config.json +22 -0
- package/templates/official/researcher/CLAUDE.md +46 -0
- package/templates/official/researcher/IDENTITY.md +28 -0
- package/templates/official/researcher/SOUL.md +43 -0
- package/templates/official/researcher/config.json +21 -0
- package/templates/official/reviewer/CLAUDE.md +63 -0
- package/templates/official/reviewer/IDENTITY.md +28 -0
- package/templates/official/reviewer/SOUL.md +45 -0
- package/templates/official/reviewer/config.json +21 -0
- package/templates/official/tester/CLAUDE.md +53 -0
- package/templates/official/tester/IDENTITY.md +28 -0
- package/templates/official/tester/SOUL.md +55 -0
- package/templates/official/tester/config.json +21 -0
- package/templates/schema.ts +35 -0
- package/.claude/settings.local.json +0 -115
- package/.dockerignore +0 -61
- package/.editorconfig +0 -15
- package/.env.docker.example +0 -39
- package/.env.example +0 -40
- package/.github/workflows/ci.yml +0 -76
- package/.github/workflows/docker-and-deploy.yml +0 -117
- package/.wts-config.json +0 -4
- package/.wts-setup.ts +0 -102
- package/CLAUDE.md +0 -104
- package/CONTRIBUTING.md +0 -270
- package/DEPLOYMENT.md +0 -605
- package/Dockerfile +0 -57
- package/Dockerfile.worker +0 -157
- package/FAQ.md +0 -19
- package/MCP.md +0 -406
- package/UI.md +0 -40
- package/assets/agent-swarm-logo-orange.png +0 -0
- package/assets/agent-swarm-logo.png +0 -0
- package/assets/agent-swarm.mp4 +0 -0
- package/assets/agent-swarm.png +0 -0
- package/biome.json +0 -39
- package/deploy/DEPLOY.md +0 -60
- package/deploy/agent-swarm.service +0 -17
- package/deploy/docker-push.ts +0 -30
- package/deploy/install.ts +0 -85
- package/deploy/prod-db.ts +0 -42
- package/deploy/uninstall.ts +0 -12
- package/deploy/update.ts +0 -21
- package/docker-compose.example.yml +0 -159
- package/docker-entrypoint.sh +0 -352
- package/ecosystem.config.cjs +0 -66
- package/plugin/README.md +0 -1
- package/plugin/hooks/hooks.json +0 -71
- package/pyproject.toml +0 -9
- package/scripts/generate-mcp-docs.ts +0 -415
- package/slack-manifest.json +0 -71
- package/src/tests/get-inbox-message.test.ts +0 -145
- package/src/tools/get-inbox-message.ts +0 -89
- package/src/tools/inbox-delegate.ts +0 -113
- package/thoughts/shared/plans/2025-12-18-slack-integration.md +0 -1195
- package/thoughts/shared/plans/2025-12-19-agent-log-streaming.md +0 -732
- package/thoughts/shared/plans/2025-12-19-role-based-swarm-plugin.md +0 -361
- package/thoughts/shared/plans/2025-12-20-mobile-responsive-ui.md +0 -501
- package/thoughts/shared/plans/2025-12-20-startup-team-swarm.md +0 -560
- package/thoughts/shared/plans/2025-12-23-runner-level-polling.md +0 -934
- package/thoughts/shared/plans/2025-12-23-runner-session-logs.md +0 -1000
- package/thoughts/shared/plans/2025-12-23-worker-lead-spawn-triggers.md +0 -568
- package/thoughts/shared/plans/2026-01-09-inverse-teleport.md +0 -1516
- package/thoughts/shared/plans/2026-01-12-agent-rename-pm2-control.md +0 -1133
- package/thoughts/shared/plans/2026-01-12-github-app-integration.md +0 -380
- package/thoughts/shared/plans/2026-01-12-lead-inbox-model.md +0 -876
- package/thoughts/shared/plans/2026-01-12-ralph-wiggum-integration.md +0 -463
- package/thoughts/shared/plans/2026-01-13-agent-concurrency.md +0 -691
- package/thoughts/shared/plans/2026-01-13-github-assignment-handling.md +0 -690
- package/thoughts/shared/plans/2026-01-13-prevent-duplicate-trigger-processing.md +0 -1071
- package/thoughts/shared/plans/2026-01-14-fix-slack-thread-context.md +0 -507
- package/thoughts/shared/plans/2026-01-15-scheduled-tasks-implementation.md +0 -565
- package/thoughts/shared/plans/2026-01-15-usage-cost-tracking-ui.md +0 -1479
- package/thoughts/shared/plans/2026-01-16-epics-feature-implementation.md +0 -1230
- package/thoughts/shared/research/.gitkeep +0 -0
- package/thoughts/shared/research/2025-01-09-inverse-teleport-plan-review.md +0 -420
- package/thoughts/shared/research/2025-12-18-slack-integration.md +0 -442
- package/thoughts/shared/research/2025-12-19-agent-log-streaming.md +0 -339
- package/thoughts/shared/research/2025-12-19-agent-secrets-cli-research.md +0 -390
- package/thoughts/shared/research/2025-12-21-gemini-cli-integration.md +0 -376
- package/thoughts/shared/research/2025-12-22-runner-loop-architecture.md +0 -582
- package/thoughts/shared/research/2025-12-22-setup-experience-improvements.md +0 -264
- package/thoughts/shared/research/2026-01-13-lead-duplicate-trigger-processing.md +0 -223
- package/thoughts/shared/research/2026-01-14-lead-slack-thread-context.md +0 -277
- package/thoughts/shared/research/2026-01-15-ai-tracker-agent-swarm-integration.md +0 -376
- package/thoughts/shared/research/2026-01-15-auto-starting-processes-in-worker-containers.md +0 -787
- package/thoughts/shared/research/2026-01-15-scheduled-tasks.md +0 -390
- package/thoughts/shared/research/2026-01-16-epics-feature-research.md +0 -437
- package/thoughts/taras/plans/2026-01-22-agent-swarm-schemas.md +0 -98
- package/thoughts/taras/plans/2026-01-28-per-worker-claude-md.md +0 -617
- package/thoughts/taras/plans/2026-01-28-sentry-cli-integration.md +0 -214
- package/thoughts/taras/research/2026-01-22-vercel-cli-integration.md +0 -287
- package/thoughts/taras/research/2026-01-27-excessive-polling-issue.md +0 -311
- package/thoughts/taras/research/2026-01-28-per-worker-claude-md.md +0 -383
- package/thoughts/taras/research/2026-01-28-sentry-cli-integration.md +0 -240
- package/tsconfig.json +0 -37
- package/ui/CLAUDE.md +0 -49
- package/ui/bun.lock +0 -771
- package/ui/index.html +0 -22
- package/ui/package-lock.json +0 -5290
- package/ui/package.json +0 -33
- package/ui/pnpm-lock.yaml +0 -3341
- package/ui/postcss.config.js +0 -6
- package/ui/public/logo.png +0 -0
- package/ui/src/App.tsx +0 -63
- package/ui/src/components/ActivityFeed.tsx +0 -440
- package/ui/src/components/AgentDetailPanel.tsx +0 -733
- package/ui/src/components/AgentsPanel.tsx +0 -815
- package/ui/src/components/ChatPanel.tsx +0 -1920
- package/ui/src/components/ConfigModal.tsx +0 -253
- package/ui/src/components/Dashboard.tsx +0 -832
- package/ui/src/components/EditAgentProfileModal.tsx +0 -433
- package/ui/src/components/EpicDetailPage.tsx +0 -741
- package/ui/src/components/EpicsPanel.tsx +0 -566
- package/ui/src/components/Header.tsx +0 -160
- package/ui/src/components/JsonViewer.tsx +0 -171
- package/ui/src/components/ScheduledTaskDetailPanel.tsx +0 -517
- package/ui/src/components/ScheduledTasksPanel.tsx +0 -639
- package/ui/src/components/ServicesPanel.tsx +0 -622
- package/ui/src/components/SessionLogPanel.tsx +0 -1219
- package/ui/src/components/StatsBar.tsx +0 -321
- package/ui/src/components/StatusBadge.tsx +0 -168
- package/ui/src/components/TaskDetailPanel.tsx +0 -903
- package/ui/src/components/TasksPanel.tsx +0 -614
- package/ui/src/components/UsageCharts.tsx +0 -216
- package/ui/src/components/UsageTab.tsx +0 -394
- package/ui/src/hooks/queries.ts +0 -353
- package/ui/src/hooks/useAutoScroll.ts +0 -83
- package/ui/src/index.css +0 -257
- package/ui/src/lib/api.ts +0 -268
- package/ui/src/lib/config.ts +0 -35
- package/ui/src/lib/contentPreview.ts +0 -208
- package/ui/src/lib/theme.ts +0 -214
- package/ui/src/lib/utils.ts +0 -88
- package/ui/src/main.tsx +0 -28
- package/ui/src/types/api.ts +0 -323
- package/ui/src/vite-env.d.ts +0 -1
- package/ui/tailwind.config.js +0 -37
- package/ui/tsconfig.json +0 -31
- package/ui/vite.config.ts +0 -35
- /package/{thoughts/shared/plans → templates/community}/.gitkeep +0 -0
|
@@ -0,0 +1,86 @@
|
|
|
1
|
+
import { Database } from "bun:sqlite";
|
|
2
|
+
import { afterEach, describe, expect, test } from "bun:test";
|
|
3
|
+
import { unlink } from "node:fs/promises";
|
|
4
|
+
import { closeDb, initDb } from "../be/db";
|
|
5
|
+
|
|
6
|
+
const INCOMPLETE_DB_PATH = "./test-migration-incomplete.sqlite";
|
|
7
|
+
const FRESH_DB_PATH = "./test-migration-fresh.sqlite";
|
|
8
|
+
|
|
9
|
+
async function removeDbFiles(dbPath: string): Promise<void> {
|
|
10
|
+
for (const suffix of ["", "-wal", "-shm"]) {
|
|
11
|
+
try {
|
|
12
|
+
await unlink(dbPath + suffix);
|
|
13
|
+
} catch (error) {
|
|
14
|
+
if ((error as NodeJS.ErrnoException).code !== "ENOENT") {
|
|
15
|
+
throw error;
|
|
16
|
+
}
|
|
17
|
+
}
|
|
18
|
+
}
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
afterEach(async () => {
|
|
22
|
+
closeDb();
|
|
23
|
+
await removeDbFiles(INCOMPLETE_DB_PATH);
|
|
24
|
+
await removeDbFiles(FRESH_DB_PATH);
|
|
25
|
+
});
|
|
26
|
+
|
|
27
|
+
describe("migration regressions", () => {
|
|
28
|
+
test("incomplete existing DB runs 001_initial instead of blind bootstrap", () => {
|
|
29
|
+
const now = new Date().toISOString();
|
|
30
|
+
const legacyDb = new Database(INCOMPLETE_DB_PATH, { create: true });
|
|
31
|
+
legacyDb.run(`
|
|
32
|
+
CREATE TABLE agents (
|
|
33
|
+
id TEXT PRIMARY KEY,
|
|
34
|
+
name TEXT NOT NULL,
|
|
35
|
+
isLead INTEGER NOT NULL DEFAULT 0,
|
|
36
|
+
status TEXT NOT NULL,
|
|
37
|
+
maxTasks INTEGER DEFAULT 1,
|
|
38
|
+
emptyPollCount INTEGER DEFAULT 0,
|
|
39
|
+
createdAt TEXT NOT NULL,
|
|
40
|
+
lastUpdatedAt TEXT NOT NULL
|
|
41
|
+
)
|
|
42
|
+
`);
|
|
43
|
+
legacyDb.run(
|
|
44
|
+
"INSERT INTO agents (id, name, isLead, status, createdAt, lastUpdatedAt) VALUES (?, ?, ?, ?, ?, ?)",
|
|
45
|
+
[crypto.randomUUID(), "legacy", 0, "idle", now, now],
|
|
46
|
+
);
|
|
47
|
+
legacyDb.close();
|
|
48
|
+
|
|
49
|
+
const database = initDb(INCOMPLETE_DB_PATH);
|
|
50
|
+
|
|
51
|
+
const channelsTable = database
|
|
52
|
+
.prepare<{ name: string }, []>(
|
|
53
|
+
"SELECT name FROM sqlite_master WHERE type='table' AND name='channels'",
|
|
54
|
+
)
|
|
55
|
+
.get();
|
|
56
|
+
expect(channelsTable?.name).toBe("channels");
|
|
57
|
+
|
|
58
|
+
const generalChannel = database
|
|
59
|
+
.prepare<{ id: string }, []>("SELECT id FROM channels WHERE name = 'general'")
|
|
60
|
+
.get();
|
|
61
|
+
expect(generalChannel?.id).toBe("00000000-0000-4000-8000-000000000001");
|
|
62
|
+
|
|
63
|
+
const columns = database
|
|
64
|
+
.prepare<{ name: string }, []>("PRAGMA table_info(agents)")
|
|
65
|
+
.all()
|
|
66
|
+
.map((column) => column.name);
|
|
67
|
+
expect(columns).toContain("soulMd");
|
|
68
|
+
expect(columns).toContain("identityMd");
|
|
69
|
+
expect(columns).toContain("toolsMd");
|
|
70
|
+
expect(columns).toContain("claudeMd");
|
|
71
|
+
expect(columns).toContain("setupScript");
|
|
72
|
+
});
|
|
73
|
+
|
|
74
|
+
test("fresh DB preserves source CHECK constraint on agent_tasks", () => {
|
|
75
|
+
const database = initDb(FRESH_DB_PATH);
|
|
76
|
+
const now = new Date().toISOString();
|
|
77
|
+
|
|
78
|
+
expect(() => {
|
|
79
|
+
database.run(
|
|
80
|
+
`INSERT INTO agent_tasks (id, task, status, source, createdAt, lastUpdatedAt)
|
|
81
|
+
VALUES (?, ?, ?, ?, ?, ?)`,
|
|
82
|
+
[crypto.randomUUID(), "invalid source", "pending", "not-valid", now, now],
|
|
83
|
+
);
|
|
84
|
+
}).toThrow();
|
|
85
|
+
});
|
|
86
|
+
});
|
|
@@ -0,0 +1,338 @@
|
|
|
1
|
+
import { afterAll, beforeAll, describe, expect, test } from "bun:test";
|
|
2
|
+
import { unlinkSync } from "node:fs";
|
|
3
|
+
import {
|
|
4
|
+
closeDb,
|
|
5
|
+
createAgent,
|
|
6
|
+
createScheduledTask,
|
|
7
|
+
createTaskExtended,
|
|
8
|
+
getResolvedConfig,
|
|
9
|
+
getScheduledTaskById,
|
|
10
|
+
getTaskById,
|
|
11
|
+
initDb,
|
|
12
|
+
updateScheduledTask,
|
|
13
|
+
upsertSwarmConfig,
|
|
14
|
+
} from "../be/db";
|
|
15
|
+
import { runScheduleNow } from "../scheduler";
|
|
16
|
+
|
|
17
|
+
const TEST_DB_PATH = "./test-model-control.sqlite";
|
|
18
|
+
|
|
19
|
+
beforeAll(() => {
|
|
20
|
+
initDb(TEST_DB_PATH);
|
|
21
|
+
});
|
|
22
|
+
|
|
23
|
+
afterAll(() => {
|
|
24
|
+
closeDb();
|
|
25
|
+
try {
|
|
26
|
+
unlinkSync(TEST_DB_PATH);
|
|
27
|
+
unlinkSync(`${TEST_DB_PATH}-wal`);
|
|
28
|
+
unlinkSync(`${TEST_DB_PATH}-shm`);
|
|
29
|
+
} catch {
|
|
30
|
+
// ignore if files don't exist
|
|
31
|
+
}
|
|
32
|
+
});
|
|
33
|
+
|
|
34
|
+
describe("Model Control - Task Creation", () => {
|
|
35
|
+
test("should store model when creating a task with model='sonnet'", () => {
|
|
36
|
+
const task = createTaskExtended("Test task with sonnet", { model: "sonnet" });
|
|
37
|
+
expect(task.model).toBe("sonnet");
|
|
38
|
+
|
|
39
|
+
const retrieved = getTaskById(task.id);
|
|
40
|
+
expect(retrieved?.model).toBe("sonnet");
|
|
41
|
+
});
|
|
42
|
+
|
|
43
|
+
test("should store model when creating a task with model='haiku'", () => {
|
|
44
|
+
const task = createTaskExtended("Test task with haiku", { model: "haiku" });
|
|
45
|
+
expect(task.model).toBe("haiku");
|
|
46
|
+
});
|
|
47
|
+
|
|
48
|
+
test("should store model when creating a task with model='opus'", () => {
|
|
49
|
+
const task = createTaskExtended("Test task with opus", { model: "opus" });
|
|
50
|
+
expect(task.model).toBe("opus");
|
|
51
|
+
});
|
|
52
|
+
|
|
53
|
+
test("should default model to undefined when not specified", () => {
|
|
54
|
+
const task = createTaskExtended("Test task without model");
|
|
55
|
+
expect(task.model).toBeUndefined();
|
|
56
|
+
});
|
|
57
|
+
|
|
58
|
+
test("should preserve model alongside other task options", () => {
|
|
59
|
+
const agent = createAgent({ name: "model-test-agent", isLead: false, status: "idle" });
|
|
60
|
+
|
|
61
|
+
const task = createTaskExtended("Task with model and options", {
|
|
62
|
+
model: "haiku",
|
|
63
|
+
agentId: agent.id,
|
|
64
|
+
priority: 80,
|
|
65
|
+
taskType: "test",
|
|
66
|
+
tags: ["model-test"],
|
|
67
|
+
});
|
|
68
|
+
|
|
69
|
+
expect(task.model).toBe("haiku");
|
|
70
|
+
expect(task.agentId).toBe(agent.id);
|
|
71
|
+
expect(task.priority).toBe(80);
|
|
72
|
+
expect(task.taskType).toBe("test");
|
|
73
|
+
expect(task.tags).toContain("model-test");
|
|
74
|
+
});
|
|
75
|
+
|
|
76
|
+
test("should store model on offered tasks", () => {
|
|
77
|
+
const agent = createAgent({ name: "offer-model-agent", isLead: false, status: "idle" });
|
|
78
|
+
|
|
79
|
+
const task = createTaskExtended("Offered task with model", {
|
|
80
|
+
model: "sonnet",
|
|
81
|
+
offeredTo: agent.id,
|
|
82
|
+
});
|
|
83
|
+
|
|
84
|
+
expect(task.model).toBe("sonnet");
|
|
85
|
+
expect(task.status).toBe("offered");
|
|
86
|
+
});
|
|
87
|
+
});
|
|
88
|
+
|
|
89
|
+
describe("Model Control - Schedule Creation", () => {
|
|
90
|
+
test("should store model on scheduled task creation", () => {
|
|
91
|
+
const schedule = createScheduledTask({
|
|
92
|
+
name: "model-schedule-sonnet",
|
|
93
|
+
intervalMs: 60000,
|
|
94
|
+
taskTemplate: "Scheduled with sonnet",
|
|
95
|
+
model: "sonnet",
|
|
96
|
+
});
|
|
97
|
+
|
|
98
|
+
expect(schedule.model).toBe("sonnet");
|
|
99
|
+
|
|
100
|
+
const retrieved = getScheduledTaskById(schedule.id);
|
|
101
|
+
expect(retrieved?.model).toBe("sonnet");
|
|
102
|
+
});
|
|
103
|
+
|
|
104
|
+
test("should store all valid model values on schedules", () => {
|
|
105
|
+
for (const model of ["haiku", "sonnet", "opus"] as const) {
|
|
106
|
+
const schedule = createScheduledTask({
|
|
107
|
+
name: `model-schedule-all-${model}-${Date.now()}`,
|
|
108
|
+
intervalMs: 60000,
|
|
109
|
+
taskTemplate: `Scheduled with ${model}`,
|
|
110
|
+
model,
|
|
111
|
+
});
|
|
112
|
+
|
|
113
|
+
expect(schedule.model).toBe(model);
|
|
114
|
+
}
|
|
115
|
+
});
|
|
116
|
+
|
|
117
|
+
test("should default model to undefined when not specified on schedule", () => {
|
|
118
|
+
const schedule = createScheduledTask({
|
|
119
|
+
name: "model-schedule-default",
|
|
120
|
+
intervalMs: 60000,
|
|
121
|
+
taskTemplate: "Scheduled without model",
|
|
122
|
+
});
|
|
123
|
+
|
|
124
|
+
expect(schedule.model).toBeUndefined();
|
|
125
|
+
});
|
|
126
|
+
});
|
|
127
|
+
|
|
128
|
+
describe("Model Control - Schedule Update", () => {
|
|
129
|
+
test("should update model on existing schedule", () => {
|
|
130
|
+
const schedule = createScheduledTask({
|
|
131
|
+
name: "model-update-test",
|
|
132
|
+
intervalMs: 60000,
|
|
133
|
+
taskTemplate: "Update model test",
|
|
134
|
+
model: "opus",
|
|
135
|
+
});
|
|
136
|
+
|
|
137
|
+
expect(schedule.model).toBe("opus");
|
|
138
|
+
|
|
139
|
+
const updated = updateScheduledTask(schedule.id, { model: "haiku" });
|
|
140
|
+
expect(updated?.model).toBe("haiku");
|
|
141
|
+
|
|
142
|
+
const retrieved = getScheduledTaskById(schedule.id);
|
|
143
|
+
expect(retrieved?.model).toBe("haiku");
|
|
144
|
+
});
|
|
145
|
+
|
|
146
|
+
test("should clear model by setting to null", () => {
|
|
147
|
+
const schedule = createScheduledTask({
|
|
148
|
+
name: "model-clear-test",
|
|
149
|
+
intervalMs: 60000,
|
|
150
|
+
taskTemplate: "Clear model test",
|
|
151
|
+
model: "sonnet",
|
|
152
|
+
});
|
|
153
|
+
|
|
154
|
+
expect(schedule.model).toBe("sonnet");
|
|
155
|
+
|
|
156
|
+
const updated = updateScheduledTask(schedule.id, { model: null });
|
|
157
|
+
expect(updated?.model).toBeUndefined();
|
|
158
|
+
});
|
|
159
|
+
|
|
160
|
+
test("should preserve model when updating other fields", () => {
|
|
161
|
+
const schedule = createScheduledTask({
|
|
162
|
+
name: "model-preserve-test",
|
|
163
|
+
intervalMs: 60000,
|
|
164
|
+
taskTemplate: "Preserve model test",
|
|
165
|
+
model: "haiku",
|
|
166
|
+
});
|
|
167
|
+
|
|
168
|
+
const updated = updateScheduledTask(schedule.id, { priority: 90 });
|
|
169
|
+
expect(updated?.model).toBe("haiku");
|
|
170
|
+
expect(updated?.priority).toBe(90);
|
|
171
|
+
});
|
|
172
|
+
});
|
|
173
|
+
|
|
174
|
+
describe("Model Control - Schedule to Task Propagation", () => {
|
|
175
|
+
test("should propagate model from schedule to task on manual run", async () => {
|
|
176
|
+
const schedule = createScheduledTask({
|
|
177
|
+
name: "model-propagate-manual",
|
|
178
|
+
intervalMs: 60000,
|
|
179
|
+
taskTemplate: "Propagated model task (manual)",
|
|
180
|
+
model: "haiku",
|
|
181
|
+
enabled: true,
|
|
182
|
+
});
|
|
183
|
+
|
|
184
|
+
await runScheduleNow(schedule.id);
|
|
185
|
+
|
|
186
|
+
// Find the created task by its template text
|
|
187
|
+
const { getDb } = await import("../be/db");
|
|
188
|
+
const row = getDb()
|
|
189
|
+
.query("SELECT id FROM agent_tasks WHERE task = ? ORDER BY createdAt DESC LIMIT 1")
|
|
190
|
+
.get("Propagated model task (manual)") as { id: string } | null;
|
|
191
|
+
|
|
192
|
+
expect(row).not.toBeNull();
|
|
193
|
+
const task = getTaskById(row!.id);
|
|
194
|
+
expect(task?.model).toBe("haiku");
|
|
195
|
+
});
|
|
196
|
+
|
|
197
|
+
test("should create task without model when schedule has no model", async () => {
|
|
198
|
+
const schedule = createScheduledTask({
|
|
199
|
+
name: "model-propagate-none",
|
|
200
|
+
intervalMs: 60000,
|
|
201
|
+
taskTemplate: "Propagated no-model task",
|
|
202
|
+
enabled: true,
|
|
203
|
+
});
|
|
204
|
+
|
|
205
|
+
await runScheduleNow(schedule.id);
|
|
206
|
+
|
|
207
|
+
const { getDb } = await import("../be/db");
|
|
208
|
+
const row = getDb()
|
|
209
|
+
.query("SELECT id FROM agent_tasks WHERE task = ? ORDER BY createdAt DESC LIMIT 1")
|
|
210
|
+
.get("Propagated no-model task") as { id: string } | null;
|
|
211
|
+
|
|
212
|
+
expect(row).not.toBeNull();
|
|
213
|
+
const task = getTaskById(row!.id);
|
|
214
|
+
expect(task?.model).toBeUndefined();
|
|
215
|
+
});
|
|
216
|
+
});
|
|
217
|
+
|
|
218
|
+
describe("Model Control - Config MODEL_OVERRIDE Resolution", () => {
|
|
219
|
+
test("should resolve global MODEL_OVERRIDE config", () => {
|
|
220
|
+
upsertSwarmConfig({
|
|
221
|
+
scope: "global",
|
|
222
|
+
key: "MODEL_OVERRIDE",
|
|
223
|
+
value: "sonnet",
|
|
224
|
+
});
|
|
225
|
+
|
|
226
|
+
const configs = getResolvedConfig();
|
|
227
|
+
const modelOverride = configs.find((c) => c.key === "MODEL_OVERRIDE");
|
|
228
|
+
expect(modelOverride).toBeDefined();
|
|
229
|
+
expect(modelOverride?.value).toBe("sonnet");
|
|
230
|
+
});
|
|
231
|
+
|
|
232
|
+
test("agent-scoped MODEL_OVERRIDE should override global", () => {
|
|
233
|
+
const agent = createAgent({ name: "config-agent", isLead: false, status: "idle" });
|
|
234
|
+
|
|
235
|
+
upsertSwarmConfig({
|
|
236
|
+
scope: "global",
|
|
237
|
+
key: "MODEL_OVERRIDE",
|
|
238
|
+
value: "opus",
|
|
239
|
+
});
|
|
240
|
+
|
|
241
|
+
upsertSwarmConfig({
|
|
242
|
+
scope: "agent",
|
|
243
|
+
scopeId: agent.id,
|
|
244
|
+
key: "MODEL_OVERRIDE",
|
|
245
|
+
value: "haiku",
|
|
246
|
+
});
|
|
247
|
+
|
|
248
|
+
const configs = getResolvedConfig(agent.id);
|
|
249
|
+
const modelOverride = configs.find((c) => c.key === "MODEL_OVERRIDE");
|
|
250
|
+
expect(modelOverride?.value).toBe("haiku");
|
|
251
|
+
expect(modelOverride?.scope).toBe("agent");
|
|
252
|
+
});
|
|
253
|
+
|
|
254
|
+
test("should fallback to global when no agent-scoped config exists", () => {
|
|
255
|
+
const agent = createAgent({ name: "fallback-agent", isLead: false, status: "idle" });
|
|
256
|
+
|
|
257
|
+
upsertSwarmConfig({
|
|
258
|
+
scope: "global",
|
|
259
|
+
key: "MODEL_OVERRIDE",
|
|
260
|
+
value: "sonnet",
|
|
261
|
+
});
|
|
262
|
+
|
|
263
|
+
const configs = getResolvedConfig(agent.id);
|
|
264
|
+
const modelOverride = configs.find((c) => c.key === "MODEL_OVERRIDE");
|
|
265
|
+
expect(modelOverride?.value).toBe("sonnet");
|
|
266
|
+
expect(modelOverride?.scope).toBe("global");
|
|
267
|
+
});
|
|
268
|
+
});
|
|
269
|
+
|
|
270
|
+
describe("Model Control - Priority Resolution Logic", () => {
|
|
271
|
+
// The runner resolves model as: task.model || freshEnv.MODEL_OVERRIDE || "opus"
|
|
272
|
+
// We test the same logic pattern here to ensure correctness
|
|
273
|
+
|
|
274
|
+
function resolveModel(taskModel?: string, configOverride?: string): string {
|
|
275
|
+
return taskModel || configOverride || "opus";
|
|
276
|
+
}
|
|
277
|
+
|
|
278
|
+
test("task.model takes highest priority", () => {
|
|
279
|
+
expect(resolveModel("haiku", "sonnet")).toBe("haiku");
|
|
280
|
+
});
|
|
281
|
+
|
|
282
|
+
test("config MODEL_OVERRIDE is used when task has no model", () => {
|
|
283
|
+
expect(resolveModel(undefined, "sonnet")).toBe("sonnet");
|
|
284
|
+
});
|
|
285
|
+
|
|
286
|
+
test("defaults to 'opus' when no task model and no config override", () => {
|
|
287
|
+
expect(resolveModel(undefined, undefined)).toBe("opus");
|
|
288
|
+
});
|
|
289
|
+
|
|
290
|
+
test("empty string task model falls through to config", () => {
|
|
291
|
+
expect(resolveModel("", "sonnet")).toBe("sonnet");
|
|
292
|
+
});
|
|
293
|
+
|
|
294
|
+
test("empty string config override falls through to default", () => {
|
|
295
|
+
expect(resolveModel(undefined, "")).toBe("opus");
|
|
296
|
+
});
|
|
297
|
+
|
|
298
|
+
test("all three levels specified — task wins", () => {
|
|
299
|
+
expect(resolveModel("haiku", "sonnet")).toBe("haiku");
|
|
300
|
+
// "opus" is the hardcoded default, tested implicitly
|
|
301
|
+
});
|
|
302
|
+
});
|
|
303
|
+
|
|
304
|
+
describe("Model Control - Zod Validation Schema", () => {
|
|
305
|
+
// The MCP tools use z.enum(["haiku", "sonnet", "opus"]) for validation.
|
|
306
|
+
// We test the schema directly to ensure only valid values are accepted.
|
|
307
|
+
|
|
308
|
+
test("should accept valid model values", async () => {
|
|
309
|
+
const { z } = await import("zod");
|
|
310
|
+
const modelSchema = z.enum(["haiku", "sonnet", "opus"]).optional();
|
|
311
|
+
|
|
312
|
+
expect(modelSchema.parse("haiku")).toBe("haiku");
|
|
313
|
+
expect(modelSchema.parse("sonnet")).toBe("sonnet");
|
|
314
|
+
expect(modelSchema.parse("opus")).toBe("opus");
|
|
315
|
+
expect(modelSchema.parse(undefined)).toBeUndefined();
|
|
316
|
+
});
|
|
317
|
+
|
|
318
|
+
test("should reject invalid model values", async () => {
|
|
319
|
+
const { z } = await import("zod");
|
|
320
|
+
const modelSchema = z.enum(["haiku", "sonnet", "opus"]).optional();
|
|
321
|
+
|
|
322
|
+
expect(() => modelSchema.parse("gpt-4")).toThrow();
|
|
323
|
+
expect(() => modelSchema.parse("claude")).toThrow();
|
|
324
|
+
expect(() => modelSchema.parse("turbo")).toThrow();
|
|
325
|
+
expect(() => modelSchema.parse("")).toThrow();
|
|
326
|
+
expect(() => modelSchema.parse(123)).toThrow();
|
|
327
|
+
expect(() => modelSchema.parse(null)).toThrow();
|
|
328
|
+
});
|
|
329
|
+
|
|
330
|
+
test("nullable model schema (update-schedule) should accept null", async () => {
|
|
331
|
+
const { z } = await import("zod");
|
|
332
|
+
const modelSchema = z.enum(["haiku", "sonnet", "opus"]).nullable().optional();
|
|
333
|
+
|
|
334
|
+
expect(modelSchema.parse(null)).toBeNull();
|
|
335
|
+
expect(modelSchema.parse("haiku")).toBe("haiku");
|
|
336
|
+
expect(modelSchema.parse(undefined)).toBeUndefined();
|
|
337
|
+
});
|
|
338
|
+
});
|
|
@@ -0,0 +1,147 @@
|
|
|
1
|
+
import { afterAll, beforeAll, beforeEach, describe, expect, test } from "bun:test";
|
|
2
|
+
import { unlink } from "node:fs/promises";
|
|
3
|
+
import { closeDb, initDb } from "../be/db";
|
|
4
|
+
import { upsertOAuthApp } from "../be/db-queries/oauth";
|
|
5
|
+
import {
|
|
6
|
+
_clearPendingStates,
|
|
7
|
+
_getPendingState,
|
|
8
|
+
buildAuthorizationUrl,
|
|
9
|
+
exchangeCode,
|
|
10
|
+
type OAuthProviderConfig,
|
|
11
|
+
} from "../oauth/wrapper";
|
|
12
|
+
|
|
13
|
+
const TEST_DB_PATH = "./test-oauth-wrapper.sqlite";
|
|
14
|
+
|
|
15
|
+
const testConfig: OAuthProviderConfig = {
|
|
16
|
+
provider: "test-provider",
|
|
17
|
+
clientId: "test-client-id",
|
|
18
|
+
clientSecret: "test-client-secret",
|
|
19
|
+
authorizeUrl: "https://example.com/oauth/authorize",
|
|
20
|
+
tokenUrl: "https://example.com/oauth/token",
|
|
21
|
+
redirectUri: "http://localhost:3013/callback",
|
|
22
|
+
scopes: ["read", "write"],
|
|
23
|
+
extraParams: { actor: "app" },
|
|
24
|
+
};
|
|
25
|
+
|
|
26
|
+
beforeAll(() => {
|
|
27
|
+
initDb(TEST_DB_PATH);
|
|
28
|
+
// Create an oauth_app row so token storage works (FK constraint)
|
|
29
|
+
upsertOAuthApp("test-provider", {
|
|
30
|
+
clientId: testConfig.clientId,
|
|
31
|
+
clientSecret: testConfig.clientSecret,
|
|
32
|
+
authorizeUrl: testConfig.authorizeUrl,
|
|
33
|
+
tokenUrl: testConfig.tokenUrl,
|
|
34
|
+
redirectUri: testConfig.redirectUri,
|
|
35
|
+
scopes: testConfig.scopes.join(","),
|
|
36
|
+
});
|
|
37
|
+
});
|
|
38
|
+
|
|
39
|
+
beforeEach(() => {
|
|
40
|
+
_clearPendingStates();
|
|
41
|
+
});
|
|
42
|
+
|
|
43
|
+
afterAll(async () => {
|
|
44
|
+
closeDb();
|
|
45
|
+
await unlink(TEST_DB_PATH).catch(() => {});
|
|
46
|
+
await unlink(`${TEST_DB_PATH}-wal`).catch(() => {});
|
|
47
|
+
await unlink(`${TEST_DB_PATH}-shm`).catch(() => {});
|
|
48
|
+
});
|
|
49
|
+
|
|
50
|
+
describe("buildAuthorizationUrl", () => {
|
|
51
|
+
test("generates a valid URL with PKCE params", async () => {
|
|
52
|
+
const result = await buildAuthorizationUrl(testConfig);
|
|
53
|
+
|
|
54
|
+
expect(result.url).toBeTruthy();
|
|
55
|
+
expect(result.state).toBeTruthy();
|
|
56
|
+
expect(result.codeVerifier).toBeTruthy();
|
|
57
|
+
|
|
58
|
+
const url = new URL(result.url);
|
|
59
|
+
expect(url.origin + url.pathname).toBe("https://example.com/oauth/authorize");
|
|
60
|
+
expect(url.searchParams.get("client_id")).toBe("test-client-id");
|
|
61
|
+
expect(url.searchParams.get("redirect_uri")).toBe("http://localhost:3013/callback");
|
|
62
|
+
expect(url.searchParams.get("response_type")).toBe("code");
|
|
63
|
+
expect(url.searchParams.get("scope")).toBe("read,write");
|
|
64
|
+
expect(url.searchParams.get("state")).toBe(result.state);
|
|
65
|
+
expect(url.searchParams.get("code_challenge")).toBeTruthy();
|
|
66
|
+
expect(url.searchParams.get("code_challenge_method")).toBe("S256");
|
|
67
|
+
});
|
|
68
|
+
|
|
69
|
+
test("includes extra params in the URL", async () => {
|
|
70
|
+
const result = await buildAuthorizationUrl(testConfig);
|
|
71
|
+
const url = new URL(result.url);
|
|
72
|
+
expect(url.searchParams.get("actor")).toBe("app");
|
|
73
|
+
});
|
|
74
|
+
|
|
75
|
+
test("stores pending state with code verifier", async () => {
|
|
76
|
+
const result = await buildAuthorizationUrl(testConfig);
|
|
77
|
+
const pending = _getPendingState(result.state);
|
|
78
|
+
|
|
79
|
+
expect(pending).toBeTruthy();
|
|
80
|
+
expect(pending!.codeVerifier).toBe(result.codeVerifier);
|
|
81
|
+
expect(pending!.config.provider).toBe("test-provider");
|
|
82
|
+
expect(pending!.createdAt).toBeGreaterThan(0);
|
|
83
|
+
});
|
|
84
|
+
|
|
85
|
+
test("generates unique state for each call", async () => {
|
|
86
|
+
const result1 = await buildAuthorizationUrl(testConfig);
|
|
87
|
+
const result2 = await buildAuthorizationUrl(testConfig);
|
|
88
|
+
|
|
89
|
+
expect(result1.state).not.toBe(result2.state);
|
|
90
|
+
expect(result1.codeVerifier).not.toBe(result2.codeVerifier);
|
|
91
|
+
});
|
|
92
|
+
|
|
93
|
+
test("works without extra params", async () => {
|
|
94
|
+
const configNoExtras: OAuthProviderConfig = {
|
|
95
|
+
...testConfig,
|
|
96
|
+
extraParams: undefined,
|
|
97
|
+
};
|
|
98
|
+
|
|
99
|
+
const result = await buildAuthorizationUrl(configNoExtras);
|
|
100
|
+
const url = new URL(result.url);
|
|
101
|
+
expect(url.searchParams.get("actor")).toBeNull();
|
|
102
|
+
});
|
|
103
|
+
});
|
|
104
|
+
|
|
105
|
+
describe("exchangeCode", () => {
|
|
106
|
+
test("rejects invalid state", async () => {
|
|
107
|
+
await expect(exchangeCode(testConfig, "some-code", "invalid-state")).rejects.toThrow(
|
|
108
|
+
"Invalid or expired OAuth state",
|
|
109
|
+
);
|
|
110
|
+
});
|
|
111
|
+
|
|
112
|
+
test("rejects already-consumed state", async () => {
|
|
113
|
+
const result = await buildAuthorizationUrl(testConfig);
|
|
114
|
+
|
|
115
|
+
// First exchange attempt will fail because there's no real token server,
|
|
116
|
+
// but it should consume the state
|
|
117
|
+
try {
|
|
118
|
+
await exchangeCode(testConfig, "some-code", result.state);
|
|
119
|
+
} catch {
|
|
120
|
+
// Expected: fetch to token URL fails
|
|
121
|
+
}
|
|
122
|
+
|
|
123
|
+
// Second attempt with the same state should fail with "Invalid or expired"
|
|
124
|
+
await expect(exchangeCode(testConfig, "some-code", result.state)).rejects.toThrow(
|
|
125
|
+
"Invalid or expired OAuth state",
|
|
126
|
+
);
|
|
127
|
+
});
|
|
128
|
+
});
|
|
129
|
+
|
|
130
|
+
describe("state TTL cleanup", () => {
|
|
131
|
+
test("expired states are cleaned up on next buildAuthorizationUrl call", async () => {
|
|
132
|
+
// Manually insert an "expired" entry by backdating createdAt
|
|
133
|
+
const result = await buildAuthorizationUrl(testConfig);
|
|
134
|
+
const pending = _getPendingState(result.state);
|
|
135
|
+
expect(pending).toBeTruthy();
|
|
136
|
+
|
|
137
|
+
// Backdate to 11 minutes ago (past the 10-minute TTL)
|
|
138
|
+
pending!.createdAt = Date.now() - 11 * 60 * 1000;
|
|
139
|
+
|
|
140
|
+
// Building a new URL triggers cleanup
|
|
141
|
+
await buildAuthorizationUrl(testConfig);
|
|
142
|
+
|
|
143
|
+
// The expired state should be gone
|
|
144
|
+
const expired = _getPendingState(result.state);
|
|
145
|
+
expect(expired).toBeUndefined();
|
|
146
|
+
});
|
|
147
|
+
});
|