@pi-unipi/subagents 0.1.11 → 0.1.12

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/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@pi-unipi/subagents",
3
- "version": "0.1.11",
3
+ "version": "0.1.12",
4
4
  "description": "Subagents for UniPi — parallel execution, file locking, workflow integration",
5
5
  "type": "module",
6
6
  "main": "dist/index.js",
@@ -13,7 +13,8 @@
13
13
  },
14
14
  "scripts": {
15
15
  "build": "tsc",
16
- "dev": "tsc --watch"
16
+ "dev": "tsc --watch",
17
+ "test": "node --test src/__tests__/*.test.ts"
17
18
  },
18
19
  "dependencies": {
19
20
  "@pi-unipi/core": "*",
@@ -0,0 +1,240 @@
1
+ /**
2
+ * Test: Config auto-generation and corruption recovery
3
+ *
4
+ * Verifies:
5
+ * - Missing config → auto-generated with defaults
6
+ * - Corrupted config → renamed to .json.bak, fresh generated
7
+ * - Workspace config overrides global config
8
+ * - Atomic writes prevent corruption
9
+ */
10
+
11
+ import { describe, it, beforeEach, afterEach } from "node:test";
12
+ import assert from "node:assert/strict";
13
+ import { existsSync, mkdirSync, readFileSync, writeFileSync, rmSync, renameSync } from "node:fs";
14
+ import { join } from "node:path";
15
+ import { tmpdir } from "node:os";
16
+
17
+ // Inline config implementation for testing
18
+ const DEFAULT_CONFIG = {
19
+ maxConcurrent: 4,
20
+ enabled: true,
21
+ types: {
22
+ explore: { enabled: true },
23
+ work: { enabled: true },
24
+ },
25
+ };
26
+
27
+ interface SubagentsConfig {
28
+ maxConcurrent: number;
29
+ enabled: boolean;
30
+ types: Record<string, { enabled?: boolean }>;
31
+ }
32
+
33
+ function loadConfigFromPath(filePath: string): SubagentsConfig | null {
34
+ if (!existsSync(filePath)) return null;
35
+ try {
36
+ const content = readFileSync(filePath, "utf-8");
37
+ const parsed = JSON.parse(content);
38
+ if (typeof parsed !== "object" || parsed === null) return null;
39
+ return parsed as SubagentsConfig;
40
+ } catch {
41
+ return null;
42
+ }
43
+ }
44
+
45
+ function repairCorrupted(filePath: string): SubagentsConfig {
46
+ const backupPath = filePath + ".bak";
47
+ try {
48
+ renameSync(filePath, backupPath);
49
+ } catch {
50
+ // If rename fails, just overwrite
51
+ }
52
+ writeConfigAtomic(filePath, DEFAULT_CONFIG);
53
+ return DEFAULT_CONFIG;
54
+ }
55
+
56
+ function writeConfigAtomic(filePath: string, config: SubagentsConfig): void {
57
+ const tmpPath = filePath + ".tmp";
58
+ writeFileSync(tmpPath, JSON.stringify(config, null, 2), "utf-8");
59
+ renameSync(tmpPath, filePath);
60
+ }
61
+
62
+ function initConfig(globalDir: string, workspaceDir: string): SubagentsConfig {
63
+ const globalPath = join(globalDir, "subagents.json");
64
+ const workspacePath = join(workspaceDir, "subagents.json");
65
+
66
+ // Load or create global config
67
+ let globalConfig = loadConfigFromPath(globalPath);
68
+ if (globalConfig === null) {
69
+ globalConfig = repairCorrupted(globalPath);
70
+ }
71
+
72
+ // Load workspace override if exists
73
+ const workspaceConfig = loadConfigFromPath(workspacePath);
74
+
75
+ if (workspaceConfig) {
76
+ // Merge: workspace overrides global on any field present
77
+ return {
78
+ ...globalConfig,
79
+ ...workspaceConfig,
80
+ types: {
81
+ ...globalConfig.types,
82
+ ...workspaceConfig.types,
83
+ },
84
+ };
85
+ }
86
+
87
+ return globalConfig;
88
+ }
89
+
90
+ describe("Config Management", () => {
91
+ let testDir: string;
92
+ let globalDir: string;
93
+ let workspaceDir: string;
94
+
95
+ beforeEach(() => {
96
+ // Create temp directories for testing
97
+ testDir = join(tmpdir(), `subagents-test-${Date.now()}`);
98
+ globalDir = join(testDir, "global");
99
+ workspaceDir = join(testDir, "workspace");
100
+ mkdirSync(globalDir, { recursive: true });
101
+ mkdirSync(workspaceDir, { recursive: true });
102
+ });
103
+
104
+ afterEach(() => {
105
+ // Cleanup
106
+ rmSync(testDir, { recursive: true, force: true });
107
+ });
108
+
109
+ describe("Missing config", () => {
110
+ it("should auto-generate with defaults when no config exists", () => {
111
+ const config = initConfig(globalDir, workspaceDir);
112
+
113
+ assert.deepEqual(config, DEFAULT_CONFIG);
114
+ assert.equal(existsSync(join(globalDir, "subagents.json")), true);
115
+
116
+ // Verify the generated file
117
+ const content = readFileSync(join(globalDir, "subagents.json"), "utf-8");
118
+ const parsed = JSON.parse(content);
119
+ assert.deepEqual(parsed, DEFAULT_CONFIG);
120
+ });
121
+
122
+ it("should create global config even if workspace exists", () => {
123
+ // Write workspace config only
124
+ const workspaceConfig = { maxConcurrent: 8, enabled: true, types: {} };
125
+ writeFileSync(join(workspaceDir, "subagents.json"), JSON.stringify(workspaceConfig));
126
+
127
+ const config = initConfig(globalDir, workspaceDir);
128
+
129
+ // Global should be created
130
+ assert.equal(existsSync(join(globalDir, "subagents.json")), true);
131
+ // Config should merge
132
+ assert.equal(config.maxConcurrent, 8); // workspace overrides
133
+ });
134
+ });
135
+
136
+ describe("Corrupted config", () => {
137
+ it("should backup corrupted config and generate fresh", () => {
138
+ const configPath = join(globalDir, "subagents.json");
139
+ const backupPath = configPath + ".bak";
140
+
141
+ // Write corrupted JSON
142
+ writeFileSync(configPath, "{ invalid json !!!");
143
+
144
+ const config = initConfig(globalDir, workspaceDir);
145
+
146
+ // Should have created backup
147
+ assert.equal(existsSync(backupPath), true, "Backup should exist");
148
+ // Backup should contain corrupted content
149
+ assert.equal(readFileSync(backupPath, "utf-8"), "{ invalid json !!!");
150
+ // New config should be defaults
151
+ assert.deepEqual(config, DEFAULT_CONFIG);
152
+ // New file should be valid JSON
153
+ const content = readFileSync(configPath, "utf-8");
154
+ assert.deepEqual(JSON.parse(content), DEFAULT_CONFIG);
155
+ });
156
+
157
+ it("should handle completely empty file", () => {
158
+ const configPath = join(globalDir, "subagents.json");
159
+ writeFileSync(configPath, "");
160
+
161
+ const config = initConfig(globalDir, workspaceDir);
162
+
163
+ assert.deepEqual(config, DEFAULT_CONFIG);
164
+ assert.equal(existsSync(configPath + ".bak"), true);
165
+ });
166
+
167
+ it("should handle non-object JSON", () => {
168
+ const configPath = join(globalDir, "subagents.json");
169
+ writeFileSync(configPath, '"just a string"');
170
+
171
+ const config = initConfig(globalDir, workspaceDir);
172
+
173
+ // JSON.parse succeeds but returns string, not object
174
+ // loadConfigFromPath checks typeof === "object"
175
+ assert.deepEqual(config, DEFAULT_CONFIG);
176
+ });
177
+ });
178
+
179
+ describe("Workspace override", () => {
180
+ it("should merge workspace config with global", () => {
181
+ const globalConfig = { maxConcurrent: 4, enabled: true, types: { explore: { enabled: true } } };
182
+ const workspaceConfig = { maxConcurrent: 8, types: { work: { enabled: false } } };
183
+
184
+ writeFileSync(join(globalDir, "subagents.json"), JSON.stringify(globalConfig));
185
+ writeFileSync(join(workspaceDir, "subagents.json"), JSON.stringify(workspaceConfig));
186
+
187
+ const config = initConfig(globalDir, workspaceDir);
188
+
189
+ assert.equal(config.maxConcurrent, 8); // workspace overrides
190
+ assert.equal(config.enabled, true); // global preserved
191
+ assert.deepEqual(config.types.explore, { enabled: true }); // global preserved
192
+ assert.deepEqual(config.types.work, { enabled: false }); // workspace added
193
+ });
194
+
195
+ it("should override specific fields only", () => {
196
+ const globalConfig = {
197
+ maxConcurrent: 4,
198
+ enabled: true,
199
+ types: { explore: { enabled: true }, work: { enabled: true } },
200
+ };
201
+ const workspaceConfig = { enabled: false };
202
+
203
+ writeFileSync(join(globalDir, "subagents.json"), JSON.stringify(globalConfig));
204
+ writeFileSync(join(workspaceDir, "subagents.json"), JSON.stringify(workspaceConfig));
205
+
206
+ const config = initConfig(globalDir, workspaceDir);
207
+
208
+ assert.equal(config.maxConcurrent, 4); // global preserved
209
+ assert.equal(config.enabled, false); // workspace overrides
210
+ assert.deepEqual(config.types, { explore: { enabled: true }, work: { enabled: true } }); // global preserved
211
+ });
212
+
213
+ it("should handle empty workspace config", () => {
214
+ const globalConfig = { maxConcurrent: 4, enabled: true, types: {} };
215
+ writeFileSync(join(globalDir, "subagents.json"), JSON.stringify(globalConfig));
216
+ writeFileSync(join(workspaceDir, "subagents.json"), "{}");
217
+
218
+ const config = initConfig(globalDir, workspaceDir);
219
+
220
+ assert.deepEqual(config, globalConfig);
221
+ });
222
+ });
223
+
224
+ describe("Atomic writes", () => {
225
+ it("should write config atomically", () => {
226
+ const configPath = join(globalDir, "subagents.json");
227
+ const config = { maxConcurrent: 8, enabled: false, types: {} };
228
+
229
+ writeConfigAtomic(configPath, config);
230
+
231
+ // Should have main file
232
+ assert.equal(existsSync(configPath), true);
233
+ // Should not have temp file
234
+ assert.equal(existsSync(configPath + ".tmp"), false);
235
+ // Content should be valid
236
+ const content = readFileSync(configPath, "utf-8");
237
+ assert.deepEqual(JSON.parse(content), config);
238
+ });
239
+ });
240
+ });
@@ -0,0 +1,162 @@
1
+ /**
2
+ * Test: ESC propagation — all children abort on parent ESC
3
+ *
4
+ * Verifies:
5
+ * - forwardAbortSignal wires parent signal to child session
6
+ * - abortAll stops all running agents
7
+ * - All agents stop within reasonable time
8
+ */
9
+
10
+ import { describe, it, mock, beforeEach } from "node:test";
11
+ import assert from "node:assert/strict";
12
+
13
+ // Mock AbortController to track abort calls
14
+ function createMockAbortController() {
15
+ let aborted = false;
16
+ const listeners: Array<() => void> = [];
17
+ return {
18
+ get signal() {
19
+ return {
20
+ aborted,
21
+ addEventListener: (_event: string, listener: () => void) => {
22
+ listeners.push(listener);
23
+ },
24
+ removeEventListener: (_event: string, listener: () => void) => {
25
+ const idx = listeners.indexOf(listener);
26
+ if (idx !== -1) listeners.splice(idx, 1);
27
+ },
28
+ };
29
+ },
30
+ abort() {
31
+ aborted = true;
32
+ for (const listener of listeners) listener();
33
+ },
34
+ get wasAborted() {
35
+ return aborted;
36
+ },
37
+ };
38
+ }
39
+
40
+ describe("ESC Propagation", () => {
41
+ describe("forwardAbortSignal", () => {
42
+ it("should call session.abort() when signal fires", () => {
43
+ // Simulate the forwardAbortSignal logic from agent-runner.ts
44
+ const sessionAborted = { value: false };
45
+ const session = { abort: () => { sessionAborted.value = true; } };
46
+ const controller = createMockAbortController();
47
+
48
+ // Wire abort signal
49
+ const onAbort = () => session.abort();
50
+ controller.signal.addEventListener("abort", onAbort);
51
+
52
+ // Trigger abort
53
+ controller.abort();
54
+
55
+ assert.equal(sessionAborted.value, true, "Session should be aborted");
56
+ });
57
+
58
+ it("should not call session.abort() if signal not fired", () => {
59
+ const sessionAborted = { value: false };
60
+ const session = { abort: () => { sessionAborted.value = true; } };
61
+ const controller = createMockAbortController();
62
+
63
+ const onAbort = () => session.abort();
64
+ controller.signal.addEventListener("abort", onAbort);
65
+
66
+ // Don't abort
67
+ assert.equal(sessionAborted.value, false, "Session should not be aborted");
68
+ });
69
+
70
+ it("should cleanup listener when returned function called", () => {
71
+ const controller = createMockAbortController();
72
+ let callCount = 0;
73
+ const onAbort = () => { callCount++; };
74
+ controller.signal.addEventListener("abort", onAbort);
75
+
76
+ // Simulate cleanup
77
+ const cleanup = () => controller.signal.removeEventListener("abort", onAbort);
78
+ cleanup();
79
+
80
+ controller.abort();
81
+ assert.equal(callCount, 0, "Listener should not fire after cleanup");
82
+ });
83
+ });
84
+
85
+ describe("abortAll", () => {
86
+ it("should abort all running agents", () => {
87
+ const agents = new Map<string, { abortController: ReturnType<typeof createMockAbortController>; status: string }>();
88
+
89
+ // Create 3 mock agents
90
+ for (let i = 0; i < 3; i++) {
91
+ const controller = createMockAbortController();
92
+ agents.set(`agent-${i}`, {
93
+ abortController: controller,
94
+ status: "running",
95
+ });
96
+ }
97
+
98
+ // Simulate abortAll
99
+ let abortedCount = 0;
100
+ for (const [id, record] of agents) {
101
+ if (record.status === "running") {
102
+ record.abortController.abort();
103
+ record.status = "stopped";
104
+ abortedCount++;
105
+ }
106
+ }
107
+
108
+ assert.equal(abortedCount, 3, "Should abort all 3 agents");
109
+ for (const [_, record] of agents) {
110
+ assert.equal(record.status, "stopped", "All agents should be stopped");
111
+ assert.equal(record.abortController.wasAborted, true, "All controllers should be aborted");
112
+ }
113
+ });
114
+
115
+ it("should handle queued agents by removing from queue", () => {
116
+ const queue = [
117
+ { id: "queued-1", status: "queued" },
118
+ { id: "queued-2", status: "queued" },
119
+ ];
120
+ const agents = new Map<string, { status: string }>();
121
+
122
+ for (const item of queue) {
123
+ agents.set(item.id, { status: item.status });
124
+ }
125
+
126
+ // Simulate abortAll for queued
127
+ for (const item of queue) {
128
+ const record = agents.get(item.id);
129
+ if (record) {
130
+ record.status = "stopped";
131
+ }
132
+ }
133
+ queue.length = 0;
134
+
135
+ assert.equal(queue.length, 0, "Queue should be empty");
136
+ for (const [_, record] of agents) {
137
+ assert.equal(record.status, "stopped", "All queued agents should be stopped");
138
+ }
139
+ });
140
+ });
141
+
142
+ describe("ESC timing", () => {
143
+ it("should abort within reasonable time", async () => {
144
+ const controller = createMockAbortController();
145
+ let abortedAt: number | null = null;
146
+ const startedAt = Date.now();
147
+
148
+ const onAbort = () => { abortedAt = Date.now(); };
149
+ controller.signal.addEventListener("abort", onAbort);
150
+
151
+ // Simulate abort after small delay
152
+ setTimeout(() => controller.abort(), 10);
153
+
154
+ // Wait for abort
155
+ await new Promise(resolve => setTimeout(resolve, 50));
156
+
157
+ assert.notEqual(abortedAt, null, "Should have aborted");
158
+ const elapsed = abortedAt! - startedAt;
159
+ assert.ok(elapsed < 500, `Abort should happen within 500ms, took ${elapsed}ms`);
160
+ });
161
+ });
162
+ });
@@ -0,0 +1,244 @@
1
+ /**
2
+ * Test: File locking — concurrent writes to same file queue correctly
3
+ *
4
+ * Verifies:
5
+ * - Per-file locking works correctly
6
+ * - Same file writes queue (second waits for first)
7
+ * - Different file writes proceed in parallel
8
+ * - Lock release unblocks waiting acquires
9
+ * - releaseAll releases all locks for an agent
10
+ */
11
+
12
+ import { describe, it } from "node:test";
13
+ import assert from "node:assert/strict";
14
+
15
+ // Inline FileLock implementation for testing (avoids TS import issues)
16
+ interface FileLockEntry {
17
+ agentId: string;
18
+ filePath: string;
19
+ promise: Promise<void>;
20
+ release: () => void;
21
+ }
22
+
23
+ class FileLock {
24
+ private locks = new Map<string, FileLockEntry>();
25
+ private queues = new Map<string, Array<() => void>>();
26
+
27
+ async acquire(filePath: string, agentId: string): Promise<() => void> {
28
+ while (this.locks.has(filePath)) {
29
+ await new Promise<void>((resolve) => {
30
+ const queue = this.queues.get(filePath) ?? [];
31
+ queue.push(resolve);
32
+ this.queues.set(filePath, queue);
33
+ });
34
+ }
35
+
36
+ let releaseFn: () => void;
37
+ const promise = new Promise<void>((resolve) => {
38
+ releaseFn = () => {
39
+ this.locks.delete(filePath);
40
+ resolve();
41
+ const queue = this.queues.get(filePath);
42
+ if (queue && queue.length > 0) {
43
+ const next = queue.shift()!;
44
+ next();
45
+ }
46
+ };
47
+ });
48
+
49
+ const entry: FileLockEntry = {
50
+ agentId,
51
+ filePath,
52
+ promise,
53
+ release: releaseFn!,
54
+ };
55
+
56
+ this.locks.set(filePath, entry);
57
+ return releaseFn!;
58
+ }
59
+
60
+ isLocked(filePath: string): boolean {
61
+ return this.locks.has(filePath);
62
+ }
63
+
64
+ getHolder(filePath: string): string | undefined {
65
+ return this.locks.get(filePath)?.agentId;
66
+ }
67
+
68
+ get lockCount(): number {
69
+ return this.locks.size;
70
+ }
71
+
72
+ releaseAll(agentId: string): void {
73
+ for (const [filePath, entry] of this.locks) {
74
+ if (entry.agentId === agentId) {
75
+ entry.release();
76
+ }
77
+ }
78
+ }
79
+
80
+ clear(): void {
81
+ for (const entry of this.locks.values()) {
82
+ entry.release();
83
+ }
84
+ this.locks.clear();
85
+ this.queues.clear();
86
+ }
87
+ }
88
+
89
+ describe("FileLock", () => {
90
+ describe("Basic locking", () => {
91
+ it("should acquire lock on unlocked file", async () => {
92
+ const lock = new FileLock();
93
+ const release = await lock.acquire("/src/auth.ts", "agent-1");
94
+
95
+ assert.equal(lock.isLocked("/src/auth.ts"), true);
96
+ assert.equal(lock.getHolder("/src/auth.ts"), "agent-1");
97
+ assert.equal(lock.lockCount, 1);
98
+
99
+ release();
100
+ assert.equal(lock.isLocked("/src/auth.ts"), false);
101
+ });
102
+
103
+ it("should track multiple locks on different files", async () => {
104
+ const lock = new FileLock();
105
+ const release1 = await lock.acquire("/src/auth.ts", "agent-1");
106
+ const release2 = await lock.acquire("/src/login.ts", "agent-2");
107
+
108
+ assert.equal(lock.lockCount, 2);
109
+ assert.equal(lock.getHolder("/src/auth.ts"), "agent-1");
110
+ assert.equal(lock.getHolder("/src/login.ts"), "agent-2");
111
+
112
+ release1();
113
+ release2();
114
+ assert.equal(lock.lockCount, 0);
115
+ });
116
+ });
117
+
118
+ describe("Queuing behavior", () => {
119
+ it("should queue second acquire on same file", async () => {
120
+ const lock = new FileLock();
121
+ const events: string[] = [];
122
+
123
+ // First acquire
124
+ const release1 = await lock.acquire("/src/auth.ts", "agent-1");
125
+ events.push("agent-1-acquired");
126
+
127
+ // Second acquire (should queue)
128
+ const acquire2Promise = lock.acquire("/src/auth.ts", "agent-2").then((release) => {
129
+ events.push("agent-2-acquired");
130
+ return release;
131
+ });
132
+
133
+ // agent-2 should not have acquired yet
134
+ assert.deepEqual(events, ["agent-1-acquired"]);
135
+
136
+ // Release first lock
137
+ release1();
138
+ events.push("agent-1-released");
139
+
140
+ // Wait for agent-2
141
+ const release2 = await acquire2Promise;
142
+ assert.deepEqual(events, ["agent-1-acquired", "agent-1-released", "agent-2-acquired"]);
143
+
144
+ release2();
145
+ });
146
+
147
+ it("should queue multiple acquires on same file", async () => {
148
+ const lock = new FileLock();
149
+ const events: string[] = [];
150
+
151
+ const release1 = await lock.acquire("/src/auth.ts", "agent-1");
152
+ events.push("1-acquired");
153
+
154
+ const p2 = lock.acquire("/src/auth.ts", "agent-2").then(r => { events.push("2-acquired"); return r; });
155
+ const p3 = lock.acquire("/src/auth.ts", "agent-3").then(r => { events.push("3-acquired"); return r; });
156
+
157
+ release1();
158
+ const release2 = await p2;
159
+ assert.deepEqual(events, ["1-acquired", "2-acquired"]);
160
+
161
+ release2();
162
+ const release3 = await p3;
163
+ assert.deepEqual(events, ["1-acquired", "2-acquired", "3-acquired"]);
164
+
165
+ release3();
166
+ });
167
+ });
168
+
169
+ describe("Parallel different files", () => {
170
+ it("should allow parallel writes to different files", async () => {
171
+ const lock = new FileLock();
172
+ const events: string[] = [];
173
+
174
+ // Both should acquire immediately (different files)
175
+ const release1 = await lock.acquire("/src/auth.ts", "agent-1");
176
+ events.push("auth-acquired");
177
+
178
+ const release2 = await lock.acquire("/src/login.ts", "agent-2");
179
+ events.push("login-acquired");
180
+
181
+ assert.deepEqual(events, ["auth-acquired", "login-acquired"]);
182
+ assert.equal(lock.lockCount, 2);
183
+
184
+ release1();
185
+ release2();
186
+ });
187
+ });
188
+
189
+ describe("releaseAll", () => {
190
+ it("should release all locks for a specific agent", async () => {
191
+ const lock = new FileLock();
192
+
193
+ // agent-1 holds 3 files
194
+ await lock.acquire("/src/a.ts", "agent-1");
195
+ await lock.acquire("/src/b.ts", "agent-1");
196
+ await lock.acquire("/src/c.ts", "agent-1");
197
+
198
+ // agent-2 holds 1 file
199
+ await lock.acquire("/src/d.ts", "agent-2");
200
+
201
+ assert.equal(lock.lockCount, 4);
202
+
203
+ // Release all for agent-1
204
+ lock.releaseAll("agent-1");
205
+
206
+ assert.equal(lock.lockCount, 1);
207
+ assert.equal(lock.isLocked("/src/a.ts"), false);
208
+ assert.equal(lock.isLocked("/src/b.ts"), false);
209
+ assert.equal(lock.isLocked("/src/c.ts"), false);
210
+ assert.equal(lock.isLocked("/src/d.ts"), true);
211
+ });
212
+
213
+ it("should unblock queued acquires when releasing all", async () => {
214
+ const lock = new FileLock();
215
+ const events: string[] = [];
216
+
217
+ const release1 = await lock.acquire("/src/a.ts", "agent-1");
218
+ const p2 = lock.acquire("/src/a.ts", "agent-2").then(r => { events.push("agent-2-acquired"); return r; });
219
+
220
+ // Release all for agent-1
221
+ lock.releaseAll("agent-1");
222
+
223
+ const release2 = await p2;
224
+ assert.deepEqual(events, ["agent-2-acquired"]);
225
+
226
+ release2();
227
+ });
228
+ });
229
+
230
+ describe("clear", () => {
231
+ it("should release all locks and clear queues", async () => {
232
+ const lock = new FileLock();
233
+
234
+ await lock.acquire("/src/a.ts", "agent-1");
235
+ await lock.acquire("/src/b.ts", "agent-2");
236
+
237
+ lock.clear();
238
+
239
+ assert.equal(lock.lockCount, 0);
240
+ assert.equal(lock.isLocked("/src/a.ts"), false);
241
+ assert.equal(lock.isLocked("/src/b.ts"), false);
242
+ });
243
+ });
244
+ });
@@ -0,0 +1,334 @@
1
+ /**
2
+ * Test: Workflow integration — `/unipi:work` with subagent support
3
+ *
4
+ * Verifies:
5
+ * - spawn_helper and get_helper_result tools are properly defined
6
+ * - Agent types (explore, work) are correctly configured
7
+ * - Concurrency limit is respected
8
+ * - Custom agent type loading works
9
+ * - System prompt builder generates correct prompts
10
+ */
11
+
12
+ import { describe, it } from "node:test";
13
+ import assert from "node:assert/strict";
14
+
15
+ // Test type definitions
16
+ const BUILTIN_TYPES = ["explore", "work"] as const;
17
+
18
+ interface AgentConfig {
19
+ name: string;
20
+ displayName?: string;
21
+ description: string;
22
+ builtinToolNames?: string[];
23
+ disallowedTools?: string[];
24
+ extensions: true | string[] | false;
25
+ skills: true | string[] | false;
26
+ model?: string;
27
+ thinking?: string;
28
+ maxTurns?: number;
29
+ systemPrompt: string;
30
+ promptMode: "replace" | "append";
31
+ inheritContext?: boolean;
32
+ runInBackground?: boolean;
33
+ isolated?: boolean;
34
+ memory?: string;
35
+ isDefault?: boolean;
36
+ enabled?: boolean;
37
+ source?: "builtin" | "project" | "global";
38
+ }
39
+
40
+ // Test prompt builder
41
+ function buildAgentPrompt(
42
+ config: AgentConfig,
43
+ cwd: string,
44
+ env: { isGitRepo: boolean; branch: string; platform: string },
45
+ parentSystemPrompt: string,
46
+ ): string {
47
+ if (config.promptMode === "append") {
48
+ return [
49
+ parentSystemPrompt,
50
+ "",
51
+ "---",
52
+ "",
53
+ `## Agent Role: ${config.displayName ?? config.name}`,
54
+ config.systemPrompt,
55
+ ].join("\n");
56
+ }
57
+
58
+ return [
59
+ `# ${config.displayName ?? config.name}`,
60
+ "",
61
+ config.systemPrompt,
62
+ "",
63
+ "---",
64
+ "",
65
+ `Working directory: ${cwd}`,
66
+ `Git: ${env.isGitRepo ? `${env.branch} on ${env.platform}` : "not a git repo"}`,
67
+ ].join("\n");
68
+ }
69
+
70
+ // Test concurrency manager
71
+ class ConcurrencyManager {
72
+ private maxConcurrent: number;
73
+ private running: number = 0;
74
+ private queue: Array<{ id: string; resolve: () => void }> = [];
75
+
76
+ constructor(maxConcurrent: number) {
77
+ this.maxConcurrent = maxConcurrent;
78
+ }
79
+
80
+ async acquire(id: string): Promise<() => void> {
81
+ if (this.running >= this.maxConcurrent) {
82
+ await new Promise<void>((resolve) => {
83
+ this.queue.push({ id, resolve });
84
+ });
85
+ }
86
+
87
+ this.running++;
88
+
89
+ let released = false;
90
+ return () => {
91
+ if (released) return;
92
+ released = true;
93
+ this.running--;
94
+
95
+ // Start next in queue
96
+ if (this.queue.length > 0) {
97
+ const next = this.queue.shift()!;
98
+ next.resolve();
99
+ }
100
+ };
101
+ }
102
+
103
+ getRunning(): number {
104
+ return this.running;
105
+ }
106
+
107
+ getQueueLength(): number {
108
+ return this.queue.length;
109
+ }
110
+ }
111
+
112
+ describe("Workflow Integration", () => {
113
+ describe("Tool Definitions", () => {
114
+ it("should define spawn_helper tool with correct parameters", () => {
115
+ const toolDef = {
116
+ name: "spawn_helper",
117
+ description: "Launch a sub-agent for parallel work.",
118
+ parameters: {
119
+ type: "object",
120
+ required: ["type", "prompt", "description"],
121
+ properties: {
122
+ type: { type: "string" },
123
+ prompt: { type: "string" },
124
+ description: { type: "string" },
125
+ run_in_background: { type: "boolean" },
126
+ max_turns: { type: "number" },
127
+ model: { type: "string" },
128
+ thinking: { type: "string" },
129
+ },
130
+ },
131
+ };
132
+
133
+ assert.equal(toolDef.name, "spawn_helper");
134
+ assert.equal(toolDef.parameters.required.length, 3);
135
+ assert.ok(toolDef.parameters.properties.type);
136
+ assert.ok(toolDef.parameters.properties.prompt);
137
+ assert.ok(toolDef.parameters.properties.description);
138
+ });
139
+
140
+ it("should define get_helper_result tool with correct parameters", () => {
141
+ const toolDef = {
142
+ name: "get_helper_result",
143
+ description: "Check status and retrieve results from a background agent.",
144
+ parameters: {
145
+ type: "object",
146
+ required: ["agent_id"],
147
+ properties: {
148
+ agent_id: { type: "string" },
149
+ wait: { type: "boolean" },
150
+ },
151
+ },
152
+ };
153
+
154
+ assert.equal(toolDef.name, "get_helper_result");
155
+ assert.equal(toolDef.parameters.required.length, 1);
156
+ assert.ok(toolDef.parameters.properties.agent_id);
157
+ });
158
+ });
159
+
160
+ describe("Agent Types", () => {
161
+ it("should have explore and work as builtin types", () => {
162
+ assert.deepEqual(BUILTIN_TYPES, ["explore", "work"]);
163
+ });
164
+
165
+ it("should define explore agent with read-only tools", () => {
166
+ const exploreConfig: AgentConfig = {
167
+ name: "explore",
168
+ description: "Fast parallel codebase exploration",
169
+ builtinToolNames: ["read", "bash", "grep", "find", "ls"],
170
+ systemPrompt: "You are a read-only exploration agent.",
171
+ promptMode: "replace",
172
+ extensions: true,
173
+ skills: true,
174
+ };
175
+
176
+ assert.ok(!exploreConfig.builtinToolNames?.includes("write"));
177
+ assert.ok(!exploreConfig.builtinToolNames?.includes("edit"));
178
+ assert.ok(exploreConfig.builtinToolNames?.includes("read"));
179
+ });
180
+
181
+ it("should define work agent with read-write tools", () => {
182
+ const workConfig: AgentConfig = {
183
+ name: "work",
184
+ description: "Parallel file writes with transparent locking",
185
+ builtinToolNames: ["read", "write", "edit", "bash", "grep", "find", "ls"],
186
+ systemPrompt: "You are a read-write work agent.",
187
+ promptMode: "replace",
188
+ extensions: true,
189
+ skills: true,
190
+ };
191
+
192
+ assert.ok(workConfig.builtinToolNames?.includes("write"));
193
+ assert.ok(workConfig.builtinToolNames?.includes("edit"));
194
+ assert.ok(workConfig.builtinToolNames?.includes("read"));
195
+ });
196
+ });
197
+
198
+ describe("System Prompt Builder", () => {
199
+ const env = {
200
+ isGitRepo: true,
201
+ branch: "main",
202
+ platform: "GitHub",
203
+ };
204
+
205
+ it("should build replace mode prompt", () => {
206
+ const config: AgentConfig = {
207
+ name: "explore",
208
+ displayName: "Explorer",
209
+ description: "Test",
210
+ systemPrompt: "Find all authentication files.",
211
+ promptMode: "replace",
212
+ extensions: true,
213
+ skills: true,
214
+ };
215
+
216
+ const prompt = buildAgentPrompt(config, "/workspace", env, "Parent prompt");
217
+
218
+ assert.ok(prompt.startsWith("# Explorer"));
219
+ assert.ok(prompt.includes("Find all authentication files."));
220
+ assert.ok(prompt.includes("Working directory: /workspace"));
221
+ assert.ok(!prompt.includes("Parent prompt"));
222
+ });
223
+
224
+ it("should build append mode prompt", () => {
225
+ const config: AgentConfig = {
226
+ name: "work",
227
+ displayName: "Worker",
228
+ description: "Test",
229
+ systemPrompt: "Refactor the auth module.",
230
+ promptMode: "append",
231
+ extensions: true,
232
+ skills: true,
233
+ };
234
+
235
+ const prompt = buildAgentPrompt(config, "/workspace", env, "Parent prompt");
236
+
237
+ assert.ok(prompt.startsWith("Parent prompt"));
238
+ assert.ok(prompt.includes("## Agent Role: Worker"));
239
+ assert.ok(prompt.includes("Refactor the auth module."));
240
+ });
241
+ });
242
+
243
+ describe("Concurrency Limit", () => {
244
+ it("should respect max concurrent limit", async () => {
245
+ const manager = new ConcurrencyManager(2);
246
+ const events: string[] = [];
247
+
248
+ // Start 3 agents, only 2 should run immediately
249
+ const release1 = await manager.acquire("agent-1");
250
+ events.push("agent-1-started");
251
+
252
+ const release2 = await manager.acquire("agent-2");
253
+ events.push("agent-2-started");
254
+
255
+ assert.equal(manager.getRunning(), 2);
256
+ assert.deepEqual(events, ["agent-1-started", "agent-2-started"]);
257
+
258
+ // Third agent should queue
259
+ const acquire3Promise = manager.acquire("agent-3").then((release) => {
260
+ events.push("agent-3-started");
261
+ return release;
262
+ });
263
+
264
+ assert.equal(manager.getQueueLength(), 1);
265
+
266
+ // Release first agent
267
+ release1();
268
+ const release3 = await acquire3Promise;
269
+
270
+ assert.equal(manager.getRunning(), 2);
271
+ assert.deepEqual(events, ["agent-1-started", "agent-2-started", "agent-3-started"]);
272
+
273
+ release2();
274
+ release3();
275
+ });
276
+
277
+ it("should queue agents in order", async () => {
278
+ const manager = new ConcurrencyManager(1);
279
+ const events: string[] = [];
280
+
281
+ const release1 = await manager.acquire("agent-1");
282
+ events.push("1");
283
+
284
+ const p2 = manager.acquire("agent-2").then(r => { events.push("2"); return r; });
285
+ const p3 = manager.acquire("agent-3").then(r => { events.push("3"); return r; });
286
+
287
+ release1();
288
+ const release2 = await p2;
289
+ release2();
290
+ const release3 = await p3;
291
+
292
+ assert.deepEqual(events, ["1", "2", "3"]);
293
+ release3();
294
+ });
295
+ });
296
+
297
+ describe("Custom Agent Loading", () => {
298
+ it("should parse agent markdown frontmatter", () => {
299
+ const markdown = `---
300
+ name: code-checker
301
+ description: Code quality checker
302
+ tools: read, grep, find, bash
303
+ thinking: high
304
+ ---
305
+ You are a code quality checker. Review code for issues.`;
306
+
307
+ // Simple frontmatter parser
308
+ const match = markdown.match(/^---\n([\s\S]*?)\n---\n([\s\S]*)$/);
309
+ assert.ok(match);
310
+
311
+ const frontmatter = match![1];
312
+ const body = match![2];
313
+
314
+ assert.ok(frontmatter.includes("name: code-checker"));
315
+ assert.ok(frontmatter.includes("tools: read, grep, find, bash"));
316
+ assert.ok(body.includes("You are a code quality checker."));
317
+ });
318
+
319
+ it("should validate required fields", () => {
320
+ const validAgent = {
321
+ name: "test-agent",
322
+ description: "Test agent",
323
+ systemPrompt: "Do something.",
324
+ promptMode: "replace" as const,
325
+ extensions: true as const,
326
+ skills: true as const,
327
+ };
328
+
329
+ assert.ok(validAgent.name);
330
+ assert.ok(validAgent.description);
331
+ assert.ok(validAgent.systemPrompt);
332
+ });
333
+ });
334
+ });