@mclawnet/swarm 0.1.3 → 0.1.5
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 +118 -0
- package/dist/__tests__/action-parser.test.js +29 -82
- package/dist/__tests__/action-parser.test.js.map +1 -1
- package/dist/__tests__/coordinator-create-tx.test.d.ts +2 -0
- package/dist/__tests__/coordinator-create-tx.test.d.ts.map +1 -0
- package/dist/__tests__/coordinator-create-tx.test.js +114 -0
- package/dist/__tests__/coordinator-create-tx.test.js.map +1 -0
- package/dist/__tests__/coordinator-inbox-migration.test.d.ts +2 -0
- package/dist/__tests__/coordinator-inbox-migration.test.d.ts.map +1 -0
- package/dist/__tests__/coordinator-inbox-migration.test.js +56 -0
- package/dist/__tests__/coordinator-inbox-migration.test.js.map +1 -0
- package/dist/__tests__/inbox-integration.test.d.ts +2 -0
- package/dist/__tests__/inbox-integration.test.d.ts.map +1 -0
- package/dist/__tests__/inbox-integration.test.js +120 -0
- package/dist/__tests__/inbox-integration.test.js.map +1 -0
- package/dist/__tests__/inbox-persistence-recovery.test.d.ts +2 -0
- package/dist/__tests__/inbox-persistence-recovery.test.d.ts.map +1 -0
- package/dist/__tests__/inbox-persistence-recovery.test.js +139 -0
- package/dist/__tests__/inbox-persistence-recovery.test.js.map +1 -0
- package/dist/__tests__/inbox-relay-interceptor.test.d.ts +2 -0
- package/dist/__tests__/inbox-relay-interceptor.test.d.ts.map +1 -0
- package/dist/__tests__/inbox-relay-interceptor.test.js +156 -0
- package/dist/__tests__/inbox-relay-interceptor.test.js.map +1 -0
- package/dist/__tests__/inbox-relay.test.d.ts +2 -0
- package/dist/__tests__/inbox-relay.test.d.ts.map +1 -0
- package/dist/__tests__/inbox-relay.test.js +318 -0
- package/dist/__tests__/inbox-relay.test.js.map +1 -0
- package/dist/__tests__/inbox-store.test.d.ts +2 -0
- package/dist/__tests__/inbox-store.test.d.ts.map +1 -0
- package/dist/__tests__/inbox-store.test.js +129 -0
- package/dist/__tests__/inbox-store.test.js.map +1 -0
- package/dist/__tests__/inbox-watcher.test.d.ts +2 -0
- package/dist/__tests__/inbox-watcher.test.d.ts.map +1 -0
- package/dist/__tests__/inbox-watcher.test.js +104 -0
- package/dist/__tests__/inbox-watcher.test.js.map +1 -0
- package/dist/__tests__/persistence-path.test.d.ts +2 -0
- package/dist/__tests__/persistence-path.test.d.ts.map +1 -0
- package/dist/__tests__/persistence-path.test.js +79 -0
- package/dist/__tests__/persistence-path.test.js.map +1 -0
- package/dist/__tests__/persistence-robust.test.d.ts +2 -0
- package/dist/__tests__/persistence-robust.test.d.ts.map +1 -0
- package/dist/__tests__/persistence-robust.test.js +125 -0
- package/dist/__tests__/persistence-robust.test.js.map +1 -0
- package/dist/__tests__/persistence.test.d.ts +2 -0
- package/dist/__tests__/persistence.test.d.ts.map +1 -0
- package/dist/__tests__/persistence.test.js +105 -0
- package/dist/__tests__/persistence.test.js.map +1 -0
- package/dist/__tests__/phase4-5-e2e.test.d.ts +2 -0
- package/dist/__tests__/phase4-5-e2e.test.d.ts.map +1 -0
- package/dist/__tests__/phase4-5-e2e.test.js +203 -0
- package/dist/__tests__/phase4-5-e2e.test.js.map +1 -0
- package/dist/__tests__/phase6-7-e2e.test.d.ts +2 -0
- package/dist/__tests__/phase6-7-e2e.test.d.ts.map +1 -0
- package/dist/__tests__/phase6-7-e2e.test.js +93 -0
- package/dist/__tests__/phase6-7-e2e.test.js.map +1 -0
- package/dist/__tests__/recovery-cross-project.test.d.ts +2 -0
- package/dist/__tests__/recovery-cross-project.test.d.ts.map +1 -0
- package/dist/__tests__/recovery-cross-project.test.js +87 -0
- package/dist/__tests__/recovery-cross-project.test.js.map +1 -0
- package/dist/__tests__/recovery-forwards-to-coordinator.test.d.ts +2 -0
- package/dist/__tests__/recovery-forwards-to-coordinator.test.d.ts.map +1 -0
- package/dist/__tests__/recovery-forwards-to-coordinator.test.js +59 -0
- package/dist/__tests__/recovery-forwards-to-coordinator.test.js.map +1 -0
- package/dist/__tests__/recovery-resume.test.d.ts +2 -0
- package/dist/__tests__/recovery-resume.test.d.ts.map +1 -0
- package/dist/__tests__/recovery-resume.test.js +132 -0
- package/dist/__tests__/recovery-resume.test.js.map +1 -0
- package/dist/__tests__/retrospective.test.js +1 -0
- package/dist/__tests__/retrospective.test.js.map +1 -1
- package/dist/__tests__/role-loader-preamble-all.test.d.ts +2 -0
- package/dist/__tests__/role-loader-preamble-all.test.d.ts.map +1 -0
- package/dist/__tests__/role-loader-preamble-all.test.js +38 -0
- package/dist/__tests__/role-loader-preamble-all.test.js.map +1 -0
- package/dist/__tests__/role-loader-tools.test.d.ts +2 -0
- package/dist/__tests__/role-loader-tools.test.d.ts.map +1 -0
- package/dist/__tests__/role-loader-tools.test.js +39 -0
- package/dist/__tests__/role-loader-tools.test.js.map +1 -0
- package/dist/__tests__/role-loader.test.js +116 -1
- package/dist/__tests__/role-loader.test.js.map +1 -1
- package/dist/__tests__/role-prompt-no-legacy-protocol.test.d.ts +2 -0
- package/dist/__tests__/role-prompt-no-legacy-protocol.test.d.ts.map +1 -0
- package/dist/__tests__/role-prompt-no-legacy-protocol.test.js +37 -0
- package/dist/__tests__/role-prompt-no-legacy-protocol.test.js.map +1 -0
- package/dist/__tests__/role-tools.test.d.ts +2 -0
- package/dist/__tests__/role-tools.test.d.ts.map +1 -0
- package/dist/__tests__/role-tools.test.js +80 -0
- package/dist/__tests__/role-tools.test.js.map +1 -0
- package/dist/__tests__/spawn-role-injects-briefings.test.d.ts +2 -0
- package/dist/__tests__/spawn-role-injects-briefings.test.d.ts.map +1 -0
- package/dist/__tests__/spawn-role-injects-briefings.test.js +182 -0
- package/dist/__tests__/spawn-role-injects-briefings.test.js.map +1 -0
- package/dist/__tests__/spawn-role-tool-policy.test.d.ts +2 -0
- package/dist/__tests__/spawn-role-tool-policy.test.d.ts.map +1 -0
- package/dist/__tests__/spawn-role-tool-policy.test.js +96 -0
- package/dist/__tests__/spawn-role-tool-policy.test.js.map +1 -0
- package/dist/__tests__/swarm-coordinator-inbox-watcher.test.d.ts +2 -0
- package/dist/__tests__/swarm-coordinator-inbox-watcher.test.d.ts.map +1 -0
- package/dist/__tests__/swarm-coordinator-inbox-watcher.test.js +61 -0
- package/dist/__tests__/swarm-coordinator-inbox-watcher.test.js.map +1 -0
- package/dist/__tests__/swarm-coordinator-inbox.test.d.ts +2 -0
- package/dist/__tests__/swarm-coordinator-inbox.test.d.ts.map +1 -0
- package/dist/__tests__/swarm-coordinator-inbox.test.js +182 -0
- package/dist/__tests__/swarm-coordinator-inbox.test.js.map +1 -0
- package/dist/__tests__/swarm-coordinator-init.test.js +36 -8
- package/dist/__tests__/swarm-coordinator-init.test.js.map +1 -1
- package/dist/__tests__/swarm-coordinator-legacy-plan-review-warn.test.d.ts +2 -0
- package/dist/__tests__/swarm-coordinator-legacy-plan-review-warn.test.d.ts.map +1 -0
- package/dist/__tests__/swarm-coordinator-legacy-plan-review-warn.test.js +113 -0
- package/dist/__tests__/swarm-coordinator-legacy-plan-review-warn.test.js.map +1 -0
- package/dist/__tests__/swarm-coordinator-plan-review-intercept.test.d.ts +2 -0
- package/dist/__tests__/swarm-coordinator-plan-review-intercept.test.d.ts.map +1 -0
- package/dist/__tests__/swarm-coordinator-plan-review-intercept.test.js +465 -0
- package/dist/__tests__/swarm-coordinator-plan-review-intercept.test.js.map +1 -0
- package/dist/__tests__/swarm-coordinator-plan-review-recovery.test.d.ts +2 -0
- package/dist/__tests__/swarm-coordinator-plan-review-recovery.test.d.ts.map +1 -0
- package/dist/__tests__/swarm-coordinator-plan-review-recovery.test.js +284 -0
- package/dist/__tests__/swarm-coordinator-plan-review-recovery.test.js.map +1 -0
- package/dist/__tests__/swarm-coordinator-plan-review.test.d.ts +2 -0
- package/dist/__tests__/swarm-coordinator-plan-review.test.d.ts.map +1 -0
- package/dist/__tests__/swarm-coordinator-plan-review.test.js +294 -0
- package/dist/__tests__/swarm-coordinator-plan-review.test.js.map +1 -0
- package/dist/__tests__/swarm-coordinator-resume.test.d.ts +2 -0
- package/dist/__tests__/swarm-coordinator-resume.test.d.ts.map +1 -0
- package/dist/__tests__/swarm-coordinator-resume.test.js +93 -0
- package/dist/__tests__/swarm-coordinator-resume.test.js.map +1 -0
- package/dist/__tests__/swarm-coordinator-roleId.test.js +2 -2
- package/dist/__tests__/swarm-coordinator-roleId.test.js.map +1 -1
- package/dist/__tests__/swarm-destroy-detach.test.d.ts +2 -0
- package/dist/__tests__/swarm-destroy-detach.test.d.ts.map +1 -0
- package/dist/__tests__/swarm-destroy-detach.test.js +135 -0
- package/dist/__tests__/swarm-destroy-detach.test.js.map +1 -0
- package/dist/action-parser.d.ts +0 -9
- package/dist/action-parser.d.ts.map +1 -1
- package/dist/action-parser.js +0 -114
- package/dist/action-parser.js.map +1 -1
- package/dist/inbox-relay.d.ts +50 -0
- package/dist/inbox-relay.d.ts.map +1 -0
- package/dist/inbox-relay.js +168 -0
- package/dist/inbox-relay.js.map +1 -0
- package/dist/inbox-store.d.ts +25 -0
- package/dist/inbox-store.d.ts.map +1 -0
- package/dist/inbox-store.js +95 -0
- package/dist/inbox-store.js.map +1 -0
- package/dist/inbox-watcher.d.ts +13 -0
- package/dist/inbox-watcher.d.ts.map +1 -0
- package/dist/inbox-watcher.js +89 -0
- package/dist/inbox-watcher.js.map +1 -0
- package/dist/index.d.ts +4 -3
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +3 -2
- package/dist/index.js.map +1 -1
- package/dist/persistence.d.ts +19 -5
- package/dist/persistence.d.ts.map +1 -1
- package/dist/persistence.js +97 -22
- package/dist/persistence.js.map +1 -1
- package/dist/recovery.d.ts +12 -0
- package/dist/recovery.d.ts.map +1 -1
- package/dist/recovery.js +14 -19
- package/dist/recovery.js.map +1 -1
- package/dist/roles/role-loader.d.ts +28 -1
- package/dist/roles/role-loader.d.ts.map +1 -1
- package/dist/roles/role-loader.js +73 -1
- package/dist/roles/role-loader.js.map +1 -1
- package/dist/roles/role-tools.d.ts +16 -0
- package/dist/roles/role-tools.d.ts.map +1 -0
- package/dist/roles/role-tools.js +25 -0
- package/dist/roles/role-tools.js.map +1 -0
- package/dist/roles/types.d.ts +4 -0
- package/dist/roles/types.d.ts.map +1 -1
- package/dist/swarm-coordinator.d.ts +176 -12
- package/dist/swarm-coordinator.d.ts.map +1 -1
- package/dist/swarm-coordinator.js +863 -370
- package/dist/swarm-coordinator.js.map +1 -1
- package/dist/types.d.ts +26 -0
- package/dist/types.d.ts.map +1 -1
- package/package.json +9 -6
- package/roles/analyst-livermore.md +6 -30
- package/roles/designer-rams.md +2 -30
- package/roles/dev-torvalds.md +8 -44
- package/roles/developer.md +5 -21
- package/roles/director-jia.md +20 -49
- package/roles/editor-boyong.md +8 -40
- package/roles/macro-dalio.md +6 -30
- package/roles/planner-maoni.md +24 -53
- package/roles/pm-jobs.md +20 -71
- package/roles/preset-analyst-simons.md +2 -18
- package/roles/preset-architect-knuth.md +2 -18
- package/roles/preset-designer-norman.md +2 -18
- package/roles/preset-designer.md +2 -18
- package/roles/preset-dev-carmack.md +2 -18
- package/roles/preset-dev-gosling.md +2 -18
- package/roles/preset-developer.md +7 -23
- package/roles/preset-manager-grove.md +2 -18
- package/roles/preset-manager-musk.md +2 -18
- package/roles/preset-pm.md +7 -34
- package/roles/preset-researcher-feynman.md +2 -18
- package/roles/preset-reviewer.md +5 -21
- package/roles/preset-strategist-buffett.md +2 -18
- package/roles/preset-strategist-munger.md +2 -18
- package/roles/preset-strategist-sunzi.md +2 -18
- package/roles/preset-tester-beck.md +2 -18
- package/roles/preset-tester.md +5 -21
- package/roles/preset-writer-orwell.md +2 -18
- package/roles/preset-writer.md +2 -18
- package/roles/quant-simons.md +5 -32
- package/roles/queen.md +25 -41
- package/roles/reviewer-martin.md +11 -37
- package/roles/reviewer.md +20 -21
- package/roles/rhythm-tangsan.md +5 -29
- package/roles/risk-taleb.md +4 -32
- package/roles/script-shitiesheng.md +8 -31
- package/roles/storyboard-xuke.md +9 -29
- package/roles/strategist-soros.md +16 -73
- package/roles/tester-beck.md +4 -40
- package/roles/tester.md +5 -21
- package/roles/trader-jones.md +4 -32
- package/roles/vfx-guchangwei.md +8 -27
- package/roles/writer-zhouzi.md +7 -39
- package/templates/dev-team-pro.md +4 -1
- package/templates/dev-team.md +3 -1
- package/templates/minimal.md +2 -1
- package/templates/trading-team.md +6 -1
- package/templates/video-team.md +4 -1
- package/templates/writing-team.md +4 -1
|
@@ -0,0 +1,182 @@
|
|
|
1
|
+
import { describe, it, expect, vi, beforeEach, afterEach } from "vitest";
|
|
2
|
+
import { mkdtempSync, rmSync } from "node:fs";
|
|
3
|
+
import { tmpdir } from "node:os";
|
|
4
|
+
import { join } from "node:path";
|
|
5
|
+
// Mock logger
|
|
6
|
+
vi.mock("@mclawnet/logger", () => ({
|
|
7
|
+
createLogger: () => ({
|
|
8
|
+
info: vi.fn(),
|
|
9
|
+
warn: vi.fn(),
|
|
10
|
+
error: vi.fn(),
|
|
11
|
+
debug: vi.fn(),
|
|
12
|
+
}),
|
|
13
|
+
}));
|
|
14
|
+
// Mock persistence
|
|
15
|
+
vi.mock("../persistence.js", () => ({
|
|
16
|
+
saveSwarmSnapshot: vi.fn(),
|
|
17
|
+
deleteSwarmSnapshot: vi.fn(),
|
|
18
|
+
appendMessageLog: vi.fn(),
|
|
19
|
+
loadSwarmSnapshot: vi.fn(),
|
|
20
|
+
readMessageLog: vi.fn(() => []),
|
|
21
|
+
}));
|
|
22
|
+
vi.mock("../retrospective.js", () => ({
|
|
23
|
+
runRetrospective: vi.fn(async () => ({})),
|
|
24
|
+
}));
|
|
25
|
+
import { SwarmCoordinator } from "../swarm-coordinator.js";
|
|
26
|
+
import { InboxStore } from "../inbox-store.js";
|
|
27
|
+
function createMockSessionAdapter() {
|
|
28
|
+
const calls = [];
|
|
29
|
+
const inputs = [];
|
|
30
|
+
const adapter = {
|
|
31
|
+
_calls: calls,
|
|
32
|
+
_inputs: inputs,
|
|
33
|
+
createSession: vi.fn(async (options) => {
|
|
34
|
+
calls.push(options);
|
|
35
|
+
return `proc-${options.sessionId}`;
|
|
36
|
+
}),
|
|
37
|
+
sendInput: vi.fn((sessionId, input) => {
|
|
38
|
+
inputs.push({ sessionId, input });
|
|
39
|
+
}),
|
|
40
|
+
closeSession: vi.fn(async () => { }),
|
|
41
|
+
};
|
|
42
|
+
return adapter;
|
|
43
|
+
}
|
|
44
|
+
function createMockHubAdapter() {
|
|
45
|
+
const msgs = [];
|
|
46
|
+
const adapter = {
|
|
47
|
+
_msgs: msgs,
|
|
48
|
+
send: vi.fn((msg) => msgs.push(msg)),
|
|
49
|
+
};
|
|
50
|
+
return adapter;
|
|
51
|
+
}
|
|
52
|
+
const baseMsg = {
|
|
53
|
+
from: "queen",
|
|
54
|
+
type: "task",
|
|
55
|
+
data: "do work",
|
|
56
|
+
timestamp: 1,
|
|
57
|
+
delivered: false,
|
|
58
|
+
};
|
|
59
|
+
let HOME;
|
|
60
|
+
let WORK;
|
|
61
|
+
beforeEach(() => {
|
|
62
|
+
HOME = mkdtempSync(join(tmpdir(), "swarm-inbox-"));
|
|
63
|
+
WORK = mkdtempSync(join(tmpdir(), "swarm-inbox-work-"));
|
|
64
|
+
process.env.CLAWNET_HOME = HOME;
|
|
65
|
+
});
|
|
66
|
+
afterEach(() => {
|
|
67
|
+
rmSync(HOME, { recursive: true, force: true });
|
|
68
|
+
rmSync(WORK, { recursive: true, force: true });
|
|
69
|
+
delete process.env.CLAWNET_HOME;
|
|
70
|
+
});
|
|
71
|
+
describe("SwarmCoordinator wires InboxRelay", () => {
|
|
72
|
+
it("after spawnRole, undelivered inbox messages are pushed via sendInput with <info_for_agent> wrap", async () => {
|
|
73
|
+
const sessionAdapter = createMockSessionAdapter();
|
|
74
|
+
const hubAdapter = createMockHubAdapter();
|
|
75
|
+
const coordinator = new SwarmCoordinator(sessionAdapter, hubAdapter);
|
|
76
|
+
const swarmId = "sw-spawn-deliver";
|
|
77
|
+
// Pre-populate inbox for queen-0 BEFORE create() so deliver-after-spawn finds it.
|
|
78
|
+
// Use the InboxStore with same workDir/swarmId so projectRoot resolves identically.
|
|
79
|
+
const store = new InboxStore(WORK, swarmId);
|
|
80
|
+
await store.append("queen-0", { ...baseMsg, id: "preload-1", data: "preloaded task" });
|
|
81
|
+
await coordinator.create(swarmId, {
|
|
82
|
+
workDir: WORK,
|
|
83
|
+
roles: [{ roleName: "queen", count: 1, eager: true }],
|
|
84
|
+
task: "Outer task",
|
|
85
|
+
});
|
|
86
|
+
// Wait for fire-and-forget deliver to flush.
|
|
87
|
+
await new Promise((r) => setTimeout(r, 50));
|
|
88
|
+
const queenInbox = sessionAdapter._inputs.filter((i) => i.sessionId.includes("queen-") && i.input.includes("<info_for_agent>"));
|
|
89
|
+
expect(queenInbox.length).toBeGreaterThanOrEqual(1);
|
|
90
|
+
expect(queenInbox[0].input).toContain('id="preload-1"');
|
|
91
|
+
expect(queenInbox[0].input).toContain("preloaded task");
|
|
92
|
+
});
|
|
93
|
+
it("handleRoleTurnComplete settles pending echoes (markDelivered + re-deliver new msgs)", async () => {
|
|
94
|
+
const sessionAdapter = createMockSessionAdapter();
|
|
95
|
+
const hubAdapter = createMockHubAdapter();
|
|
96
|
+
const coordinator = new SwarmCoordinator(sessionAdapter, hubAdapter);
|
|
97
|
+
const swarmId = "sw-turn-settle";
|
|
98
|
+
const store = new InboxStore(WORK, swarmId);
|
|
99
|
+
await store.append("queen-0", { ...baseMsg, id: "m1" });
|
|
100
|
+
await coordinator.create(swarmId, {
|
|
101
|
+
workDir: WORK,
|
|
102
|
+
roles: [{ roleName: "queen", count: 1, eager: true }],
|
|
103
|
+
});
|
|
104
|
+
await new Promise((r) => setTimeout(r, 50));
|
|
105
|
+
// The first deliver should have pushed at least m1 (init-complete may
|
|
106
|
+
// also be appended by create() now that it routes through the inbox).
|
|
107
|
+
const beforeSettle = sessionAdapter._inputs.filter((i) => i.input.includes("<info_for_agent>"));
|
|
108
|
+
expect(beforeSettle.length).toBeGreaterThanOrEqual(1);
|
|
109
|
+
const beforeCount = beforeSettle.length;
|
|
110
|
+
// Append a new message during the "turn".
|
|
111
|
+
await store.append("queen-0", { ...baseMsg, id: "m2", data: "later" });
|
|
112
|
+
// Trigger turn-complete -> onAgentTurnSettled.
|
|
113
|
+
const handled = coordinator.handleRoleTurnComplete(`${swarmId}::queen-0`, {});
|
|
114
|
+
expect(handled).toBe(true);
|
|
115
|
+
await new Promise((r) => setTimeout(r, 60));
|
|
116
|
+
// m1 should now be marked delivered.
|
|
117
|
+
const all = await store.readAll("queen-0");
|
|
118
|
+
const m1 = all.find((m) => m.id === "m1");
|
|
119
|
+
expect(m1.delivered).toBe(true);
|
|
120
|
+
// m2 should have been pushed in a follow-up envelope after the settle.
|
|
121
|
+
const afterSettle = sessionAdapter._inputs.filter((i) => i.input.includes("<info_for_agent>"));
|
|
122
|
+
expect(afterSettle.length).toBe(beforeCount + 1);
|
|
123
|
+
expect(afterSettle[afterSettle.length - 1].input).toContain('id="m2"');
|
|
124
|
+
expect(afterSettle[afterSettle.length - 1].input).not.toContain('id="m1"');
|
|
125
|
+
});
|
|
126
|
+
it("resume on handleUserMessage delivers inbox to idle roles too", async () => {
|
|
127
|
+
const sessionAdapter = createMockSessionAdapter();
|
|
128
|
+
const hubAdapter = createMockHubAdapter();
|
|
129
|
+
const coordinator = new SwarmCoordinator(sessionAdapter, hubAdapter);
|
|
130
|
+
const swarmId = "sw-resume-idle";
|
|
131
|
+
await coordinator.create(swarmId, {
|
|
132
|
+
workDir: WORK,
|
|
133
|
+
roles: [{ roleName: "queen", count: 1, eager: true }],
|
|
134
|
+
});
|
|
135
|
+
await new Promise((r) => setTimeout(r, 30));
|
|
136
|
+
// Drive role to idle via a turn-complete (mirrors real lifecycle).
|
|
137
|
+
coordinator.handleRoleTurnComplete(`${swarmId}::queen-0`, {});
|
|
138
|
+
const swarm = coordinator.getSwarm(swarmId);
|
|
139
|
+
const queenRole = swarm.roles.get("queen-0");
|
|
140
|
+
expect(queenRole.status).toBe("idle");
|
|
141
|
+
// Force paused state.
|
|
142
|
+
swarm.status = "paused";
|
|
143
|
+
// Append an inbox message that resume should deliver despite role being idle.
|
|
144
|
+
const store = new InboxStore(WORK, swarmId);
|
|
145
|
+
await store.append("queen-0", { ...baseMsg, id: "idle-resume-1", data: "wake idle role" });
|
|
146
|
+
const before = sessionAdapter._inputs.filter((i) => i.input.includes("<info_for_agent>")).length;
|
|
147
|
+
await coordinator.handleUserMessage(swarmId, "wake up");
|
|
148
|
+
await new Promise((r) => setTimeout(r, 60));
|
|
149
|
+
const after = sessionAdapter._inputs.filter((i) => i.input.includes("<info_for_agent>"));
|
|
150
|
+
expect(after.length).toBeGreaterThan(before);
|
|
151
|
+
// resume should drain the pre-existing inbox entry; the "wake up" user
|
|
152
|
+
// message is itself appended via the inbox path now, so check ANY of the
|
|
153
|
+
// new envelopes (not just the last) for the pre-existing id.
|
|
154
|
+
const newEnvs = after.slice(before);
|
|
155
|
+
expect(newEnvs.some((e) => e.input.includes('id="idle-resume-1"'))).toBe(true);
|
|
156
|
+
});
|
|
157
|
+
it("resume on handleUserMessage triggers deliver for active roles", async () => {
|
|
158
|
+
const sessionAdapter = createMockSessionAdapter();
|
|
159
|
+
const hubAdapter = createMockHubAdapter();
|
|
160
|
+
const coordinator = new SwarmCoordinator(sessionAdapter, hubAdapter);
|
|
161
|
+
const swarmId = "sw-resume";
|
|
162
|
+
await coordinator.create(swarmId, {
|
|
163
|
+
workDir: WORK,
|
|
164
|
+
roles: [{ roleName: "queen", count: 1, eager: true }],
|
|
165
|
+
});
|
|
166
|
+
await new Promise((r) => setTimeout(r, 30));
|
|
167
|
+
// Force paused state.
|
|
168
|
+
const swarm = coordinator.getSwarm(swarmId);
|
|
169
|
+
swarm.status = "paused";
|
|
170
|
+
// Add an inbox message that resume should deliver.
|
|
171
|
+
const store = new InboxStore(WORK, swarmId);
|
|
172
|
+
await store.append("queen-0", { ...baseMsg, id: "resumed-1", data: "after resume" });
|
|
173
|
+
const before = sessionAdapter._inputs.filter((i) => i.input.includes("<info_for_agent>")).length;
|
|
174
|
+
await coordinator.handleUserMessage(swarmId, "wake up");
|
|
175
|
+
await new Promise((r) => setTimeout(r, 50));
|
|
176
|
+
const after = sessionAdapter._inputs.filter((i) => i.input.includes("<info_for_agent>"));
|
|
177
|
+
expect(after.length).toBeGreaterThan(before);
|
|
178
|
+
const newEnvs = after.slice(before);
|
|
179
|
+
expect(newEnvs.some((e) => e.input.includes('id="resumed-1"'))).toBe(true);
|
|
180
|
+
});
|
|
181
|
+
});
|
|
182
|
+
//# sourceMappingURL=swarm-coordinator-inbox.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"swarm-coordinator-inbox.test.js","sourceRoot":"","sources":["../../src/__tests__/swarm-coordinator-inbox.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -1,4 +1,7 @@
|
|
|
1
|
-
import { describe, it, expect, vi, beforeEach } from "vitest";
|
|
1
|
+
import { describe, it, expect, vi, beforeEach, afterEach } from "vitest";
|
|
2
|
+
import { mkdtempSync, rmSync } from "node:fs";
|
|
3
|
+
import { tmpdir } from "node:os";
|
|
4
|
+
import { join } from "node:path";
|
|
2
5
|
// Mock logger
|
|
3
6
|
vi.mock("@mclawnet/logger", () => ({
|
|
4
7
|
createLogger: () => ({
|
|
@@ -80,14 +83,24 @@ describe("SwarmCoordinator — two-phase initialization", () => {
|
|
|
80
83
|
let sessionAdapter;
|
|
81
84
|
let hubAdapter;
|
|
82
85
|
let coordinator;
|
|
86
|
+
let HOME;
|
|
87
|
+
let WORK;
|
|
83
88
|
beforeEach(() => {
|
|
89
|
+
HOME = mkdtempSync(join(tmpdir(), "swarm-init-home-"));
|
|
90
|
+
WORK = mkdtempSync(join(tmpdir(), "swarm-init-work-"));
|
|
91
|
+
process.env.CLAWNET_HOME = HOME;
|
|
84
92
|
sessionAdapter = createMockSessionAdapter();
|
|
85
93
|
hubAdapter = createMockHubAdapter();
|
|
86
94
|
coordinator = new SwarmCoordinator(sessionAdapter, hubAdapter);
|
|
87
95
|
});
|
|
96
|
+
afterEach(() => {
|
|
97
|
+
rmSync(HOME, { recursive: true, force: true });
|
|
98
|
+
rmSync(WORK, { recursive: true, force: true });
|
|
99
|
+
delete process.env.CLAWNET_HOME;
|
|
100
|
+
});
|
|
88
101
|
it("should spawn non-queen eager roles before queen (reviewer before queen)", async () => {
|
|
89
102
|
await coordinator.create("swarm-phase", {
|
|
90
|
-
workDir:
|
|
103
|
+
workDir: WORK,
|
|
91
104
|
roles: [
|
|
92
105
|
{ roleName: "queen", count: 1, eager: true },
|
|
93
106
|
{ roleName: "reviewer", count: 1, eager: true },
|
|
@@ -105,7 +118,7 @@ describe("SwarmCoordinator — two-phase initialization", () => {
|
|
|
105
118
|
});
|
|
106
119
|
it("should create swarm from templateName", async () => {
|
|
107
120
|
await coordinator.create("swarm-tpl", {
|
|
108
|
-
workDir:
|
|
121
|
+
workDir: WORK,
|
|
109
122
|
templateName: "dev-team",
|
|
110
123
|
task: "Build feature",
|
|
111
124
|
});
|
|
@@ -121,7 +134,7 @@ describe("SwarmCoordinator — two-phase initialization", () => {
|
|
|
121
134
|
});
|
|
122
135
|
it("should send init-complete message to queen after all eager roles spawned", async () => {
|
|
123
136
|
await coordinator.create("swarm-init", {
|
|
124
|
-
workDir:
|
|
137
|
+
workDir: WORK,
|
|
125
138
|
roles: [
|
|
126
139
|
{ roleName: "queen", count: 1, eager: true },
|
|
127
140
|
{ roleName: "reviewer", count: 1, eager: true },
|
|
@@ -141,7 +154,7 @@ describe("SwarmCoordinator — two-phase initialization", () => {
|
|
|
141
154
|
});
|
|
142
155
|
it("should ensure queen is always eager even if not explicitly marked", async () => {
|
|
143
156
|
await coordinator.create("swarm-queen-eager", {
|
|
144
|
-
workDir:
|
|
157
|
+
workDir: WORK,
|
|
145
158
|
roles: [
|
|
146
159
|
{ roleName: "queen", count: 1 }, // no eager flag
|
|
147
160
|
{ roleName: "developer", count: 1 },
|
|
@@ -154,7 +167,7 @@ describe("SwarmCoordinator — two-phase initialization", () => {
|
|
|
154
167
|
});
|
|
155
168
|
it("should not spawn non-eager roles during creation", async () => {
|
|
156
169
|
await coordinator.create("swarm-lazy", {
|
|
157
|
-
workDir:
|
|
170
|
+
workDir: WORK,
|
|
158
171
|
roles: [
|
|
159
172
|
{ roleName: "queen", count: 1, eager: true },
|
|
160
173
|
{ roleName: "developer", count: 2 },
|
|
@@ -168,7 +181,7 @@ describe("SwarmCoordinator — two-phase initialization", () => {
|
|
|
168
181
|
});
|
|
169
182
|
it("should create minimal template with only queen eager", async () => {
|
|
170
183
|
await coordinator.create("swarm-minimal", {
|
|
171
|
-
workDir:
|
|
184
|
+
workDir: WORK,
|
|
172
185
|
templateName: "minimal",
|
|
173
186
|
task: "Quick prototype",
|
|
174
187
|
});
|
|
@@ -176,9 +189,24 @@ describe("SwarmCoordinator — two-phase initialization", () => {
|
|
|
176
189
|
expect(sessionAdapter._calls).toHaveLength(1);
|
|
177
190
|
expect(sessionAdapter._calls[0].sessionId).toContain("queen-");
|
|
178
191
|
});
|
|
192
|
+
it("should pass resumeId=undefined to SessionAdapter.createSession during normal spawn (Task 5 will populate)", async () => {
|
|
193
|
+
await coordinator.create("swarm-resume-undef", {
|
|
194
|
+
workDir: WORK,
|
|
195
|
+
roles: [
|
|
196
|
+
{ roleName: "queen", count: 1, eager: true },
|
|
197
|
+
],
|
|
198
|
+
task: "Test",
|
|
199
|
+
});
|
|
200
|
+
// Each createSession call should explicitly carry resumeId, defaulting to undefined
|
|
201
|
+
expect(sessionAdapter._calls.length).toBeGreaterThan(0);
|
|
202
|
+
for (const call of sessionAdapter._calls) {
|
|
203
|
+
expect("resumeId" in call).toBe(true);
|
|
204
|
+
expect(call.resumeId).toBeUndefined();
|
|
205
|
+
}
|
|
206
|
+
});
|
|
179
207
|
it("should include queen's roleList with already-spawned reviewer", async () => {
|
|
180
208
|
await coordinator.create("swarm-rolelist", {
|
|
181
|
-
workDir:
|
|
209
|
+
workDir: WORK,
|
|
182
210
|
roles: [
|
|
183
211
|
{ roleName: "queen", count: 1, eager: true },
|
|
184
212
|
{ roleName: "reviewer", count: 1, eager: true },
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"swarm-coordinator-init.test.js","sourceRoot":"","sources":["../../src/__tests__/swarm-coordinator-init.test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,EAAE,EAAE,MAAM,EAAE,EAAE,EAAE,UAAU,EAAE,MAAM,QAAQ,CAAC;
|
|
1
|
+
{"version":3,"file":"swarm-coordinator-init.test.js","sourceRoot":"","sources":["../../src/__tests__/swarm-coordinator-init.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"swarm-coordinator-legacy-plan-review-warn.test.d.ts","sourceRoot":"","sources":["../../src/__tests__/swarm-coordinator-legacy-plan-review-warn.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,113 @@
|
|
|
1
|
+
import { describe, it, expect, vi, beforeEach, afterEach } from "vitest";
|
|
2
|
+
import { mkdtempSync, rmSync } from "node:fs";
|
|
3
|
+
import { tmpdir } from "node:os";
|
|
4
|
+
import { join } from "node:path";
|
|
5
|
+
// Capture warn calls so the test can assert the safety-net log fires.
|
|
6
|
+
// vi.hoisted ensures the spy is available inside the (hoisted) vi.mock factory.
|
|
7
|
+
const { warnSpy } = vi.hoisted(() => ({ warnSpy: vi.fn() }));
|
|
8
|
+
vi.mock("@mclawnet/logger", () => ({
|
|
9
|
+
createLogger: () => ({
|
|
10
|
+
info: vi.fn(),
|
|
11
|
+
warn: warnSpy,
|
|
12
|
+
error: vi.fn(),
|
|
13
|
+
debug: vi.fn(),
|
|
14
|
+
}),
|
|
15
|
+
}));
|
|
16
|
+
vi.mock("../persistence.js", () => ({
|
|
17
|
+
saveSwarmSnapshot: vi.fn(),
|
|
18
|
+
deleteSwarmSnapshot: vi.fn(),
|
|
19
|
+
appendMessageLog: vi.fn(),
|
|
20
|
+
loadSwarmSnapshot: vi.fn(),
|
|
21
|
+
readMessageLog: vi.fn(() => []),
|
|
22
|
+
}));
|
|
23
|
+
vi.mock("../retrospective.js", () => ({
|
|
24
|
+
runRetrospective: vi.fn(async () => ({})),
|
|
25
|
+
}));
|
|
26
|
+
import { SwarmCoordinator } from "../swarm-coordinator.js";
|
|
27
|
+
function adapters() {
|
|
28
|
+
const session = {
|
|
29
|
+
createSession: vi.fn(async () => ""),
|
|
30
|
+
sendInput: vi.fn(),
|
|
31
|
+
closeSession: vi.fn(async () => { }),
|
|
32
|
+
};
|
|
33
|
+
const hub = { send: vi.fn() };
|
|
34
|
+
return { session, hub };
|
|
35
|
+
}
|
|
36
|
+
let HOME;
|
|
37
|
+
let WORK;
|
|
38
|
+
beforeEach(() => {
|
|
39
|
+
warnSpy.mockReset();
|
|
40
|
+
HOME = mkdtempSync(join(tmpdir(), "swarm-legacy-pr-warn-home-"));
|
|
41
|
+
WORK = mkdtempSync(join(tmpdir(), "swarm-legacy-pr-warn-work-"));
|
|
42
|
+
process.env.CLAWNET_HOME = HOME;
|
|
43
|
+
});
|
|
44
|
+
afterEach(() => {
|
|
45
|
+
rmSync(HOME, { recursive: true, force: true });
|
|
46
|
+
rmSync(WORK, { recursive: true, force: true });
|
|
47
|
+
delete process.env.CLAWNET_HOME;
|
|
48
|
+
});
|
|
49
|
+
async function setupSwarm(swarmId) {
|
|
50
|
+
const { session, hub } = adapters();
|
|
51
|
+
const coord = new SwarmCoordinator(session, hub);
|
|
52
|
+
await coord.create(swarmId, {
|
|
53
|
+
workDir: WORK,
|
|
54
|
+
roles: [
|
|
55
|
+
{ roleName: "queen", count: 1, eager: true },
|
|
56
|
+
{ roleName: "reviewer", count: 1, eager: true },
|
|
57
|
+
],
|
|
58
|
+
});
|
|
59
|
+
await new Promise((r) => setTimeout(r, 30));
|
|
60
|
+
return coord;
|
|
61
|
+
}
|
|
62
|
+
function streamingTextEvent(text) {
|
|
63
|
+
// Mirrors the Anthropic streaming `assistant` content_block shape that
|
|
64
|
+
// extractTextFromEvent reads. Keep this minimal — we only need the text path.
|
|
65
|
+
return {
|
|
66
|
+
type: "assistant",
|
|
67
|
+
message: { content: [{ type: "text", text }] },
|
|
68
|
+
};
|
|
69
|
+
}
|
|
70
|
+
describe("SwarmCoordinator legacy plan_review report safety net", () => {
|
|
71
|
+
it("warns when reviewer emits legacy report block while planStatus=reviewing", async () => {
|
|
72
|
+
const swarmId = "sw-legacy-warn";
|
|
73
|
+
const coord = await setupSwarm(swarmId);
|
|
74
|
+
const swarm = coord.getSwarm(swarmId);
|
|
75
|
+
swarm.planStatus = "reviewing";
|
|
76
|
+
const text = 'Here is my review:\n\n```swarm\n{"action":"report","status":"completed","taskId":"plan_review","output":"approved"}\n```';
|
|
77
|
+
coord.handleRoleOutput(`${swarmId}::reviewer-0`, streamingTextEvent(text));
|
|
78
|
+
const matched = warnSpy.mock.calls.find(([_ctx, msg]) => typeof msg === "string" && msg.includes("plan_review_submit"));
|
|
79
|
+
expect(matched).toBeDefined();
|
|
80
|
+
});
|
|
81
|
+
it("does NOT warn when reviewer emits the same block while planStatus is not reviewing", async () => {
|
|
82
|
+
const swarmId = "sw-legacy-no-warn-status";
|
|
83
|
+
const coord = await setupSwarm(swarmId);
|
|
84
|
+
const swarm = coord.getSwarm(swarmId);
|
|
85
|
+
swarm.planStatus = "approved"; // not reviewing → should not trigger
|
|
86
|
+
const text = '```swarm\n{"action":"report","status":"completed","taskId":"plan_review","output":"x"}\n```';
|
|
87
|
+
coord.handleRoleOutput(`${swarmId}::reviewer-0`, streamingTextEvent(text));
|
|
88
|
+
const matched = warnSpy.mock.calls.find(([_ctx, msg]) => typeof msg === "string" && msg.includes("plan_review_submit"));
|
|
89
|
+
expect(matched).toBeUndefined();
|
|
90
|
+
});
|
|
91
|
+
it("does NOT warn for non-reviewer roles even if the text matches", async () => {
|
|
92
|
+
const swarmId = "sw-legacy-no-warn-role";
|
|
93
|
+
const coord = await setupSwarm(swarmId);
|
|
94
|
+
const swarm = coord.getSwarm(swarmId);
|
|
95
|
+
swarm.planStatus = "reviewing";
|
|
96
|
+
const text = '```swarm\n{"action":"report","status":"completed","taskId":"plan_review","output":"x"}\n```';
|
|
97
|
+
// queen-0 cannot trigger the safety net (only reviewer roles)
|
|
98
|
+
coord.handleRoleOutput(`${swarmId}::queen-0`, streamingTextEvent(text));
|
|
99
|
+
const matched = warnSpy.mock.calls.find(([_ctx, msg]) => typeof msg === "string" && msg.includes("plan_review_submit"));
|
|
100
|
+
expect(matched).toBeUndefined();
|
|
101
|
+
});
|
|
102
|
+
it("does NOT warn when reviewer's text contains report block with a different taskId", async () => {
|
|
103
|
+
const swarmId = "sw-legacy-no-warn-taskid";
|
|
104
|
+
const coord = await setupSwarm(swarmId);
|
|
105
|
+
const swarm = coord.getSwarm(swarmId);
|
|
106
|
+
swarm.planStatus = "reviewing";
|
|
107
|
+
const text = '```swarm\n{"action":"report","status":"completed","taskId":"some_other_task","output":"x"}\n```';
|
|
108
|
+
coord.handleRoleOutput(`${swarmId}::reviewer-0`, streamingTextEvent(text));
|
|
109
|
+
const matched = warnSpy.mock.calls.find(([_ctx, msg]) => typeof msg === "string" && msg.includes("plan_review_submit"));
|
|
110
|
+
expect(matched).toBeUndefined();
|
|
111
|
+
});
|
|
112
|
+
});
|
|
113
|
+
//# sourceMappingURL=swarm-coordinator-legacy-plan-review-warn.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"swarm-coordinator-legacy-plan-review-warn.test.js","sourceRoot":"","sources":["../../src/__tests__/swarm-coordinator-legacy-plan-review-warn.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"swarm-coordinator-plan-review-intercept.test.d.ts","sourceRoot":"","sources":["../../src/__tests__/swarm-coordinator-plan-review-intercept.test.ts"],"names":[],"mappings":""}
|