@herdctl/core 0.0.1 → 0.0.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.
Files changed (275) hide show
  1. package/dist/config/__tests__/agent.test.js +31 -13
  2. package/dist/config/__tests__/agent.test.js.map +1 -1
  3. package/dist/config/__tests__/merge.test.js +9 -2
  4. package/dist/config/__tests__/merge.test.js.map +1 -1
  5. package/dist/config/__tests__/schema.test.js +350 -1
  6. package/dist/config/__tests__/schema.test.js.map +1 -1
  7. package/dist/config/index.d.ts +1 -1
  8. package/dist/config/index.d.ts.map +1 -1
  9. package/dist/config/index.js +3 -1
  10. package/dist/config/index.js.map +1 -1
  11. package/dist/config/schema.d.ts +828 -24
  12. package/dist/config/schema.d.ts.map +1 -1
  13. package/dist/config/schema.js +118 -6
  14. package/dist/config/schema.js.map +1 -1
  15. package/dist/fleet-manager/__tests__/coverage.test.js +11 -332
  16. package/dist/fleet-manager/__tests__/coverage.test.js.map +1 -1
  17. package/dist/fleet-manager/__tests__/errors.test.js +1 -49
  18. package/dist/fleet-manager/__tests__/errors.test.js.map +1 -1
  19. package/dist/fleet-manager/__tests__/integration.test.js +109 -0
  20. package/dist/fleet-manager/__tests__/integration.test.js.map +1 -1
  21. package/dist/fleet-manager/__tests__/reload.test.js +1 -1
  22. package/dist/fleet-manager/__tests__/reload.test.js.map +1 -1
  23. package/dist/fleet-manager/config-reload.d.ts +164 -0
  24. package/dist/fleet-manager/config-reload.d.ts.map +1 -0
  25. package/dist/fleet-manager/config-reload.js +445 -0
  26. package/dist/fleet-manager/config-reload.js.map +1 -0
  27. package/dist/fleet-manager/context.d.ts +76 -0
  28. package/dist/fleet-manager/context.d.ts.map +1 -0
  29. package/dist/fleet-manager/context.js +11 -0
  30. package/dist/fleet-manager/context.js.map +1 -0
  31. package/dist/fleet-manager/errors.d.ts +0 -25
  32. package/dist/fleet-manager/errors.d.ts.map +1 -1
  33. package/dist/fleet-manager/errors.js +0 -38
  34. package/dist/fleet-manager/errors.js.map +1 -1
  35. package/dist/fleet-manager/event-emitters.d.ts +123 -0
  36. package/dist/fleet-manager/event-emitters.d.ts.map +1 -0
  37. package/dist/fleet-manager/event-emitters.js +136 -0
  38. package/dist/fleet-manager/event-emitters.js.map +1 -0
  39. package/dist/fleet-manager/event-types.d.ts +0 -15
  40. package/dist/fleet-manager/event-types.d.ts.map +1 -1
  41. package/dist/fleet-manager/fleet-manager.d.ts +40 -653
  42. package/dist/fleet-manager/fleet-manager.d.ts.map +1 -1
  43. package/dist/fleet-manager/fleet-manager.js +95 -1720
  44. package/dist/fleet-manager/fleet-manager.js.map +1 -1
  45. package/dist/fleet-manager/index.d.ts +13 -2
  46. package/dist/fleet-manager/index.d.ts.map +1 -1
  47. package/dist/fleet-manager/index.js +19 -6
  48. package/dist/fleet-manager/index.js.map +1 -1
  49. package/dist/fleet-manager/job-control.d.ts +64 -0
  50. package/dist/fleet-manager/job-control.d.ts.map +1 -0
  51. package/dist/fleet-manager/job-control.js +296 -0
  52. package/dist/fleet-manager/job-control.js.map +1 -0
  53. package/dist/fleet-manager/log-streaming.d.ts +171 -0
  54. package/dist/fleet-manager/log-streaming.d.ts.map +1 -0
  55. package/dist/fleet-manager/log-streaming.js +503 -0
  56. package/dist/fleet-manager/log-streaming.js.map +1 -0
  57. package/dist/fleet-manager/schedule-executor.d.ts +63 -0
  58. package/dist/fleet-manager/schedule-executor.d.ts.map +1 -0
  59. package/dist/fleet-manager/schedule-executor.js +209 -0
  60. package/dist/fleet-manager/schedule-executor.js.map +1 -0
  61. package/dist/fleet-manager/schedule-management.d.ts +71 -0
  62. package/dist/fleet-manager/schedule-management.d.ts.map +1 -0
  63. package/dist/fleet-manager/schedule-management.js +171 -0
  64. package/dist/fleet-manager/schedule-management.js.map +1 -0
  65. package/dist/fleet-manager/status-queries.d.ts +105 -0
  66. package/dist/fleet-manager/status-queries.d.ts.map +1 -0
  67. package/dist/fleet-manager/status-queries.js +247 -0
  68. package/dist/fleet-manager/status-queries.js.map +1 -0
  69. package/dist/fleet-manager/types.d.ts +0 -39
  70. package/dist/fleet-manager/types.d.ts.map +1 -1
  71. package/dist/runner/__tests__/job-executor.test.js +206 -1
  72. package/dist/runner/__tests__/job-executor.test.js.map +1 -1
  73. package/dist/runner/job-executor.d.ts +9 -0
  74. package/dist/runner/job-executor.d.ts.map +1 -1
  75. package/dist/runner/job-executor.js +78 -4
  76. package/dist/runner/job-executor.js.map +1 -1
  77. package/dist/runner/types.d.ts +2 -0
  78. package/dist/runner/types.d.ts.map +1 -1
  79. package/dist/scheduler/__tests__/cron.test.d.ts +2 -0
  80. package/dist/scheduler/__tests__/cron.test.d.ts.map +1 -0
  81. package/dist/scheduler/__tests__/cron.test.js +867 -0
  82. package/dist/scheduler/__tests__/cron.test.js.map +1 -0
  83. package/dist/scheduler/__tests__/scheduler.test.js +164 -5
  84. package/dist/scheduler/__tests__/scheduler.test.js.map +1 -1
  85. package/dist/scheduler/cron.d.ts +126 -0
  86. package/dist/scheduler/cron.d.ts.map +1 -0
  87. package/dist/scheduler/cron.js +390 -0
  88. package/dist/scheduler/cron.js.map +1 -0
  89. package/dist/scheduler/errors.d.ts +81 -1
  90. package/dist/scheduler/errors.d.ts.map +1 -1
  91. package/dist/scheduler/errors.js +81 -6
  92. package/dist/scheduler/errors.js.map +1 -1
  93. package/dist/scheduler/index.d.ts +1 -0
  94. package/dist/scheduler/index.d.ts.map +1 -1
  95. package/dist/scheduler/index.js +2 -0
  96. package/dist/scheduler/index.js.map +1 -1
  97. package/dist/scheduler/schedule-runner.d.ts +2 -2
  98. package/dist/scheduler/schedule-runner.d.ts.map +1 -1
  99. package/dist/scheduler/schedule-runner.js +20 -8
  100. package/dist/scheduler/schedule-runner.js.map +1 -1
  101. package/dist/scheduler/scheduler.d.ts +4 -4
  102. package/dist/scheduler/scheduler.d.ts.map +1 -1
  103. package/dist/scheduler/scheduler.js +86 -20
  104. package/dist/scheduler/scheduler.js.map +1 -1
  105. package/dist/scheduler/types.d.ts +1 -1
  106. package/dist/scheduler/types.d.ts.map +1 -1
  107. package/dist/state/schemas/job-metadata.d.ts +2 -2
  108. package/package.json +33 -8
  109. package/.turbo/turbo-build.log +0 -4
  110. package/.turbo/turbo-test.log +0 -219
  111. package/.turbo/turbo-typecheck.log +0 -4
  112. package/coverage/base.css +0 -224
  113. package/coverage/block-navigation.js +0 -87
  114. package/coverage/coverage-final.json +0 -51
  115. package/coverage/favicon.png +0 -0
  116. package/coverage/index.html +0 -251
  117. package/coverage/prettify.css +0 -1
  118. package/coverage/prettify.js +0 -2
  119. package/coverage/sort-arrow-sprite.png +0 -0
  120. package/coverage/sorter.js +0 -210
  121. package/coverage/src/config/index.html +0 -191
  122. package/coverage/src/config/index.ts.html +0 -442
  123. package/coverage/src/config/interpolate.ts.html +0 -652
  124. package/coverage/src/config/loader.ts.html +0 -1501
  125. package/coverage/src/config/merge.ts.html +0 -823
  126. package/coverage/src/config/parser.ts.html +0 -1213
  127. package/coverage/src/config/schema.ts.html +0 -1123
  128. package/coverage/src/fleet-manager/errors.ts.html +0 -2326
  129. package/coverage/src/fleet-manager/event-types.ts.html +0 -1219
  130. package/coverage/src/fleet-manager/fleet-manager.ts.html +0 -7030
  131. package/coverage/src/fleet-manager/index.html +0 -206
  132. package/coverage/src/fleet-manager/index.ts.html +0 -469
  133. package/coverage/src/fleet-manager/job-manager.ts.html +0 -2074
  134. package/coverage/src/fleet-manager/job-queue.ts.html +0 -2479
  135. package/coverage/src/fleet-manager/types.ts.html +0 -2602
  136. package/coverage/src/index.html +0 -116
  137. package/coverage/src/index.ts.html +0 -181
  138. package/coverage/src/runner/errors.ts.html +0 -1006
  139. package/coverage/src/runner/index.html +0 -191
  140. package/coverage/src/runner/index.ts.html +0 -256
  141. package/coverage/src/runner/job-executor.ts.html +0 -1429
  142. package/coverage/src/runner/message-processor.ts.html +0 -1150
  143. package/coverage/src/runner/sdk-adapter.ts.html +0 -658
  144. package/coverage/src/runner/types.ts.html +0 -559
  145. package/coverage/src/scheduler/errors.ts.html +0 -388
  146. package/coverage/src/scheduler/index.html +0 -206
  147. package/coverage/src/scheduler/index.ts.html +0 -244
  148. package/coverage/src/scheduler/interval.ts.html +0 -652
  149. package/coverage/src/scheduler/schedule-runner.ts.html +0 -1411
  150. package/coverage/src/scheduler/schedule-state.ts.html +0 -718
  151. package/coverage/src/scheduler/scheduler.ts.html +0 -1795
  152. package/coverage/src/scheduler/types.ts.html +0 -733
  153. package/coverage/src/state/directory.ts.html +0 -736
  154. package/coverage/src/state/errors.ts.html +0 -376
  155. package/coverage/src/state/fleet-state.ts.html +0 -937
  156. package/coverage/src/state/index.html +0 -221
  157. package/coverage/src/state/index.ts.html +0 -322
  158. package/coverage/src/state/job-metadata.ts.html +0 -1420
  159. package/coverage/src/state/job-output.ts.html +0 -1033
  160. package/coverage/src/state/schemas/fleet-state.ts.html +0 -445
  161. package/coverage/src/state/schemas/index.html +0 -176
  162. package/coverage/src/state/schemas/index.ts.html +0 -286
  163. package/coverage/src/state/schemas/job-metadata.ts.html +0 -628
  164. package/coverage/src/state/schemas/job-output.ts.html +0 -616
  165. package/coverage/src/state/schemas/session-info.ts.html +0 -361
  166. package/coverage/src/state/session.ts.html +0 -844
  167. package/coverage/src/state/types.ts.html +0 -262
  168. package/coverage/src/state/utils/atomic.ts.html +0 -748
  169. package/coverage/src/state/utils/index.html +0 -146
  170. package/coverage/src/state/utils/index.ts.html +0 -103
  171. package/coverage/src/state/utils/reads.ts.html +0 -1621
  172. package/coverage/src/work-sources/adapters/github.ts.html +0 -3583
  173. package/coverage/src/work-sources/adapters/index.html +0 -131
  174. package/coverage/src/work-sources/adapters/index.ts.html +0 -277
  175. package/coverage/src/work-sources/errors.ts.html +0 -298
  176. package/coverage/src/work-sources/index.html +0 -176
  177. package/coverage/src/work-sources/index.ts.html +0 -529
  178. package/coverage/src/work-sources/manager.ts.html +0 -1324
  179. package/coverage/src/work-sources/registry.ts.html +0 -619
  180. package/coverage/src/work-sources/types.ts.html +0 -568
  181. package/dist/fleet-manager/__tests__/event-helpers.test.d.ts +0 -7
  182. package/dist/fleet-manager/__tests__/event-helpers.test.d.ts.map +0 -1
  183. package/dist/fleet-manager/__tests__/event-helpers.test.js +0 -368
  184. package/dist/fleet-manager/__tests__/event-helpers.test.js.map +0 -1
  185. package/src/config/__tests__/agent.test.ts +0 -864
  186. package/src/config/__tests__/interpolate.test.ts +0 -644
  187. package/src/config/__tests__/loader.test.ts +0 -784
  188. package/src/config/__tests__/merge.test.ts +0 -751
  189. package/src/config/__tests__/parser.test.ts +0 -533
  190. package/src/config/__tests__/schema.test.ts +0 -873
  191. package/src/config/index.ts +0 -119
  192. package/src/config/interpolate.ts +0 -189
  193. package/src/config/loader.ts +0 -472
  194. package/src/config/merge.ts +0 -246
  195. package/src/config/parser.ts +0 -376
  196. package/src/config/schema.ts +0 -346
  197. package/src/fleet-manager/__tests__/coverage.test.ts +0 -2869
  198. package/src/fleet-manager/__tests__/errors.test.ts +0 -660
  199. package/src/fleet-manager/__tests__/event-helpers.test.ts +0 -448
  200. package/src/fleet-manager/__tests__/integration.test.ts +0 -1209
  201. package/src/fleet-manager/__tests__/job-control.test.ts +0 -283
  202. package/src/fleet-manager/__tests__/job-manager.test.ts +0 -869
  203. package/src/fleet-manager/__tests__/job-queue.test.ts +0 -401
  204. package/src/fleet-manager/__tests__/reload.test.ts +0 -751
  205. package/src/fleet-manager/__tests__/status-queries.test.ts +0 -595
  206. package/src/fleet-manager/__tests__/trigger.test.ts +0 -601
  207. package/src/fleet-manager/errors.ts +0 -747
  208. package/src/fleet-manager/event-types.ts +0 -378
  209. package/src/fleet-manager/fleet-manager.ts +0 -2315
  210. package/src/fleet-manager/index.ts +0 -128
  211. package/src/fleet-manager/job-manager.ts +0 -663
  212. package/src/fleet-manager/job-queue.ts +0 -798
  213. package/src/fleet-manager/types.ts +0 -839
  214. package/src/index.ts +0 -32
  215. package/src/runner/__tests__/errors.test.ts +0 -382
  216. package/src/runner/__tests__/job-executor.test.ts +0 -1708
  217. package/src/runner/__tests__/message-processor.test.ts +0 -960
  218. package/src/runner/__tests__/sdk-adapter.test.ts +0 -626
  219. package/src/runner/errors.ts +0 -307
  220. package/src/runner/index.ts +0 -57
  221. package/src/runner/job-executor.ts +0 -448
  222. package/src/runner/message-processor.ts +0 -355
  223. package/src/runner/sdk-adapter.ts +0 -191
  224. package/src/runner/types.ts +0 -158
  225. package/src/scheduler/__tests__/errors.test.ts +0 -159
  226. package/src/scheduler/__tests__/interval.test.ts +0 -515
  227. package/src/scheduler/__tests__/schedule-runner.test.ts +0 -798
  228. package/src/scheduler/__tests__/schedule-state.test.ts +0 -671
  229. package/src/scheduler/__tests__/scheduler.test.ts +0 -1280
  230. package/src/scheduler/errors.ts +0 -101
  231. package/src/scheduler/index.ts +0 -53
  232. package/src/scheduler/interval.ts +0 -189
  233. package/src/scheduler/schedule-runner.ts +0 -442
  234. package/src/scheduler/schedule-state.ts +0 -211
  235. package/src/scheduler/scheduler.ts +0 -570
  236. package/src/scheduler/types.ts +0 -216
  237. package/src/state/__tests__/directory.test.ts +0 -595
  238. package/src/state/__tests__/fleet-state.test.ts +0 -868
  239. package/src/state/__tests__/job-metadata-schema.test.ts +0 -414
  240. package/src/state/__tests__/job-metadata.test.ts +0 -831
  241. package/src/state/__tests__/job-output.test.ts +0 -856
  242. package/src/state/__tests__/session-schema.test.ts +0 -378
  243. package/src/state/__tests__/session.test.ts +0 -604
  244. package/src/state/directory.ts +0 -217
  245. package/src/state/errors.ts +0 -97
  246. package/src/state/fleet-state.ts +0 -284
  247. package/src/state/index.ts +0 -79
  248. package/src/state/job-metadata.ts +0 -445
  249. package/src/state/job-output.ts +0 -316
  250. package/src/state/schemas/__tests__/job-output.test.ts +0 -338
  251. package/src/state/schemas/fleet-state.ts +0 -120
  252. package/src/state/schemas/index.ts +0 -67
  253. package/src/state/schemas/job-metadata.ts +0 -181
  254. package/src/state/schemas/job-output.ts +0 -177
  255. package/src/state/schemas/session-info.ts +0 -92
  256. package/src/state/session.ts +0 -253
  257. package/src/state/types.ts +0 -59
  258. package/src/state/utils/__tests__/atomic.test.ts +0 -723
  259. package/src/state/utils/__tests__/reads.test.ts +0 -1071
  260. package/src/state/utils/atomic.ts +0 -221
  261. package/src/state/utils/index.ts +0 -6
  262. package/src/state/utils/reads.ts +0 -512
  263. package/src/work-sources/__tests__/github.test.ts +0 -1800
  264. package/src/work-sources/__tests__/manager.test.ts +0 -529
  265. package/src/work-sources/__tests__/registry.test.ts +0 -477
  266. package/src/work-sources/__tests__/types.test.ts +0 -479
  267. package/src/work-sources/adapters/github.ts +0 -1166
  268. package/src/work-sources/adapters/index.ts +0 -64
  269. package/src/work-sources/errors.ts +0 -71
  270. package/src/work-sources/index.ts +0 -148
  271. package/src/work-sources/manager.ts +0 -413
  272. package/src/work-sources/registry.ts +0 -178
  273. package/src/work-sources/types.ts +0 -161
  274. package/tsconfig.json +0 -9
  275. package/vitest.config.ts +0 -19
@@ -1,1209 +0,0 @@
1
- /**
2
- * Integration tests for FleetManager (US-13)
3
- *
4
- * Comprehensive integration tests covering:
5
- * - Full flow: initialize → start → trigger → complete → stop
6
- * - Scheduler integration: schedules trigger jobs correctly
7
- * - State persistence: survives restart with correct state
8
- * - Edge cases: start when running, stop when stopped, etc.
9
- */
10
-
11
- import { describe, it, expect, beforeEach, afterEach, vi } from "vitest";
12
- import { mkdtemp, rm, mkdir, writeFile, readFile } from "fs/promises";
13
- import { tmpdir } from "os";
14
- import { join } from "path";
15
- import { FleetManager } from "../fleet-manager.js";
16
- import {
17
- InvalidStateError,
18
- AgentNotFoundError,
19
- ScheduleNotFoundError,
20
- } from "../errors.js";
21
- import type {
22
- FleetManagerLogger,
23
- JobCreatedPayload,
24
- JobCompletedPayload,
25
- ScheduleTriggeredPayload,
26
- } from "../types.js";
27
-
28
- describe("FleetManager Integration Tests (US-13)", () => {
29
- let tempDir: string;
30
- let configDir: string;
31
- let stateDir: string;
32
-
33
- beforeEach(async () => {
34
- tempDir = await mkdtemp(join(tmpdir(), "fleet-integration-test-"));
35
- configDir = join(tempDir, "config");
36
- stateDir = join(tempDir, ".herdctl");
37
- await mkdir(configDir, { recursive: true });
38
- });
39
-
40
- afterEach(async () => {
41
- await rm(tempDir, { recursive: true, force: true });
42
- });
43
-
44
- // Helper to create a test config file
45
- async function createConfig(config: object) {
46
- const configPath = join(configDir, "herdctl.yaml");
47
- const yaml = await import("yaml");
48
- await writeFile(configPath, yaml.stringify(config));
49
- return configPath;
50
- }
51
-
52
- // Helper to create an agent config file
53
- async function createAgentConfig(name: string, config: object) {
54
- const agentDir = join(configDir, "agents");
55
- await mkdir(agentDir, { recursive: true });
56
- const agentPath = join(agentDir, `${name}.yaml`);
57
- const yaml = await import("yaml");
58
- await writeFile(agentPath, yaml.stringify(config));
59
- return agentPath;
60
- }
61
-
62
- // Create a silent logger for tests
63
- function createSilentLogger(): FleetManagerLogger {
64
- return {
65
- debug: vi.fn(),
66
- info: vi.fn(),
67
- warn: vi.fn(),
68
- error: vi.fn(),
69
- };
70
- }
71
-
72
- // Create a test manager with common options
73
- function createTestManager(
74
- configPath: string,
75
- options: { checkInterval?: number } = {}
76
- ) {
77
- return new FleetManager({
78
- configPath,
79
- stateDir,
80
- checkInterval: options.checkInterval ?? 10000, // Long interval by default to avoid unexpected triggers
81
- logger: createSilentLogger(),
82
- });
83
- }
84
-
85
- // ==========================================================================
86
- // Full Flow Integration Tests
87
- // ==========================================================================
88
-
89
- describe("Full Flow: initialize → start → trigger → complete → stop", () => {
90
- it("completes a full lifecycle with manual trigger", async () => {
91
- // Setup: Create agent config
92
- await createAgentConfig("workflow-agent", {
93
- name: "workflow-agent",
94
- description: "Agent for testing full workflow",
95
- schedules: {
96
- hourly: {
97
- type: "interval",
98
- interval: "1h",
99
- prompt: "Check hourly tasks",
100
- },
101
- },
102
- });
103
-
104
- const configPath = await createConfig({
105
- version: 1,
106
- agents: [{ path: "./agents/workflow-agent.yaml" }],
107
- });
108
-
109
- // Create manager
110
- const manager = createTestManager(configPath);
111
- const events: string[] = [];
112
-
113
- // Track all events
114
- manager.on("initialized", () => events.push("initialized"));
115
- manager.on("started", () => events.push("started"));
116
- manager.on("stopped", () => events.push("stopped"));
117
- manager.on("job:created", () => events.push("job:created"));
118
-
119
- // 1. Verify initial state
120
- expect(manager.state.status).toBe("uninitialized");
121
-
122
- // 2. Initialize
123
- await manager.initialize();
124
- expect(manager.state.status).toBe("initialized");
125
- expect(manager.state.agentCount).toBe(1);
126
- expect(events).toContain("initialized");
127
-
128
- // 3. Start
129
- await manager.start();
130
- await new Promise((resolve) => setTimeout(resolve, 50)); // Wait for async start
131
- expect(manager.state.status).toBe("running");
132
- expect(events).toContain("started");
133
-
134
- // 4. Trigger agent
135
- const result = await manager.trigger("workflow-agent", "hourly");
136
- expect(result.agentName).toBe("workflow-agent");
137
- expect(result.scheduleName).toBe("hourly");
138
- expect(result.prompt).toBe("Check hourly tasks");
139
- expect(result.jobId).toMatch(/^job-\d{4}-\d{2}-\d{2}-[a-z0-9]{6}$/);
140
- expect(events).toContain("job:created");
141
-
142
- // 5. Verify fleet status while running
143
- const fleetStatus = await manager.getFleetStatus();
144
- expect(fleetStatus.state).toBe("running");
145
- expect(fleetStatus.counts.totalAgents).toBe(1);
146
- expect(fleetStatus.scheduler.status).toBe("running");
147
-
148
- // 6. Stop
149
- await manager.stop();
150
- expect(manager.state.status).toBe("stopped");
151
- expect(events).toContain("stopped");
152
-
153
- // 7. Verify final state
154
- const finalStatus = await manager.getFleetStatus();
155
- expect(finalStatus.state).toBe("stopped");
156
- expect(finalStatus.stoppedAt).not.toBeNull();
157
- });
158
-
159
- it("emits events in correct order during lifecycle", async () => {
160
- await createAgentConfig("event-order-agent", {
161
- name: "event-order-agent",
162
- });
163
-
164
- const configPath = await createConfig({
165
- version: 1,
166
- agents: [{ path: "./agents/event-order-agent.yaml" }],
167
- });
168
-
169
- const manager = createTestManager(configPath);
170
- const eventOrder: string[] = [];
171
-
172
- manager.on("initialized", () => eventOrder.push("initialized"));
173
- manager.on("started", () => eventOrder.push("started"));
174
- manager.on("job:created", () => eventOrder.push("job:created"));
175
- manager.on("stopped", () => eventOrder.push("stopped"));
176
-
177
- await manager.initialize();
178
- await manager.start();
179
- await new Promise((resolve) => setTimeout(resolve, 50));
180
- await manager.trigger("event-order-agent");
181
- await manager.stop();
182
-
183
- expect(eventOrder).toEqual([
184
- "initialized",
185
- "started",
186
- "job:created",
187
- "stopped",
188
- ]);
189
- });
190
-
191
- it("handles multiple agents in full workflow", async () => {
192
- await createAgentConfig("agent-alpha", {
193
- name: "agent-alpha",
194
- description: "First agent",
195
- });
196
-
197
- await createAgentConfig("agent-beta", {
198
- name: "agent-beta",
199
- description: "Second agent",
200
- });
201
-
202
- await createAgentConfig("agent-gamma", {
203
- name: "agent-gamma",
204
- description: "Third agent",
205
- });
206
-
207
- const configPath = await createConfig({
208
- version: 1,
209
- agents: [
210
- { path: "./agents/agent-alpha.yaml" },
211
- { path: "./agents/agent-beta.yaml" },
212
- { path: "./agents/agent-gamma.yaml" },
213
- ],
214
- });
215
-
216
- const manager = createTestManager(configPath);
217
-
218
- await manager.initialize();
219
- expect(manager.state.agentCount).toBe(3);
220
-
221
- await manager.start();
222
- await new Promise((resolve) => setTimeout(resolve, 50));
223
-
224
- // Trigger all agents
225
- const results = await Promise.all([
226
- manager.trigger("agent-alpha"),
227
- manager.trigger("agent-beta"),
228
- manager.trigger("agent-gamma"),
229
- ]);
230
-
231
- expect(results).toHaveLength(3);
232
- expect(results.map((r) => r.agentName).sort()).toEqual([
233
- "agent-alpha",
234
- "agent-beta",
235
- "agent-gamma",
236
- ]);
237
-
238
- // Verify all have unique job IDs
239
- const jobIds = results.map((r) => r.jobId);
240
- expect(new Set(jobIds).size).toBe(3);
241
-
242
- // Verify fleet status
243
- const status = await manager.getFleetStatus();
244
- expect(status.counts.totalAgents).toBe(3);
245
-
246
- await manager.stop();
247
- });
248
-
249
- it("correctly tracks timing through lifecycle", async () => {
250
- await createAgentConfig("timing-agent", {
251
- name: "timing-agent",
252
- });
253
-
254
- const configPath = await createConfig({
255
- version: 1,
256
- agents: [{ path: "./agents/timing-agent.yaml" }],
257
- });
258
-
259
- const manager = createTestManager(configPath);
260
-
261
- // Before init
262
- const beforeInit = new Date().toISOString();
263
- expect(manager.state.initializedAt).toBeNull();
264
- expect(manager.state.startedAt).toBeNull();
265
- expect(manager.state.stoppedAt).toBeNull();
266
-
267
- await manager.initialize();
268
- const afterInit = new Date().toISOString();
269
-
270
- expect(manager.state.initializedAt).not.toBeNull();
271
- expect(manager.state.initializedAt! >= beforeInit).toBe(true);
272
- expect(manager.state.initializedAt! <= afterInit).toBe(true);
273
-
274
- const beforeStart = new Date().toISOString();
275
- await manager.start();
276
- await new Promise((resolve) => setTimeout(resolve, 50));
277
- const afterStart = new Date().toISOString();
278
-
279
- expect(manager.state.startedAt).not.toBeNull();
280
- expect(manager.state.startedAt! >= beforeStart).toBe(true);
281
- expect(manager.state.startedAt! <= afterStart).toBe(true);
282
-
283
- // Check uptime
284
- const status = await manager.getFleetStatus();
285
- expect(status.uptimeSeconds).toBeGreaterThanOrEqual(0);
286
-
287
- const beforeStop = new Date().toISOString();
288
- await manager.stop();
289
- const afterStop = new Date().toISOString();
290
-
291
- expect(manager.state.stoppedAt).not.toBeNull();
292
- expect(manager.state.stoppedAt! >= beforeStop).toBe(true);
293
- expect(manager.state.stoppedAt! <= afterStop).toBe(true);
294
- });
295
- });
296
-
297
- // ==========================================================================
298
- // Scheduler Integration Tests
299
- // ==========================================================================
300
-
301
- describe("Scheduler Integration", () => {
302
- it("scheduler triggers jobs on schedule", async () => {
303
- await createAgentConfig("scheduled-agent", {
304
- name: "scheduled-agent",
305
- schedules: {
306
- frequent: {
307
- type: "interval",
308
- interval: "100ms", // Very short for testing
309
- prompt: "Scheduled prompt",
310
- },
311
- },
312
- });
313
-
314
- const configPath = await createConfig({
315
- version: 1,
316
- agents: [{ path: "./agents/scheduled-agent.yaml" }],
317
- });
318
-
319
- const manager = new FleetManager({
320
- configPath,
321
- stateDir,
322
- checkInterval: 50, // Check frequently for testing
323
- logger: createSilentLogger(),
324
- });
325
-
326
- const scheduleTriggers: ScheduleTriggeredPayload[] = [];
327
- manager.on("schedule:triggered", (payload) => {
328
- scheduleTriggers.push(payload);
329
- });
330
-
331
- await manager.initialize();
332
- await manager.start();
333
-
334
- // Wait for at least one scheduled trigger
335
- await new Promise((resolve) => setTimeout(resolve, 300));
336
-
337
- await manager.stop();
338
-
339
- // Scheduler should have triggered at least once
340
- expect(scheduleTriggers.length).toBeGreaterThanOrEqual(1);
341
- expect(scheduleTriggers[0].agentName).toBe("scheduled-agent");
342
- expect(scheduleTriggers[0].scheduleName).toBe("frequent");
343
- });
344
-
345
- it("scheduler respects disabled schedules", async () => {
346
- await createAgentConfig("disabled-schedule-agent", {
347
- name: "disabled-schedule-agent",
348
- schedules: {
349
- active: {
350
- type: "interval",
351
- interval: "5s", // Longer interval to avoid race conditions
352
- },
353
- },
354
- });
355
-
356
- const configPath = await createConfig({
357
- version: 1,
358
- agents: [{ path: "./agents/disabled-schedule-agent.yaml" }],
359
- });
360
-
361
- const manager = new FleetManager({
362
- configPath,
363
- stateDir,
364
- checkInterval: 50,
365
- logger: createSilentLogger(),
366
- });
367
-
368
- await manager.initialize();
369
-
370
- // Disable the schedule BEFORE starting
371
- await manager.disableSchedule("disabled-schedule-agent", "active");
372
-
373
- // Track all triggers
374
- const triggers: string[] = [];
375
- manager.on("schedule:triggered", () => {
376
- triggers.push("triggered");
377
- });
378
-
379
- await manager.start();
380
-
381
- // Wait a bit - disabled schedule should not trigger
382
- await new Promise((resolve) => setTimeout(resolve, 150));
383
-
384
- await manager.stop();
385
-
386
- // No triggers should have occurred since schedule was disabled
387
- expect(triggers.length).toBe(0);
388
- });
389
-
390
- it("getSchedules returns correct schedule information", async () => {
391
- await createAgentConfig("multi-schedule-agent", {
392
- name: "multi-schedule-agent",
393
- schedules: {
394
- hourly: {
395
- type: "interval",
396
- interval: "1h",
397
- },
398
- daily: {
399
- type: "interval",
400
- interval: "24h",
401
- },
402
- },
403
- });
404
-
405
- const configPath = await createConfig({
406
- version: 1,
407
- agents: [{ path: "./agents/multi-schedule-agent.yaml" }],
408
- });
409
-
410
- const manager = createTestManager(configPath);
411
- await manager.initialize();
412
-
413
- const schedules = await manager.getSchedules();
414
-
415
- expect(schedules).toHaveLength(2);
416
- expect(schedules.map((s) => s.name).sort()).toEqual(["daily", "hourly"]);
417
- expect(schedules.every((s) => s.agentName === "multi-schedule-agent")).toBe(
418
- true
419
- );
420
- expect(schedules.every((s) => s.status === "idle")).toBe(true);
421
- });
422
-
423
- it("getSchedule returns specific schedule", async () => {
424
- await createAgentConfig("specific-schedule-agent", {
425
- name: "specific-schedule-agent",
426
- schedules: {
427
- target: {
428
- type: "interval",
429
- interval: "30m",
430
- },
431
- },
432
- });
433
-
434
- const configPath = await createConfig({
435
- version: 1,
436
- agents: [{ path: "./agents/specific-schedule-agent.yaml" }],
437
- });
438
-
439
- const manager = createTestManager(configPath);
440
- await manager.initialize();
441
-
442
- const schedule = await manager.getSchedule(
443
- "specific-schedule-agent",
444
- "target"
445
- );
446
-
447
- expect(schedule.name).toBe("target");
448
- expect(schedule.agentName).toBe("specific-schedule-agent");
449
- expect(schedule.type).toBe("interval");
450
- expect(schedule.interval).toBe("30m");
451
- });
452
-
453
- it("enableSchedule and disableSchedule toggle schedule status", async () => {
454
- await createAgentConfig("toggle-agent", {
455
- name: "toggle-agent",
456
- schedules: {
457
- toggleable: {
458
- type: "interval",
459
- interval: "1h",
460
- },
461
- },
462
- });
463
-
464
- const configPath = await createConfig({
465
- version: 1,
466
- agents: [{ path: "./agents/toggle-agent.yaml" }],
467
- });
468
-
469
- const manager = createTestManager(configPath);
470
- await manager.initialize();
471
-
472
- // Initially idle
473
- let schedule = await manager.getSchedule("toggle-agent", "toggleable");
474
- expect(schedule.status).toBe("idle");
475
-
476
- // Disable
477
- await manager.disableSchedule("toggle-agent", "toggleable");
478
- schedule = await manager.getSchedule("toggle-agent", "toggleable");
479
- expect(schedule.status).toBe("disabled");
480
-
481
- // Re-enable
482
- await manager.enableSchedule("toggle-agent", "toggleable");
483
- schedule = await manager.getSchedule("toggle-agent", "toggleable");
484
- expect(schedule.status).toBe("idle");
485
- });
486
- });
487
-
488
- // ==========================================================================
489
- // State Persistence Tests
490
- // ==========================================================================
491
-
492
- describe("State Persistence", () => {
493
- it("persists fleet state across restart", async () => {
494
- await createAgentConfig("persistent-agent", {
495
- name: "persistent-agent",
496
- schedules: {
497
- check: {
498
- type: "interval",
499
- interval: "1h",
500
- },
501
- },
502
- });
503
-
504
- const configPath = await createConfig({
505
- version: 1,
506
- agents: [{ path: "./agents/persistent-agent.yaml" }],
507
- });
508
-
509
- // First manager instance - trigger a job
510
- const manager1 = createTestManager(configPath);
511
- await manager1.initialize();
512
- await manager1.start();
513
- await new Promise((resolve) => setTimeout(resolve, 50));
514
-
515
- const triggerResult = await manager1.trigger("persistent-agent", "check");
516
- const jobId = triggerResult.jobId;
517
-
518
- await manager1.stop();
519
-
520
- // Second manager instance - verify state was persisted
521
- const manager2 = createTestManager(configPath);
522
- await manager2.initialize();
523
-
524
- // State should show agent info correctly
525
- const agentInfo = await manager2.getAgentInfoByName("persistent-agent");
526
- expect(agentInfo.name).toBe("persistent-agent");
527
-
528
- // Verify the job was created with metadata (stored as YAML)
529
- const yaml = await import("yaml");
530
- const jobFilePath = join(stateDir, "jobs", `${jobId}.yaml`);
531
- const jobContent = await readFile(jobFilePath, "utf-8");
532
- const metadata = yaml.parse(jobContent);
533
-
534
- expect(metadata.id).toBe(jobId);
535
- expect(metadata.agent).toBe("persistent-agent");
536
- });
537
-
538
- it("schedule state survives restart", async () => {
539
- await createAgentConfig("schedule-persist-agent", {
540
- name: "schedule-persist-agent",
541
- schedules: {
542
- persist: {
543
- type: "interval",
544
- interval: "1h",
545
- },
546
- },
547
- });
548
-
549
- const configPath = await createConfig({
550
- version: 1,
551
- agents: [{ path: "./agents/schedule-persist-agent.yaml" }],
552
- });
553
-
554
- // First instance - disable schedule
555
- const manager1 = createTestManager(configPath);
556
- await manager1.initialize();
557
- await manager1.disableSchedule("schedule-persist-agent", "persist");
558
-
559
- let schedule = await manager1.getSchedule(
560
- "schedule-persist-agent",
561
- "persist"
562
- );
563
- expect(schedule.status).toBe("disabled");
564
-
565
- // Note: The current implementation may or may not persist schedule disabled state
566
- // This test documents the expected behavior
567
-
568
- await manager1.stop();
569
-
570
- // Second instance - check if schedule state was preserved
571
- const manager2 = createTestManager(configPath);
572
- await manager2.initialize();
573
-
574
- // Get schedule state
575
- schedule = await manager2.getSchedule("schedule-persist-agent", "persist");
576
- // Schedule status after restart - depends on implementation
577
- // Currently schedules start fresh as "idle" on restart
578
- expect(["idle", "disabled"]).toContain(schedule.status);
579
- });
580
-
581
- it("job metadata persists to disk", async () => {
582
- await createAgentConfig("job-persist-agent", {
583
- name: "job-persist-agent",
584
- });
585
-
586
- const configPath = await createConfig({
587
- version: 1,
588
- agents: [{ path: "./agents/job-persist-agent.yaml" }],
589
- });
590
-
591
- const manager = createTestManager(configPath);
592
- await manager.initialize();
593
- await manager.start();
594
- await new Promise((resolve) => setTimeout(resolve, 50));
595
-
596
- const result = await manager.trigger("job-persist-agent", undefined, {
597
- prompt: "Persisted prompt",
598
- });
599
-
600
- // Verify job metadata was written to disk (stored as YAML)
601
- const yaml = await import("yaml");
602
- const jobFilePath = join(stateDir, "jobs", `${result.jobId}.yaml`);
603
-
604
- const metadataContent = await readFile(jobFilePath, "utf-8");
605
- const metadata = yaml.parse(metadataContent);
606
-
607
- expect(metadata.id).toBe(result.jobId);
608
- expect(metadata.agent).toBe("job-persist-agent");
609
- expect(metadata.prompt).toBe("Persisted prompt");
610
- expect(metadata.trigger_type).toBe("manual");
611
-
612
- await manager.stop();
613
- });
614
-
615
- it("state directory is created if it does not exist", async () => {
616
- await createAgentConfig("state-dir-agent", {
617
- name: "state-dir-agent",
618
- });
619
-
620
- const configPath = await createConfig({
621
- version: 1,
622
- agents: [{ path: "./agents/state-dir-agent.yaml" }],
623
- });
624
-
625
- // Use a new state directory that doesn't exist
626
- const newStateDir = join(tempDir, "new-state-dir");
627
-
628
- const manager = new FleetManager({
629
- configPath,
630
- stateDir: newStateDir,
631
- logger: createSilentLogger(),
632
- });
633
-
634
- await manager.initialize();
635
- await manager.start();
636
- await new Promise((resolve) => setTimeout(resolve, 50));
637
-
638
- // Trigger a job to ensure state is persisted
639
- const result = await manager.trigger("state-dir-agent");
640
-
641
- // Verify job file was created in state directory (stored as YAML)
642
- const yaml = await import("yaml");
643
- const jobFilePath = join(newStateDir, "jobs", `${result.jobId}.yaml`);
644
- const content = await readFile(jobFilePath, "utf-8");
645
- expect(yaml.parse(content)).toHaveProperty("id", result.jobId);
646
-
647
- await manager.stop();
648
- });
649
- });
650
-
651
- // ==========================================================================
652
- // Edge Case Tests
653
- // ==========================================================================
654
-
655
- describe("Edge Cases", () => {
656
- describe("start() edge cases", () => {
657
- it("throws InvalidStateError when calling start before initialize", async () => {
658
- const configPath = await createConfig({
659
- version: 1,
660
- agents: [],
661
- });
662
-
663
- const manager = createTestManager(configPath);
664
-
665
- await expect(manager.start()).rejects.toThrow(InvalidStateError);
666
- await expect(manager.start()).rejects.toMatchObject({
667
- operation: "start",
668
- currentState: "uninitialized",
669
- });
670
- });
671
-
672
- it("handles start when already running (idempotent)", async () => {
673
- await createAgentConfig("idempotent-start", {
674
- name: "idempotent-start",
675
- });
676
-
677
- const configPath = await createConfig({
678
- version: 1,
679
- agents: [{ path: "./agents/idempotent-start.yaml" }],
680
- });
681
-
682
- const manager = createTestManager(configPath);
683
- await manager.initialize();
684
- await manager.start();
685
- await new Promise((resolve) => setTimeout(resolve, 50));
686
-
687
- expect(manager.state.status).toBe("running");
688
-
689
- // Second start should be safe (idempotent or throw)
690
- // Based on implementation, may throw InvalidStateError or be no-op
691
- try {
692
- await manager.start();
693
- // If no error, should still be running
694
- expect(manager.state.status).toBe("running");
695
- } catch (error) {
696
- expect(error).toBeInstanceOf(InvalidStateError);
697
- }
698
-
699
- await manager.stop();
700
- });
701
-
702
- it("requires re-initialization to restart after stop", async () => {
703
- await createAgentConfig("restart-agent", {
704
- name: "restart-agent",
705
- });
706
-
707
- const configPath = await createConfig({
708
- version: 1,
709
- agents: [{ path: "./agents/restart-agent.yaml" }],
710
- });
711
-
712
- const manager = createTestManager(configPath);
713
- await manager.initialize();
714
-
715
- // First start/stop cycle
716
- await manager.start();
717
- await new Promise((resolve) => setTimeout(resolve, 50));
718
- expect(manager.state.status).toBe("running");
719
- await manager.stop();
720
- expect(manager.state.status).toBe("stopped");
721
-
722
- // Cannot restart without re-initialization
723
- // This documents the current behavior - must create new manager instance
724
- await expect(manager.start()).rejects.toThrow();
725
- });
726
- });
727
-
728
- describe("stop() edge cases", () => {
729
- it("handles stop when already stopped (idempotent)", async () => {
730
- await createAgentConfig("idempotent-stop", {
731
- name: "idempotent-stop",
732
- });
733
-
734
- const configPath = await createConfig({
735
- version: 1,
736
- agents: [{ path: "./agents/idempotent-stop.yaml" }],
737
- });
738
-
739
- const manager = createTestManager(configPath);
740
- await manager.initialize();
741
- await manager.start();
742
- await new Promise((resolve) => setTimeout(resolve, 50));
743
- await manager.stop();
744
-
745
- expect(manager.state.status).toBe("stopped");
746
-
747
- // Second stop should be safe
748
- await manager.stop();
749
- expect(manager.state.status).toBe("stopped");
750
- });
751
-
752
- it("handles stop when never started (no-op)", async () => {
753
- await createAgentConfig("never-started", {
754
- name: "never-started",
755
- });
756
-
757
- const configPath = await createConfig({
758
- version: 1,
759
- agents: [{ path: "./agents/never-started.yaml" }],
760
- });
761
-
762
- const manager = createTestManager(configPath);
763
- await manager.initialize();
764
-
765
- // Stop without ever starting is a no-op - stays in initialized state
766
- await manager.stop();
767
- expect(manager.state.status).toBe("initialized");
768
- });
769
-
770
- it("stop respects timeout option", async () => {
771
- await createAgentConfig("timeout-agent", {
772
- name: "timeout-agent",
773
- });
774
-
775
- const configPath = await createConfig({
776
- version: 1,
777
- agents: [{ path: "./agents/timeout-agent.yaml" }],
778
- });
779
-
780
- const manager = createTestManager(configPath);
781
- await manager.initialize();
782
- await manager.start();
783
- await new Promise((resolve) => setTimeout(resolve, 50));
784
-
785
- // Stop with short timeout
786
- const beforeStop = Date.now();
787
- await manager.stop({ timeout: 100 });
788
- const afterStop = Date.now();
789
-
790
- expect(manager.state.status).toBe("stopped");
791
- // Stop should complete quickly
792
- expect(afterStop - beforeStop).toBeLessThan(1000);
793
- });
794
- });
795
-
796
- describe("initialize() edge cases", () => {
797
- it("throws error for invalid config", async () => {
798
- const configPath = join(configDir, "herdctl.yaml");
799
- await writeFile(configPath, "invalid: yaml: content:");
800
-
801
- const manager = new FleetManager({
802
- configPath,
803
- stateDir,
804
- logger: createSilentLogger(),
805
- });
806
-
807
- await expect(manager.initialize()).rejects.toThrow();
808
- });
809
-
810
- it("throws error for non-existent config", async () => {
811
- const manager = new FleetManager({
812
- configPath: "/nonexistent/path/config.yaml",
813
- stateDir,
814
- logger: createSilentLogger(),
815
- });
816
-
817
- await expect(manager.initialize()).rejects.toThrow();
818
- });
819
-
820
- it("handles re-initialization (idempotent or error)", async () => {
821
- await createAgentConfig("reinit-agent", {
822
- name: "reinit-agent",
823
- });
824
-
825
- const configPath = await createConfig({
826
- version: 1,
827
- agents: [{ path: "./agents/reinit-agent.yaml" }],
828
- });
829
-
830
- const manager = createTestManager(configPath);
831
- await manager.initialize();
832
-
833
- // Second initialize - should be idempotent or throw
834
- try {
835
- await manager.initialize();
836
- expect(manager.state.status).toBe("initialized");
837
- } catch (error) {
838
- expect(error).toBeInstanceOf(InvalidStateError);
839
- }
840
- });
841
- });
842
-
843
- describe("trigger() edge cases", () => {
844
- it("throws AgentNotFoundError for non-existent agent", async () => {
845
- await createAgentConfig("existing-agent", {
846
- name: "existing-agent",
847
- });
848
-
849
- const configPath = await createConfig({
850
- version: 1,
851
- agents: [{ path: "./agents/existing-agent.yaml" }],
852
- });
853
-
854
- const manager = createTestManager(configPath);
855
- await manager.initialize();
856
-
857
- await expect(manager.trigger("nonexistent-agent")).rejects.toThrow(
858
- AgentNotFoundError
859
- );
860
- await expect(manager.trigger("nonexistent-agent")).rejects.toMatchObject({
861
- agentName: "nonexistent-agent",
862
- availableAgents: ["existing-agent"],
863
- });
864
- });
865
-
866
- it("throws ScheduleNotFoundError for non-existent schedule", async () => {
867
- await createAgentConfig("schedule-edge-agent", {
868
- name: "schedule-edge-agent",
869
- schedules: {
870
- existing: {
871
- type: "interval",
872
- interval: "1h",
873
- },
874
- },
875
- });
876
-
877
- const configPath = await createConfig({
878
- version: 1,
879
- agents: [{ path: "./agents/schedule-edge-agent.yaml" }],
880
- });
881
-
882
- const manager = createTestManager(configPath);
883
- await manager.initialize();
884
-
885
- await expect(
886
- manager.trigger("schedule-edge-agent", "nonexistent")
887
- ).rejects.toThrow(ScheduleNotFoundError);
888
- await expect(
889
- manager.trigger("schedule-edge-agent", "nonexistent")
890
- ).rejects.toMatchObject({
891
- scheduleName: "nonexistent",
892
- availableSchedules: ["existing"],
893
- });
894
- });
895
-
896
- it("throws InvalidStateError before initialize", async () => {
897
- const configPath = await createConfig({
898
- version: 1,
899
- agents: [],
900
- });
901
-
902
- const manager = createTestManager(configPath);
903
-
904
- await expect(manager.trigger("any-agent")).rejects.toThrow(
905
- InvalidStateError
906
- );
907
- });
908
-
909
- it("trigger works after stop", async () => {
910
- await createAgentConfig("trigger-after-stop", {
911
- name: "trigger-after-stop",
912
- });
913
-
914
- const configPath = await createConfig({
915
- version: 1,
916
- agents: [{ path: "./agents/trigger-after-stop.yaml" }],
917
- });
918
-
919
- const manager = createTestManager(configPath);
920
- await manager.initialize();
921
- await manager.start();
922
- await new Promise((resolve) => setTimeout(resolve, 50));
923
- await manager.stop();
924
-
925
- // Should still be able to trigger after stop
926
- const result = await manager.trigger("trigger-after-stop");
927
- expect(result.agentName).toBe("trigger-after-stop");
928
- });
929
- });
930
-
931
- describe("getAgentInfoByName() edge cases", () => {
932
- it("throws AgentNotFoundError before initialize", async () => {
933
- const configPath = await createConfig({
934
- version: 1,
935
- agents: [],
936
- });
937
-
938
- const manager = createTestManager(configPath);
939
-
940
- await expect(manager.getAgentInfoByName("any")).rejects.toThrow(
941
- AgentNotFoundError
942
- );
943
- });
944
-
945
- it("throws AgentNotFoundError for unknown agent", async () => {
946
- await createAgentConfig("known", {
947
- name: "known",
948
- });
949
-
950
- const configPath = await createConfig({
951
- version: 1,
952
- agents: [{ path: "./agents/known.yaml" }],
953
- });
954
-
955
- const manager = createTestManager(configPath);
956
- await manager.initialize();
957
-
958
- await expect(manager.getAgentInfoByName("unknown")).rejects.toThrow(
959
- AgentNotFoundError
960
- );
961
- });
962
- });
963
-
964
- describe("reload() edge cases", () => {
965
- it("throws InvalidStateError before initialize", async () => {
966
- const configPath = await createConfig({
967
- version: 1,
968
- agents: [],
969
- });
970
-
971
- const manager = createTestManager(configPath);
972
-
973
- await expect(manager.reload()).rejects.toThrow(InvalidStateError);
974
- });
975
-
976
- it("reload works in all valid states", async () => {
977
- await createAgentConfig("reload-states", {
978
- name: "reload-states",
979
- });
980
-
981
- const configPath = await createConfig({
982
- version: 1,
983
- agents: [{ path: "./agents/reload-states.yaml" }],
984
- });
985
-
986
- const manager = createTestManager(configPath);
987
-
988
- // Test in initialized state
989
- await manager.initialize();
990
- let result = await manager.reload();
991
- expect(result.agentCount).toBe(1);
992
-
993
- // Test in running state
994
- await manager.start();
995
- await new Promise((resolve) => setTimeout(resolve, 50));
996
- result = await manager.reload();
997
- expect(result.agentCount).toBe(1);
998
-
999
- // Test in stopped state
1000
- await manager.stop();
1001
- result = await manager.reload();
1002
- expect(result.agentCount).toBe(1);
1003
- });
1004
- });
1005
-
1006
- describe("getSchedule() edge cases", () => {
1007
- it("throws AgentNotFoundError for unknown agent", async () => {
1008
- await createAgentConfig("schedule-agent", {
1009
- name: "schedule-agent",
1010
- schedules: {
1011
- test: { type: "interval", interval: "1h" },
1012
- },
1013
- });
1014
-
1015
- const configPath = await createConfig({
1016
- version: 1,
1017
- agents: [{ path: "./agents/schedule-agent.yaml" }],
1018
- });
1019
-
1020
- const manager = createTestManager(configPath);
1021
- await manager.initialize();
1022
-
1023
- await expect(
1024
- manager.getSchedule("unknown-agent", "test")
1025
- ).rejects.toThrow(AgentNotFoundError);
1026
- });
1027
-
1028
- it("throws ScheduleNotFoundError for unknown schedule", async () => {
1029
- await createAgentConfig("schedule-not-found", {
1030
- name: "schedule-not-found",
1031
- schedules: {
1032
- exists: { type: "interval", interval: "1h" },
1033
- },
1034
- });
1035
-
1036
- const configPath = await createConfig({
1037
- version: 1,
1038
- agents: [{ path: "./agents/schedule-not-found.yaml" }],
1039
- });
1040
-
1041
- const manager = createTestManager(configPath);
1042
- await manager.initialize();
1043
-
1044
- await expect(
1045
- manager.getSchedule("schedule-not-found", "does-not-exist")
1046
- ).rejects.toThrow(ScheduleNotFoundError);
1047
- });
1048
- });
1049
-
1050
- describe("empty fleet edge cases", () => {
1051
- it("handles fleet with no agents", async () => {
1052
- const configPath = await createConfig({
1053
- version: 1,
1054
- agents: [],
1055
- });
1056
-
1057
- const manager = createTestManager(configPath);
1058
- await manager.initialize();
1059
-
1060
- expect(manager.state.agentCount).toBe(0);
1061
-
1062
- const status = await manager.getFleetStatus();
1063
- expect(status.counts.totalAgents).toBe(0);
1064
- expect(status.counts.totalSchedules).toBe(0);
1065
-
1066
- const agents = await manager.getAgentInfo();
1067
- expect(agents).toEqual([]);
1068
-
1069
- const schedules = await manager.getSchedules();
1070
- expect(schedules).toEqual([]);
1071
-
1072
- await manager.start();
1073
- await new Promise((resolve) => setTimeout(resolve, 50));
1074
- expect(manager.state.status).toBe("running");
1075
-
1076
- await manager.stop();
1077
- expect(manager.state.status).toBe("stopped");
1078
- });
1079
-
1080
- it("handles agent with no schedules", async () => {
1081
- await createAgentConfig("no-schedules", {
1082
- name: "no-schedules",
1083
- description: "Agent without schedules",
1084
- });
1085
-
1086
- const configPath = await createConfig({
1087
- version: 1,
1088
- agents: [{ path: "./agents/no-schedules.yaml" }],
1089
- });
1090
-
1091
- const manager = createTestManager(configPath);
1092
- await manager.initialize();
1093
-
1094
- const agentInfo = await manager.getAgentInfoByName("no-schedules");
1095
- expect(agentInfo.scheduleCount).toBe(0);
1096
- expect(agentInfo.schedules).toEqual([]);
1097
-
1098
- // Trigger should still work
1099
- const result = await manager.trigger("no-schedules");
1100
- expect(result.agentName).toBe("no-schedules");
1101
- expect(result.scheduleName).toBeNull();
1102
- });
1103
- });
1104
-
1105
- describe("concurrency edge cases", () => {
1106
- it("single start/stop cycle completes correctly", async () => {
1107
- await createAgentConfig("rapid-cycle", {
1108
- name: "rapid-cycle",
1109
- });
1110
-
1111
- const configPath = await createConfig({
1112
- version: 1,
1113
- agents: [{ path: "./agents/rapid-cycle.yaml" }],
1114
- });
1115
-
1116
- const manager = createTestManager(configPath);
1117
- await manager.initialize();
1118
-
1119
- // Single start/stop cycle
1120
- await manager.start();
1121
- await new Promise((resolve) => setTimeout(resolve, 20));
1122
- await manager.stop();
1123
-
1124
- expect(manager.state.status).toBe("stopped");
1125
- });
1126
-
1127
- it("concurrent triggers to same agent", async () => {
1128
- await createAgentConfig("concurrent-agent", {
1129
- name: "concurrent-agent",
1130
- instances: { max_concurrent: 5 },
1131
- });
1132
-
1133
- const configPath = await createConfig({
1134
- version: 1,
1135
- agents: [{ path: "./agents/concurrent-agent.yaml" }],
1136
- });
1137
-
1138
- const manager = createTestManager(configPath);
1139
- await manager.initialize();
1140
-
1141
- // Trigger multiple jobs concurrently
1142
- const triggers = await Promise.all([
1143
- manager.trigger("concurrent-agent", undefined, { prompt: "Job 1" }),
1144
- manager.trigger("concurrent-agent", undefined, { prompt: "Job 2" }),
1145
- manager.trigger("concurrent-agent", undefined, { prompt: "Job 3" }),
1146
- ]);
1147
-
1148
- expect(triggers).toHaveLength(3);
1149
- const jobIds = triggers.map((t) => t.jobId);
1150
- expect(new Set(jobIds).size).toBe(3); // All unique IDs
1151
- });
1152
- });
1153
-
1154
- describe("stop with cancelOnTimeout", () => {
1155
- it("stops with cancelOnTimeout option", async () => {
1156
- await createAgentConfig("cancel-agent", {
1157
- name: "cancel-agent",
1158
- });
1159
-
1160
- const configPath = await createConfig({
1161
- version: 1,
1162
- agents: [{ path: "./agents/cancel-agent.yaml" }],
1163
- });
1164
-
1165
- const manager = createTestManager(configPath);
1166
- await manager.initialize();
1167
- await manager.start();
1168
- await new Promise((resolve) => setTimeout(resolve, 50));
1169
-
1170
- // Stop with cancelOnTimeout enabled (doesn't matter here since no running jobs)
1171
- await manager.stop({
1172
- timeout: 100,
1173
- cancelOnTimeout: true,
1174
- cancelTimeout: 50,
1175
- });
1176
-
1177
- expect(manager.state.status).toBe("stopped");
1178
- });
1179
- });
1180
-
1181
- describe("error state handling", () => {
1182
- it("tracks errors during initialization", async () => {
1183
- // Create invalid config with missing agent file
1184
- const configPath = await createConfig({
1185
- version: 1,
1186
- agents: [{ path: "./agents/nonexistent.yaml" }],
1187
- });
1188
-
1189
- const logger = createSilentLogger();
1190
- const manager = new FleetManager({
1191
- configPath,
1192
- stateDir,
1193
- logger,
1194
- });
1195
-
1196
- // Initialize should fail
1197
- try {
1198
- await manager.initialize();
1199
- } catch {
1200
- // Expected
1201
- }
1202
-
1203
- // Status should be error
1204
- expect(manager.state.status).toBe("error");
1205
- expect(manager.state.lastError).toBeDefined();
1206
- });
1207
- });
1208
- });
1209
- });