@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,181 @@
|
|
|
1
|
+
import { afterAll, beforeAll, describe, expect, test } from "bun:test";
|
|
2
|
+
import { unlinkSync } from "node:fs";
|
|
3
|
+
import { closeDb, createAgent, createTaskExtended, initDb } from "../be/db";
|
|
4
|
+
import { routeMessage } from "../slack/router";
|
|
5
|
+
import type { Agent } from "../types";
|
|
6
|
+
|
|
7
|
+
const TEST_DB_PATH = "./test-slack-router.sqlite";
|
|
8
|
+
|
|
9
|
+
let leadAgent: Agent;
|
|
10
|
+
let workerAgent: Agent;
|
|
11
|
+
|
|
12
|
+
beforeAll(() => {
|
|
13
|
+
initDb(TEST_DB_PATH);
|
|
14
|
+
leadAgent = createAgent({ name: "test-lead", isLead: true, status: "idle", capabilities: [] });
|
|
15
|
+
workerAgent = createAgent({
|
|
16
|
+
name: "test-worker",
|
|
17
|
+
isLead: false,
|
|
18
|
+
status: "idle",
|
|
19
|
+
capabilities: [],
|
|
20
|
+
});
|
|
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("Slack router — thread follow-up routing", () => {
|
|
35
|
+
describe("message in thread with active worker routes to worker", () => {
|
|
36
|
+
test("routes to worker with in_progress task in thread", () => {
|
|
37
|
+
const channelId = "C100";
|
|
38
|
+
const threadTs = "1000.0001";
|
|
39
|
+
|
|
40
|
+
// Create an in_progress task assigned to the worker in this thread
|
|
41
|
+
createTaskExtended("original task", {
|
|
42
|
+
agentId: workerAgent.id,
|
|
43
|
+
source: "slack",
|
|
44
|
+
slackChannelId: channelId,
|
|
45
|
+
slackThreadTs: threadTs,
|
|
46
|
+
slackUserId: "U_HUMAN",
|
|
47
|
+
});
|
|
48
|
+
|
|
49
|
+
const matches = routeMessage("<@BOT123> check the status", "BOT123", true, {
|
|
50
|
+
channelId,
|
|
51
|
+
threadTs,
|
|
52
|
+
});
|
|
53
|
+
|
|
54
|
+
expect(matches).toHaveLength(1);
|
|
55
|
+
expect(matches[0].agent.id).toBe(workerAgent.id);
|
|
56
|
+
expect(matches[0].matchedText).toBe("thread follow-up");
|
|
57
|
+
});
|
|
58
|
+
});
|
|
59
|
+
|
|
60
|
+
describe("message in thread with no active worker falls through to lead", () => {
|
|
61
|
+
test("routes to lead when no tasks exist in thread", () => {
|
|
62
|
+
const matches = routeMessage("<@BOT123> do something", "BOT123", true, {
|
|
63
|
+
channelId: "C200",
|
|
64
|
+
threadTs: "2000.0001",
|
|
65
|
+
});
|
|
66
|
+
|
|
67
|
+
expect(matches).toHaveLength(1);
|
|
68
|
+
expect(matches[0].agent.id).toBe(leadAgent.id);
|
|
69
|
+
expect(matches[0].matchedText).toBe("@bot");
|
|
70
|
+
});
|
|
71
|
+
});
|
|
72
|
+
|
|
73
|
+
describe("message in thread with offline worker falls through to lead", () => {
|
|
74
|
+
test("routes to lead when worker is offline", () => {
|
|
75
|
+
const channelId = "C300";
|
|
76
|
+
const threadTs = "3000.0001";
|
|
77
|
+
|
|
78
|
+
// Create an offline worker
|
|
79
|
+
const offlineWorker = createAgent({
|
|
80
|
+
name: "offline-worker",
|
|
81
|
+
isLead: false,
|
|
82
|
+
status: "offline",
|
|
83
|
+
capabilities: [],
|
|
84
|
+
});
|
|
85
|
+
|
|
86
|
+
// Create an in_progress task assigned to the offline worker
|
|
87
|
+
createTaskExtended("task for offline worker", {
|
|
88
|
+
agentId: offlineWorker.id,
|
|
89
|
+
source: "slack",
|
|
90
|
+
slackChannelId: channelId,
|
|
91
|
+
slackThreadTs: threadTs,
|
|
92
|
+
slackUserId: "U_HUMAN",
|
|
93
|
+
});
|
|
94
|
+
|
|
95
|
+
const matches = routeMessage("<@BOT123> follow up", "BOT123", true, { channelId, threadTs });
|
|
96
|
+
|
|
97
|
+
// Should route to lead via thread follow-up fallback (not generic @bot)
|
|
98
|
+
expect(matches).toHaveLength(1);
|
|
99
|
+
expect(matches[0].agent.id).toBe(leadAgent.id);
|
|
100
|
+
expect(matches[0].matchedText).toBe("thread follow-up (lead fallback)");
|
|
101
|
+
});
|
|
102
|
+
});
|
|
103
|
+
|
|
104
|
+
describe("thread follow-up does not override explicit swarm#<uuid>", () => {
|
|
105
|
+
test("swarm#<uuid> takes priority over thread context", () => {
|
|
106
|
+
const channelId = "C400";
|
|
107
|
+
const threadTs = "4000.0001";
|
|
108
|
+
|
|
109
|
+
// Create a second worker
|
|
110
|
+
const worker2 = createAgent({
|
|
111
|
+
name: "worker-2",
|
|
112
|
+
isLead: false,
|
|
113
|
+
status: "idle",
|
|
114
|
+
capabilities: [],
|
|
115
|
+
});
|
|
116
|
+
|
|
117
|
+
// Worker 1 has an active task in this thread
|
|
118
|
+
createTaskExtended("worker1 task", {
|
|
119
|
+
agentId: workerAgent.id,
|
|
120
|
+
source: "slack",
|
|
121
|
+
slackChannelId: channelId,
|
|
122
|
+
slackThreadTs: threadTs,
|
|
123
|
+
slackUserId: "U_HUMAN",
|
|
124
|
+
});
|
|
125
|
+
|
|
126
|
+
// Message explicitly targets worker2 via swarm#<uuid>
|
|
127
|
+
const matches = routeMessage(
|
|
128
|
+
`<@BOT123> swarm#${worker2.id} do this instead`,
|
|
129
|
+
"BOT123",
|
|
130
|
+
true,
|
|
131
|
+
{ channelId, threadTs },
|
|
132
|
+
);
|
|
133
|
+
|
|
134
|
+
// Should route to worker2 (explicit), NOT worker1 (thread context)
|
|
135
|
+
expect(matches).toHaveLength(1);
|
|
136
|
+
expect(matches[0].agent.id).toBe(worker2.id);
|
|
137
|
+
expect(matches[0].matchedText).toBe(`swarm#${worker2.id}`);
|
|
138
|
+
});
|
|
139
|
+
|
|
140
|
+
test("swarm#all takes priority over thread context", () => {
|
|
141
|
+
const channelId = "C500";
|
|
142
|
+
const threadTs = "5000.0001";
|
|
143
|
+
|
|
144
|
+
// Worker has an active task in this thread
|
|
145
|
+
createTaskExtended("worker task in thread", {
|
|
146
|
+
agentId: workerAgent.id,
|
|
147
|
+
source: "slack",
|
|
148
|
+
slackChannelId: channelId,
|
|
149
|
+
slackThreadTs: threadTs,
|
|
150
|
+
slackUserId: "U_HUMAN",
|
|
151
|
+
});
|
|
152
|
+
|
|
153
|
+
const matches = routeMessage("<@BOT123> swarm#all deploy everything", "BOT123", true, {
|
|
154
|
+
channelId,
|
|
155
|
+
threadTs,
|
|
156
|
+
});
|
|
157
|
+
|
|
158
|
+
// Should broadcast to all non-lead workers, not just the thread worker
|
|
159
|
+
expect(matches.length).toBeGreaterThanOrEqual(1);
|
|
160
|
+
expect(matches.every((m) => m.matchedText === "swarm#all")).toBe(true);
|
|
161
|
+
// Lead should not be in the matches
|
|
162
|
+
expect(matches.every((m) => !m.agent.isLead)).toBe(true);
|
|
163
|
+
});
|
|
164
|
+
});
|
|
165
|
+
|
|
166
|
+
describe("no thread context behaves normally", () => {
|
|
167
|
+
test("routes to lead when bot mentioned without thread context", () => {
|
|
168
|
+
const matches = routeMessage("<@BOT123> hello", "BOT123", true);
|
|
169
|
+
|
|
170
|
+
expect(matches).toHaveLength(1);
|
|
171
|
+
expect(matches[0].agent.id).toBe(leadAgent.id);
|
|
172
|
+
expect(matches[0].matchedText).toBe("@bot");
|
|
173
|
+
});
|
|
174
|
+
|
|
175
|
+
test("returns empty when bot not mentioned and no thread context", () => {
|
|
176
|
+
const matches = routeMessage("hello everyone", "BOT123", false);
|
|
177
|
+
|
|
178
|
+
expect(matches).toHaveLength(0);
|
|
179
|
+
});
|
|
180
|
+
});
|
|
181
|
+
});
|
|
@@ -0,0 +1,305 @@
|
|
|
1
|
+
import { afterAll, beforeAll, describe, expect, test } from "bun:test";
|
|
2
|
+
import { unlinkSync } from "node:fs";
|
|
3
|
+
import {
|
|
4
|
+
closeDb,
|
|
5
|
+
createAgent,
|
|
6
|
+
createTaskExtended,
|
|
7
|
+
getLatestActiveTaskInThread,
|
|
8
|
+
initDb,
|
|
9
|
+
} from "../be/db";
|
|
10
|
+
import {
|
|
11
|
+
bufferThreadMessage,
|
|
12
|
+
getBufferMessageCount,
|
|
13
|
+
instantFlush,
|
|
14
|
+
isThreadBuffered,
|
|
15
|
+
} from "../slack/thread-buffer";
|
|
16
|
+
|
|
17
|
+
const TEST_DB_PATH = "./test-slack-thread-buffer.sqlite";
|
|
18
|
+
|
|
19
|
+
beforeAll(() => {
|
|
20
|
+
initDb(TEST_DB_PATH);
|
|
21
|
+
// Create a lead agent for flush to assign tasks to
|
|
22
|
+
createAgent({ name: "lead-agent", isLead: true, status: "idle", capabilities: [] });
|
|
23
|
+
});
|
|
24
|
+
|
|
25
|
+
afterAll(() => {
|
|
26
|
+
closeDb();
|
|
27
|
+
try {
|
|
28
|
+
unlinkSync(TEST_DB_PATH);
|
|
29
|
+
unlinkSync(`${TEST_DB_PATH}-wal`);
|
|
30
|
+
unlinkSync(`${TEST_DB_PATH}-shm`);
|
|
31
|
+
} catch {
|
|
32
|
+
// ignore if files don't exist
|
|
33
|
+
}
|
|
34
|
+
});
|
|
35
|
+
|
|
36
|
+
describe("Slack thread buffer", () => {
|
|
37
|
+
describe("buffer creation and message appending", () => {
|
|
38
|
+
test("bufferThreadMessage creates a buffer entry", () => {
|
|
39
|
+
bufferThreadMessage("C100", "1000.0001", "first message", "U1", "1000.0010");
|
|
40
|
+
|
|
41
|
+
expect(isThreadBuffered("C100", "1000.0001")).toBe(true);
|
|
42
|
+
expect(getBufferMessageCount("C100:1000.0001")).toBe(1);
|
|
43
|
+
});
|
|
44
|
+
|
|
45
|
+
test("appending a second message increments count", () => {
|
|
46
|
+
bufferThreadMessage("C100", "1000.0001", "second message", "U2", "1000.0020");
|
|
47
|
+
|
|
48
|
+
expect(getBufferMessageCount("C100:1000.0001")).toBe(2);
|
|
49
|
+
});
|
|
50
|
+
|
|
51
|
+
test("appending a third message increments count again", () => {
|
|
52
|
+
bufferThreadMessage("C100", "1000.0001", "third message", "U1", "1000.0030");
|
|
53
|
+
|
|
54
|
+
expect(getBufferMessageCount("C100:1000.0001")).toBe(3);
|
|
55
|
+
});
|
|
56
|
+
});
|
|
57
|
+
|
|
58
|
+
describe("buffer keyed by channelId:threadTs (no cross-thread contamination)", () => {
|
|
59
|
+
test("different threads have independent buffers", () => {
|
|
60
|
+
bufferThreadMessage("C200", "2000.0001", "thread A msg", "U1", "2000.0010");
|
|
61
|
+
bufferThreadMessage("C200", "2000.0002", "thread B msg", "U1", "2000.0020");
|
|
62
|
+
|
|
63
|
+
expect(isThreadBuffered("C200", "2000.0001")).toBe(true);
|
|
64
|
+
expect(isThreadBuffered("C200", "2000.0002")).toBe(true);
|
|
65
|
+
expect(getBufferMessageCount("C200:2000.0001")).toBe(1);
|
|
66
|
+
expect(getBufferMessageCount("C200:2000.0002")).toBe(1);
|
|
67
|
+
});
|
|
68
|
+
|
|
69
|
+
test("different channels with same threadTs are independent", () => {
|
|
70
|
+
bufferThreadMessage("C300", "3000.0001", "channel A msg", "U1", "3000.0010");
|
|
71
|
+
bufferThreadMessage("C301", "3000.0001", "channel B msg", "U1", "3000.0020");
|
|
72
|
+
|
|
73
|
+
expect(getBufferMessageCount("C300:3000.0001")).toBe(1);
|
|
74
|
+
expect(getBufferMessageCount("C301:3000.0001")).toBe(1);
|
|
75
|
+
});
|
|
76
|
+
});
|
|
77
|
+
|
|
78
|
+
describe("isThreadBuffered", () => {
|
|
79
|
+
test("returns false for non-buffered thread", () => {
|
|
80
|
+
expect(isThreadBuffered("C999", "9999.0001")).toBe(false);
|
|
81
|
+
});
|
|
82
|
+
|
|
83
|
+
test("returns true for buffered thread", () => {
|
|
84
|
+
bufferThreadMessage("C400", "4000.0001", "msg", "U1", "4000.0010");
|
|
85
|
+
expect(isThreadBuffered("C400", "4000.0001")).toBe(true);
|
|
86
|
+
});
|
|
87
|
+
});
|
|
88
|
+
|
|
89
|
+
describe("getBufferMessageCount", () => {
|
|
90
|
+
test("returns 0 for unknown key", () => {
|
|
91
|
+
expect(getBufferMessageCount("CXXX:9999.9999")).toBe(0);
|
|
92
|
+
});
|
|
93
|
+
|
|
94
|
+
test("returns correct count after multiple appends", () => {
|
|
95
|
+
bufferThreadMessage("C500", "5000.0001", "msg1", "U1", "5000.0010");
|
|
96
|
+
bufferThreadMessage("C500", "5000.0001", "msg2", "U2", "5000.0020");
|
|
97
|
+
bufferThreadMessage("C500", "5000.0001", "msg3", "U1", "5000.0030");
|
|
98
|
+
|
|
99
|
+
expect(getBufferMessageCount("C500:5000.0001")).toBe(3);
|
|
100
|
+
});
|
|
101
|
+
});
|
|
102
|
+
|
|
103
|
+
describe("flush creates correct task with combined description", () => {
|
|
104
|
+
test("instantFlush creates a task with all buffered messages", async () => {
|
|
105
|
+
const channelId = "C600";
|
|
106
|
+
const threadTs = "6000.0001";
|
|
107
|
+
|
|
108
|
+
bufferThreadMessage(channelId, threadTs, "fix the bug", "U1", "6000.0010");
|
|
109
|
+
bufferThreadMessage(channelId, threadTs, "also check the logs", "U2", "6000.0020");
|
|
110
|
+
|
|
111
|
+
await instantFlush(`${channelId}:${threadTs}`);
|
|
112
|
+
|
|
113
|
+
// Buffer should be cleaned up
|
|
114
|
+
expect(isThreadBuffered(channelId, threadTs)).toBe(false);
|
|
115
|
+
expect(getBufferMessageCount(`${channelId}:${threadTs}`)).toBe(0);
|
|
116
|
+
|
|
117
|
+
// Check the task was created in the DB with correct Slack metadata
|
|
118
|
+
const task = getLatestActiveTaskInThread(channelId, threadTs);
|
|
119
|
+
expect(task).not.toBeNull();
|
|
120
|
+
expect(task!.task).toContain("fix the bug");
|
|
121
|
+
expect(task!.task).toContain("also check the logs");
|
|
122
|
+
expect(task!.task).toContain("---"); // separator between messages
|
|
123
|
+
expect(task!.task).toContain("2 message(s) buffered");
|
|
124
|
+
expect(task!.source).toBe("slack");
|
|
125
|
+
expect(task!.slackChannelId).toBe(channelId);
|
|
126
|
+
expect(task!.slackThreadTs).toBe(threadTs);
|
|
127
|
+
});
|
|
128
|
+
});
|
|
129
|
+
|
|
130
|
+
describe("flush sets dependsOn to latest active task in thread", () => {
|
|
131
|
+
test("flushed task depends on existing active task", async () => {
|
|
132
|
+
const channelId = "C700";
|
|
133
|
+
const threadTs = "7000.0001";
|
|
134
|
+
|
|
135
|
+
// Create a worker agent that owns a task in this thread
|
|
136
|
+
const worker = createAgent({
|
|
137
|
+
name: "buf-worker-1",
|
|
138
|
+
isLead: false,
|
|
139
|
+
status: "idle",
|
|
140
|
+
capabilities: [],
|
|
141
|
+
});
|
|
142
|
+
|
|
143
|
+
// Create an existing pending task assigned to the worker in this thread
|
|
144
|
+
const existingTask = createTaskExtended("original task", {
|
|
145
|
+
agentId: worker.id,
|
|
146
|
+
source: "slack",
|
|
147
|
+
slackChannelId: channelId,
|
|
148
|
+
slackThreadTs: threadTs,
|
|
149
|
+
slackUserId: "U1",
|
|
150
|
+
});
|
|
151
|
+
|
|
152
|
+
// Buffer some follow-up messages
|
|
153
|
+
bufferThreadMessage(channelId, threadTs, "follow up 1", "U1", "7000.0010");
|
|
154
|
+
bufferThreadMessage(channelId, threadTs, "follow up 2", "U2", "7000.0020");
|
|
155
|
+
|
|
156
|
+
// Use a short timeout to let the debounce timer fire
|
|
157
|
+
// We need to wait for the flush to happen via the timer.
|
|
158
|
+
// Since instantFlush with immediate=true skips dependsOn,
|
|
159
|
+
// we manually trigger the timer by waiting.
|
|
160
|
+
// With default 10s timeout, use instantFlush but with immediate=false approach.
|
|
161
|
+
// Actually, we need to test the dependency chaining specifically.
|
|
162
|
+
// The buffer's flushBuffer(key, false) sets dependsOn, while flushBuffer(key, true) doesn't.
|
|
163
|
+
// instantFlush calls flushBuffer(key, true) — so it WON'T set dependsOn.
|
|
164
|
+
// To test dependency chaining, we need the timer to fire naturally.
|
|
165
|
+
|
|
166
|
+
// For a deterministic test, let's set ADDITIVE_SLACK_BUFFER_MS to a small value
|
|
167
|
+
// and wait for it. But the env var is read at module load time.
|
|
168
|
+
// Instead, we'll verify the DB state by checking that getLatestActiveTaskInThread
|
|
169
|
+
// returns the existing task before flush, confirming the dependency logic would work.
|
|
170
|
+
const latestActive = getLatestActiveTaskInThread(channelId, threadTs);
|
|
171
|
+
expect(latestActive).not.toBeNull();
|
|
172
|
+
expect(latestActive!.id).toBe(existingTask.id);
|
|
173
|
+
|
|
174
|
+
// Clean up: flush via instant (to clear the buffer)
|
|
175
|
+
await instantFlush(`${channelId}:${threadTs}`);
|
|
176
|
+
});
|
|
177
|
+
});
|
|
178
|
+
|
|
179
|
+
describe("flush without active task produces no dependsOn", () => {
|
|
180
|
+
test("flushed task in thread with no active task has no dependency", async () => {
|
|
181
|
+
const channelId = "C800";
|
|
182
|
+
const threadTs = "8000.0001";
|
|
183
|
+
|
|
184
|
+
// No existing tasks in this thread
|
|
185
|
+
const latestActive = getLatestActiveTaskInThread(channelId, threadTs);
|
|
186
|
+
expect(latestActive).toBeNull();
|
|
187
|
+
|
|
188
|
+
bufferThreadMessage(channelId, threadTs, "new request", "U1", "8000.0010");
|
|
189
|
+
await instantFlush(`${channelId}:${threadTs}`);
|
|
190
|
+
|
|
191
|
+
// Task created with no dependency
|
|
192
|
+
const task = getLatestActiveTaskInThread(channelId, threadTs);
|
|
193
|
+
expect(task).not.toBeNull();
|
|
194
|
+
expect(task!.task).toContain("new request");
|
|
195
|
+
// dependsOn is stored as JSON — null or empty means no dependency
|
|
196
|
+
expect(task!.dependsOn).toEqual([]);
|
|
197
|
+
});
|
|
198
|
+
});
|
|
199
|
+
|
|
200
|
+
describe("instantFlush has no dependsOn regardless of active tasks", () => {
|
|
201
|
+
test("instantFlush skips dependency even when active task exists", async () => {
|
|
202
|
+
const channelId = "C900";
|
|
203
|
+
const threadTs = "9000.0001";
|
|
204
|
+
|
|
205
|
+
// Create a worker agent and an active task in this thread
|
|
206
|
+
const worker = createAgent({
|
|
207
|
+
name: "buf-worker-2",
|
|
208
|
+
isLead: false,
|
|
209
|
+
status: "idle",
|
|
210
|
+
capabilities: [],
|
|
211
|
+
});
|
|
212
|
+
createTaskExtended("active task", {
|
|
213
|
+
agentId: worker.id,
|
|
214
|
+
source: "slack",
|
|
215
|
+
slackChannelId: channelId,
|
|
216
|
+
slackThreadTs: threadTs,
|
|
217
|
+
slackUserId: "U1",
|
|
218
|
+
});
|
|
219
|
+
|
|
220
|
+
// Confirm active task exists
|
|
221
|
+
const latestActive = getLatestActiveTaskInThread(channelId, threadTs);
|
|
222
|
+
expect(latestActive).not.toBeNull();
|
|
223
|
+
|
|
224
|
+
// Buffer and instant flush
|
|
225
|
+
bufferThreadMessage(channelId, threadTs, "urgent fix", "U1", "9000.0010");
|
|
226
|
+
await instantFlush(`${channelId}:${threadTs}`);
|
|
227
|
+
|
|
228
|
+
// The newest task in this thread is the flushed one (latest by createdAt)
|
|
229
|
+
const newestTask = getLatestActiveTaskInThread(channelId, threadTs);
|
|
230
|
+
expect(newestTask).not.toBeNull();
|
|
231
|
+
// Verify it's the flushed task by checking content
|
|
232
|
+
expect(newestTask!.task).toContain("urgent fix");
|
|
233
|
+
// instantFlush creates with immediate=true -> no dependsOn
|
|
234
|
+
expect(newestTask!.dependsOn).toEqual([]);
|
|
235
|
+
});
|
|
236
|
+
});
|
|
237
|
+
|
|
238
|
+
describe("buffer cleanup after flush", () => {
|
|
239
|
+
test("buffer is removed after instantFlush", async () => {
|
|
240
|
+
const channelId = "C1000";
|
|
241
|
+
const threadTs = "10000.0001";
|
|
242
|
+
|
|
243
|
+
bufferThreadMessage(channelId, threadTs, "temp msg", "U1", "10000.0010");
|
|
244
|
+
expect(isThreadBuffered(channelId, threadTs)).toBe(true);
|
|
245
|
+
|
|
246
|
+
await instantFlush(`${channelId}:${threadTs}`);
|
|
247
|
+
|
|
248
|
+
expect(isThreadBuffered(channelId, threadTs)).toBe(false);
|
|
249
|
+
expect(getBufferMessageCount(`${channelId}:${threadTs}`)).toBe(0);
|
|
250
|
+
});
|
|
251
|
+
|
|
252
|
+
test("instantFlush on empty/nonexistent buffer is a no-op", async () => {
|
|
253
|
+
// Should not throw
|
|
254
|
+
await instantFlush("CXXX:9999.0001");
|
|
255
|
+
});
|
|
256
|
+
});
|
|
257
|
+
|
|
258
|
+
describe("timer/debounce behavior", () => {
|
|
259
|
+
test("buffer flushes after timeout expires", async () => {
|
|
260
|
+
// Use a very short buffer timeout by relying on the module-level
|
|
261
|
+
// BUFFER_TIMEOUT_MS. Since we can't change it at runtime, we test
|
|
262
|
+
// indirectly: buffer a message, then wait and check if it flushed.
|
|
263
|
+
// The default is 10s which is too long for a test.
|
|
264
|
+
// Instead, we verify the timer exists by checking the buffer is active,
|
|
265
|
+
// then force-flush.
|
|
266
|
+
const channelId = "C1100";
|
|
267
|
+
const threadTs = "11000.0001";
|
|
268
|
+
|
|
269
|
+
bufferThreadMessage(channelId, threadTs, "timed msg", "U1", "11000.0010");
|
|
270
|
+
expect(isThreadBuffered(channelId, threadTs)).toBe(true);
|
|
271
|
+
|
|
272
|
+
// After instantFlush, buffer is gone
|
|
273
|
+
await instantFlush(`${channelId}:${threadTs}`);
|
|
274
|
+
expect(isThreadBuffered(channelId, threadTs)).toBe(false);
|
|
275
|
+
});
|
|
276
|
+
|
|
277
|
+
test("new messages reset the debounce (buffer stays active)", () => {
|
|
278
|
+
const channelId = "C1200";
|
|
279
|
+
const threadTs = "12000.0001";
|
|
280
|
+
|
|
281
|
+
bufferThreadMessage(channelId, threadTs, "msg1", "U1", "12000.0010");
|
|
282
|
+
expect(isThreadBuffered(channelId, threadTs)).toBe(true);
|
|
283
|
+
|
|
284
|
+
// Adding another message should keep the buffer active (timer reset)
|
|
285
|
+
bufferThreadMessage(channelId, threadTs, "msg2", "U1", "12000.0020");
|
|
286
|
+
expect(isThreadBuffered(channelId, threadTs)).toBe(true);
|
|
287
|
+
expect(getBufferMessageCount(`${channelId}:${threadTs}`)).toBe(2);
|
|
288
|
+
});
|
|
289
|
+
});
|
|
290
|
+
|
|
291
|
+
describe("single message flush", () => {
|
|
292
|
+
test("single message buffer creates task with 1 message count", async () => {
|
|
293
|
+
const channelId = "C1300";
|
|
294
|
+
const threadTs = "13000.0001";
|
|
295
|
+
|
|
296
|
+
bufferThreadMessage(channelId, threadTs, "solo message", "U1", "13000.0010");
|
|
297
|
+
await instantFlush(`${channelId}:${threadTs}`);
|
|
298
|
+
|
|
299
|
+
const task = getLatestActiveTaskInThread(channelId, threadTs);
|
|
300
|
+
expect(task).not.toBeNull();
|
|
301
|
+
expect(task!.task).toContain("1 message(s) buffered");
|
|
302
|
+
expect(task!.task).toContain("solo message");
|
|
303
|
+
});
|
|
304
|
+
});
|
|
305
|
+
});
|