@herdctl/core 0.0.1

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 (520) hide show
  1. package/.turbo/turbo-build.log +4 -0
  2. package/.turbo/turbo-test.log +219 -0
  3. package/.turbo/turbo-typecheck.log +4 -0
  4. package/coverage/base.css +224 -0
  5. package/coverage/block-navigation.js +87 -0
  6. package/coverage/coverage-final.json +51 -0
  7. package/coverage/favicon.png +0 -0
  8. package/coverage/index.html +251 -0
  9. package/coverage/prettify.css +1 -0
  10. package/coverage/prettify.js +2 -0
  11. package/coverage/sort-arrow-sprite.png +0 -0
  12. package/coverage/sorter.js +210 -0
  13. package/coverage/src/config/index.html +191 -0
  14. package/coverage/src/config/index.ts.html +442 -0
  15. package/coverage/src/config/interpolate.ts.html +652 -0
  16. package/coverage/src/config/loader.ts.html +1501 -0
  17. package/coverage/src/config/merge.ts.html +823 -0
  18. package/coverage/src/config/parser.ts.html +1213 -0
  19. package/coverage/src/config/schema.ts.html +1123 -0
  20. package/coverage/src/fleet-manager/errors.ts.html +2326 -0
  21. package/coverage/src/fleet-manager/event-types.ts.html +1219 -0
  22. package/coverage/src/fleet-manager/fleet-manager.ts.html +7030 -0
  23. package/coverage/src/fleet-manager/index.html +206 -0
  24. package/coverage/src/fleet-manager/index.ts.html +469 -0
  25. package/coverage/src/fleet-manager/job-manager.ts.html +2074 -0
  26. package/coverage/src/fleet-manager/job-queue.ts.html +2479 -0
  27. package/coverage/src/fleet-manager/types.ts.html +2602 -0
  28. package/coverage/src/index.html +116 -0
  29. package/coverage/src/index.ts.html +181 -0
  30. package/coverage/src/runner/errors.ts.html +1006 -0
  31. package/coverage/src/runner/index.html +191 -0
  32. package/coverage/src/runner/index.ts.html +256 -0
  33. package/coverage/src/runner/job-executor.ts.html +1429 -0
  34. package/coverage/src/runner/message-processor.ts.html +1150 -0
  35. package/coverage/src/runner/sdk-adapter.ts.html +658 -0
  36. package/coverage/src/runner/types.ts.html +559 -0
  37. package/coverage/src/scheduler/errors.ts.html +388 -0
  38. package/coverage/src/scheduler/index.html +206 -0
  39. package/coverage/src/scheduler/index.ts.html +244 -0
  40. package/coverage/src/scheduler/interval.ts.html +652 -0
  41. package/coverage/src/scheduler/schedule-runner.ts.html +1411 -0
  42. package/coverage/src/scheduler/schedule-state.ts.html +718 -0
  43. package/coverage/src/scheduler/scheduler.ts.html +1795 -0
  44. package/coverage/src/scheduler/types.ts.html +733 -0
  45. package/coverage/src/state/directory.ts.html +736 -0
  46. package/coverage/src/state/errors.ts.html +376 -0
  47. package/coverage/src/state/fleet-state.ts.html +937 -0
  48. package/coverage/src/state/index.html +221 -0
  49. package/coverage/src/state/index.ts.html +322 -0
  50. package/coverage/src/state/job-metadata.ts.html +1420 -0
  51. package/coverage/src/state/job-output.ts.html +1033 -0
  52. package/coverage/src/state/schemas/fleet-state.ts.html +445 -0
  53. package/coverage/src/state/schemas/index.html +176 -0
  54. package/coverage/src/state/schemas/index.ts.html +286 -0
  55. package/coverage/src/state/schemas/job-metadata.ts.html +628 -0
  56. package/coverage/src/state/schemas/job-output.ts.html +616 -0
  57. package/coverage/src/state/schemas/session-info.ts.html +361 -0
  58. package/coverage/src/state/session.ts.html +844 -0
  59. package/coverage/src/state/types.ts.html +262 -0
  60. package/coverage/src/state/utils/atomic.ts.html +748 -0
  61. package/coverage/src/state/utils/index.html +146 -0
  62. package/coverage/src/state/utils/index.ts.html +103 -0
  63. package/coverage/src/state/utils/reads.ts.html +1621 -0
  64. package/coverage/src/work-sources/adapters/github.ts.html +3583 -0
  65. package/coverage/src/work-sources/adapters/index.html +131 -0
  66. package/coverage/src/work-sources/adapters/index.ts.html +277 -0
  67. package/coverage/src/work-sources/errors.ts.html +298 -0
  68. package/coverage/src/work-sources/index.html +176 -0
  69. package/coverage/src/work-sources/index.ts.html +529 -0
  70. package/coverage/src/work-sources/manager.ts.html +1324 -0
  71. package/coverage/src/work-sources/registry.ts.html +619 -0
  72. package/coverage/src/work-sources/types.ts.html +568 -0
  73. package/dist/config/__tests__/agent.test.d.ts +2 -0
  74. package/dist/config/__tests__/agent.test.d.ts.map +1 -0
  75. package/dist/config/__tests__/agent.test.js +752 -0
  76. package/dist/config/__tests__/agent.test.js.map +1 -0
  77. package/dist/config/__tests__/interpolate.test.d.ts +2 -0
  78. package/dist/config/__tests__/interpolate.test.d.ts.map +1 -0
  79. package/dist/config/__tests__/interpolate.test.js +509 -0
  80. package/dist/config/__tests__/interpolate.test.js.map +1 -0
  81. package/dist/config/__tests__/loader.test.d.ts +2 -0
  82. package/dist/config/__tests__/loader.test.d.ts.map +1 -0
  83. package/dist/config/__tests__/loader.test.js +631 -0
  84. package/dist/config/__tests__/loader.test.js.map +1 -0
  85. package/dist/config/__tests__/merge.test.d.ts +2 -0
  86. package/dist/config/__tests__/merge.test.d.ts.map +1 -0
  87. package/dist/config/__tests__/merge.test.js +672 -0
  88. package/dist/config/__tests__/merge.test.js.map +1 -0
  89. package/dist/config/__tests__/parser.test.d.ts +2 -0
  90. package/dist/config/__tests__/parser.test.d.ts.map +1 -0
  91. package/dist/config/__tests__/parser.test.js +476 -0
  92. package/dist/config/__tests__/parser.test.js.map +1 -0
  93. package/dist/config/__tests__/schema.test.d.ts +2 -0
  94. package/dist/config/__tests__/schema.test.d.ts.map +1 -0
  95. package/dist/config/__tests__/schema.test.js +776 -0
  96. package/dist/config/__tests__/schema.test.js.map +1 -0
  97. package/dist/config/index.d.ts +11 -0
  98. package/dist/config/index.d.ts.map +1 -0
  99. package/dist/config/index.js +26 -0
  100. package/dist/config/index.js.map +1 -0
  101. package/dist/config/interpolate.d.ts +76 -0
  102. package/dist/config/interpolate.d.ts.map +1 -0
  103. package/dist/config/interpolate.js +143 -0
  104. package/dist/config/interpolate.js.map +1 -0
  105. package/dist/config/loader.d.ts +147 -0
  106. package/dist/config/loader.d.ts.map +1 -0
  107. package/dist/config/loader.js +336 -0
  108. package/dist/config/loader.js.map +1 -0
  109. package/dist/config/merge.d.ts +84 -0
  110. package/dist/config/merge.d.ts.map +1 -0
  111. package/dist/config/merge.js +138 -0
  112. package/dist/config/merge.js.map +1 -0
  113. package/dist/config/parser.d.ts +143 -0
  114. package/dist/config/parser.d.ts.map +1 -0
  115. package/dist/config/parser.js +316 -0
  116. package/dist/config/parser.js.map +1 -0
  117. package/dist/config/schema.d.ts +1906 -0
  118. package/dist/config/schema.d.ts.map +1 -0
  119. package/dist/config/schema.js +268 -0
  120. package/dist/config/schema.js.map +1 -0
  121. package/dist/fleet-manager/__tests__/coverage.test.d.ts +13 -0
  122. package/dist/fleet-manager/__tests__/coverage.test.d.ts.map +1 -0
  123. package/dist/fleet-manager/__tests__/coverage.test.js +2282 -0
  124. package/dist/fleet-manager/__tests__/coverage.test.js.map +1 -0
  125. package/dist/fleet-manager/__tests__/errors.test.d.ts +7 -0
  126. package/dist/fleet-manager/__tests__/errors.test.d.ts.map +1 -0
  127. package/dist/fleet-manager/__tests__/errors.test.js +557 -0
  128. package/dist/fleet-manager/__tests__/errors.test.js.map +1 -0
  129. package/dist/fleet-manager/__tests__/event-helpers.test.d.ts +7 -0
  130. package/dist/fleet-manager/__tests__/event-helpers.test.d.ts.map +1 -0
  131. package/dist/fleet-manager/__tests__/event-helpers.test.js +368 -0
  132. package/dist/fleet-manager/__tests__/event-helpers.test.js.map +1 -0
  133. package/dist/fleet-manager/__tests__/integration.test.d.ts +11 -0
  134. package/dist/fleet-manager/__tests__/integration.test.d.ts.map +1 -0
  135. package/dist/fleet-manager/__tests__/integration.test.js +949 -0
  136. package/dist/fleet-manager/__tests__/integration.test.js.map +1 -0
  137. package/dist/fleet-manager/__tests__/job-control.test.d.ts +7 -0
  138. package/dist/fleet-manager/__tests__/job-control.test.d.ts.map +1 -0
  139. package/dist/fleet-manager/__tests__/job-control.test.js +215 -0
  140. package/dist/fleet-manager/__tests__/job-control.test.js.map +1 -0
  141. package/dist/fleet-manager/__tests__/job-manager.test.d.ts +7 -0
  142. package/dist/fleet-manager/__tests__/job-manager.test.d.ts.map +1 -0
  143. package/dist/fleet-manager/__tests__/job-manager.test.js +659 -0
  144. package/dist/fleet-manager/__tests__/job-manager.test.js.map +1 -0
  145. package/dist/fleet-manager/__tests__/job-queue.test.d.ts +5 -0
  146. package/dist/fleet-manager/__tests__/job-queue.test.d.ts.map +1 -0
  147. package/dist/fleet-manager/__tests__/job-queue.test.js +315 -0
  148. package/dist/fleet-manager/__tests__/job-queue.test.js.map +1 -0
  149. package/dist/fleet-manager/__tests__/reload.test.d.ts +7 -0
  150. package/dist/fleet-manager/__tests__/reload.test.d.ts.map +1 -0
  151. package/dist/fleet-manager/__tests__/reload.test.js +609 -0
  152. package/dist/fleet-manager/__tests__/reload.test.js.map +1 -0
  153. package/dist/fleet-manager/__tests__/status-queries.test.d.ts +7 -0
  154. package/dist/fleet-manager/__tests__/status-queries.test.d.ts.map +1 -0
  155. package/dist/fleet-manager/__tests__/status-queries.test.js +488 -0
  156. package/dist/fleet-manager/__tests__/status-queries.test.js.map +1 -0
  157. package/dist/fleet-manager/__tests__/trigger.test.d.ts +7 -0
  158. package/dist/fleet-manager/__tests__/trigger.test.d.ts.map +1 -0
  159. package/dist/fleet-manager/__tests__/trigger.test.js +471 -0
  160. package/dist/fleet-manager/__tests__/trigger.test.js.map +1 -0
  161. package/dist/fleet-manager/errors.d.ts +407 -0
  162. package/dist/fleet-manager/errors.d.ts.map +1 -0
  163. package/dist/fleet-manager/errors.js +569 -0
  164. package/dist/fleet-manager/errors.js.map +1 -0
  165. package/dist/fleet-manager/event-types.d.ts +302 -0
  166. package/dist/fleet-manager/event-types.d.ts.map +1 -0
  167. package/dist/fleet-manager/event-types.js +9 -0
  168. package/dist/fleet-manager/event-types.js.map +1 -0
  169. package/dist/fleet-manager/fleet-manager.d.ts +699 -0
  170. package/dist/fleet-manager/fleet-manager.d.ts.map +1 -0
  171. package/dist/fleet-manager/fleet-manager.js +1906 -0
  172. package/dist/fleet-manager/fleet-manager.js.map +1 -0
  173. package/dist/fleet-manager/index.d.ts +17 -0
  174. package/dist/fleet-manager/index.d.ts.map +1 -0
  175. package/dist/fleet-manager/index.js +29 -0
  176. package/dist/fleet-manager/index.js.map +1 -0
  177. package/dist/fleet-manager/job-manager.d.ts +271 -0
  178. package/dist/fleet-manager/job-manager.d.ts.map +1 -0
  179. package/dist/fleet-manager/job-manager.js +443 -0
  180. package/dist/fleet-manager/job-manager.js.map +1 -0
  181. package/dist/fleet-manager/job-queue.d.ts +422 -0
  182. package/dist/fleet-manager/job-queue.d.ts.map +1 -0
  183. package/dist/fleet-manager/job-queue.js +448 -0
  184. package/dist/fleet-manager/job-queue.js.map +1 -0
  185. package/dist/fleet-manager/types.d.ts +680 -0
  186. package/dist/fleet-manager/types.d.ts.map +1 -0
  187. package/dist/fleet-manager/types.js +8 -0
  188. package/dist/fleet-manager/types.js.map +1 -0
  189. package/dist/index.d.ts +20 -0
  190. package/dist/index.d.ts.map +1 -0
  191. package/dist/index.js +26 -0
  192. package/dist/index.js.map +1 -0
  193. package/dist/runner/__tests__/errors.test.d.ts +2 -0
  194. package/dist/runner/__tests__/errors.test.d.ts.map +1 -0
  195. package/dist/runner/__tests__/errors.test.js +264 -0
  196. package/dist/runner/__tests__/errors.test.js.map +1 -0
  197. package/dist/runner/__tests__/job-executor.test.d.ts +2 -0
  198. package/dist/runner/__tests__/job-executor.test.d.ts.map +1 -0
  199. package/dist/runner/__tests__/job-executor.test.js +1345 -0
  200. package/dist/runner/__tests__/job-executor.test.js.map +1 -0
  201. package/dist/runner/__tests__/message-processor.test.d.ts +2 -0
  202. package/dist/runner/__tests__/message-processor.test.d.ts.map +1 -0
  203. package/dist/runner/__tests__/message-processor.test.js +768 -0
  204. package/dist/runner/__tests__/message-processor.test.js.map +1 -0
  205. package/dist/runner/__tests__/sdk-adapter.test.d.ts +2 -0
  206. package/dist/runner/__tests__/sdk-adapter.test.d.ts.map +1 -0
  207. package/dist/runner/__tests__/sdk-adapter.test.js +554 -0
  208. package/dist/runner/__tests__/sdk-adapter.test.js.map +1 -0
  209. package/dist/runner/errors.d.ts +121 -0
  210. package/dist/runner/errors.d.ts.map +1 -0
  211. package/dist/runner/errors.js +212 -0
  212. package/dist/runner/errors.js.map +1 -0
  213. package/dist/runner/index.d.ts +12 -0
  214. package/dist/runner/index.d.ts.map +1 -0
  215. package/dist/runner/index.js +15 -0
  216. package/dist/runner/index.js.map +1 -0
  217. package/dist/runner/job-executor.d.ts +98 -0
  218. package/dist/runner/job-executor.d.ts.map +1 -0
  219. package/dist/runner/job-executor.js +333 -0
  220. package/dist/runner/job-executor.js.map +1 -0
  221. package/dist/runner/message-processor.d.ts +45 -0
  222. package/dist/runner/message-processor.d.ts.map +1 -0
  223. package/dist/runner/message-processor.js +294 -0
  224. package/dist/runner/message-processor.js.map +1 -0
  225. package/dist/runner/sdk-adapter.d.ts +60 -0
  226. package/dist/runner/sdk-adapter.d.ts.map +1 -0
  227. package/dist/runner/sdk-adapter.js +138 -0
  228. package/dist/runner/sdk-adapter.js.map +1 -0
  229. package/dist/runner/types.d.ts +135 -0
  230. package/dist/runner/types.d.ts.map +1 -0
  231. package/dist/runner/types.js +7 -0
  232. package/dist/runner/types.js.map +1 -0
  233. package/dist/scheduler/__tests__/errors.test.d.ts +2 -0
  234. package/dist/scheduler/__tests__/errors.test.d.ts.map +1 -0
  235. package/dist/scheduler/__tests__/errors.test.js +101 -0
  236. package/dist/scheduler/__tests__/errors.test.js.map +1 -0
  237. package/dist/scheduler/__tests__/interval.test.d.ts +2 -0
  238. package/dist/scheduler/__tests__/interval.test.d.ts.map +1 -0
  239. package/dist/scheduler/__tests__/interval.test.js +419 -0
  240. package/dist/scheduler/__tests__/interval.test.js.map +1 -0
  241. package/dist/scheduler/__tests__/schedule-runner.test.d.ts +2 -0
  242. package/dist/scheduler/__tests__/schedule-runner.test.d.ts.map +1 -0
  243. package/dist/scheduler/__tests__/schedule-runner.test.js +634 -0
  244. package/dist/scheduler/__tests__/schedule-runner.test.js.map +1 -0
  245. package/dist/scheduler/__tests__/schedule-state.test.d.ts +2 -0
  246. package/dist/scheduler/__tests__/schedule-state.test.d.ts.map +1 -0
  247. package/dist/scheduler/__tests__/schedule-state.test.js +572 -0
  248. package/dist/scheduler/__tests__/schedule-state.test.js.map +1 -0
  249. package/dist/scheduler/__tests__/scheduler.test.d.ts +2 -0
  250. package/dist/scheduler/__tests__/scheduler.test.d.ts.map +1 -0
  251. package/dist/scheduler/__tests__/scheduler.test.js +987 -0
  252. package/dist/scheduler/__tests__/scheduler.test.js.map +1 -0
  253. package/dist/scheduler/errors.d.ts +61 -0
  254. package/dist/scheduler/errors.d.ts.map +1 -0
  255. package/dist/scheduler/errors.js +81 -0
  256. package/dist/scheduler/errors.js.map +1 -0
  257. package/dist/scheduler/index.d.ts +13 -0
  258. package/dist/scheduler/index.d.ts.map +1 -0
  259. package/dist/scheduler/index.js +17 -0
  260. package/dist/scheduler/index.js.map +1 -0
  261. package/dist/scheduler/interval.d.ts +64 -0
  262. package/dist/scheduler/interval.d.ts.map +1 -0
  263. package/dist/scheduler/interval.js +139 -0
  264. package/dist/scheduler/interval.js.map +1 -0
  265. package/dist/scheduler/schedule-runner.d.ts +149 -0
  266. package/dist/scheduler/schedule-runner.d.ts.map +1 -0
  267. package/dist/scheduler/schedule-runner.js +277 -0
  268. package/dist/scheduler/schedule-runner.js.map +1 -0
  269. package/dist/scheduler/schedule-state.d.ts +105 -0
  270. package/dist/scheduler/schedule-state.d.ts.map +1 -0
  271. package/dist/scheduler/schedule-state.js +151 -0
  272. package/dist/scheduler/schedule-state.js.map +1 -0
  273. package/dist/scheduler/scheduler.d.ts +138 -0
  274. package/dist/scheduler/scheduler.d.ts.map +1 -0
  275. package/dist/scheduler/scheduler.js +423 -0
  276. package/dist/scheduler/scheduler.js.map +1 -0
  277. package/dist/scheduler/types.d.ts +160 -0
  278. package/dist/scheduler/types.d.ts.map +1 -0
  279. package/dist/scheduler/types.js +8 -0
  280. package/dist/scheduler/types.js.map +1 -0
  281. package/dist/state/__tests__/directory.test.d.ts +2 -0
  282. package/dist/state/__tests__/directory.test.d.ts.map +1 -0
  283. package/dist/state/__tests__/directory.test.js +414 -0
  284. package/dist/state/__tests__/directory.test.js.map +1 -0
  285. package/dist/state/__tests__/fleet-state.test.d.ts +2 -0
  286. package/dist/state/__tests__/fleet-state.test.d.ts.map +1 -0
  287. package/dist/state/__tests__/fleet-state.test.js +696 -0
  288. package/dist/state/__tests__/fleet-state.test.js.map +1 -0
  289. package/dist/state/__tests__/job-metadata-schema.test.d.ts +2 -0
  290. package/dist/state/__tests__/job-metadata-schema.test.d.ts.map +1 -0
  291. package/dist/state/__tests__/job-metadata-schema.test.js +329 -0
  292. package/dist/state/__tests__/job-metadata-schema.test.js.map +1 -0
  293. package/dist/state/__tests__/job-metadata.test.d.ts +2 -0
  294. package/dist/state/__tests__/job-metadata.test.d.ts.map +1 -0
  295. package/dist/state/__tests__/job-metadata.test.js +667 -0
  296. package/dist/state/__tests__/job-metadata.test.js.map +1 -0
  297. package/dist/state/__tests__/job-output.test.d.ts +2 -0
  298. package/dist/state/__tests__/job-output.test.d.ts.map +1 -0
  299. package/dist/state/__tests__/job-output.test.js +672 -0
  300. package/dist/state/__tests__/job-output.test.js.map +1 -0
  301. package/dist/state/__tests__/session-schema.test.d.ts +2 -0
  302. package/dist/state/__tests__/session-schema.test.d.ts.map +1 -0
  303. package/dist/state/__tests__/session-schema.test.js +323 -0
  304. package/dist/state/__tests__/session-schema.test.js.map +1 -0
  305. package/dist/state/__tests__/session.test.d.ts +2 -0
  306. package/dist/state/__tests__/session.test.d.ts.map +1 -0
  307. package/dist/state/__tests__/session.test.js +468 -0
  308. package/dist/state/__tests__/session.test.js.map +1 -0
  309. package/dist/state/directory.d.ts +42 -0
  310. package/dist/state/directory.d.ts.map +1 -0
  311. package/dist/state/directory.js +170 -0
  312. package/dist/state/directory.js.map +1 -0
  313. package/dist/state/errors.d.ts +44 -0
  314. package/dist/state/errors.d.ts.map +1 -0
  315. package/dist/state/errors.js +82 -0
  316. package/dist/state/errors.js.map +1 -0
  317. package/dist/state/fleet-state.d.ts +126 -0
  318. package/dist/state/fleet-state.d.ts.map +1 -0
  319. package/dist/state/fleet-state.js +196 -0
  320. package/dist/state/fleet-state.js.map +1 -0
  321. package/dist/state/index.d.ts +21 -0
  322. package/dist/state/index.d.ts.map +1 -0
  323. package/dist/state/index.js +30 -0
  324. package/dist/state/index.js.map +1 -0
  325. package/dist/state/job-metadata.d.ts +151 -0
  326. package/dist/state/job-metadata.d.ts.map +1 -0
  327. package/dist/state/job-metadata.js +287 -0
  328. package/dist/state/job-metadata.js.map +1 -0
  329. package/dist/state/job-output.d.ts +116 -0
  330. package/dist/state/job-output.d.ts.map +1 -0
  331. package/dist/state/job-output.js +218 -0
  332. package/dist/state/job-output.js.map +1 -0
  333. package/dist/state/schemas/__tests__/job-output.test.d.ts +2 -0
  334. package/dist/state/schemas/__tests__/job-output.test.d.ts.map +1 -0
  335. package/dist/state/schemas/__tests__/job-output.test.js +279 -0
  336. package/dist/state/schemas/__tests__/job-output.test.js.map +1 -0
  337. package/dist/state/schemas/fleet-state.d.ts +249 -0
  338. package/dist/state/schemas/fleet-state.d.ts.map +1 -0
  339. package/dist/state/schemas/fleet-state.js +97 -0
  340. package/dist/state/schemas/fleet-state.js.map +1 -0
  341. package/dist/state/schemas/index.d.ts +10 -0
  342. package/dist/state/schemas/index.d.ts.map +1 -0
  343. package/dist/state/schemas/index.js +10 -0
  344. package/dist/state/schemas/index.js.map +1 -0
  345. package/dist/state/schemas/job-metadata.d.ts +118 -0
  346. package/dist/state/schemas/job-metadata.d.ts.map +1 -0
  347. package/dist/state/schemas/job-metadata.js +123 -0
  348. package/dist/state/schemas/job-metadata.js.map +1 -0
  349. package/dist/state/schemas/job-output.d.ts +291 -0
  350. package/dist/state/schemas/job-output.d.ts.map +1 -0
  351. package/dist/state/schemas/job-output.js +132 -0
  352. package/dist/state/schemas/job-output.js.map +1 -0
  353. package/dist/state/schemas/session-info.d.ts +65 -0
  354. package/dist/state/schemas/session-info.d.ts.map +1 -0
  355. package/dist/state/schemas/session-info.js +58 -0
  356. package/dist/state/schemas/session-info.js.map +1 -0
  357. package/dist/state/session.d.ts +92 -0
  358. package/dist/state/session.d.ts.map +1 -0
  359. package/dist/state/session.js +173 -0
  360. package/dist/state/session.js.map +1 -0
  361. package/dist/state/types.d.ts +54 -0
  362. package/dist/state/types.d.ts.map +1 -0
  363. package/dist/state/types.js +18 -0
  364. package/dist/state/types.js.map +1 -0
  365. package/dist/state/utils/__tests__/atomic.test.d.ts +2 -0
  366. package/dist/state/utils/__tests__/atomic.test.d.ts.map +1 -0
  367. package/dist/state/utils/__tests__/atomic.test.js +537 -0
  368. package/dist/state/utils/__tests__/atomic.test.js.map +1 -0
  369. package/dist/state/utils/__tests__/reads.test.d.ts +2 -0
  370. package/dist/state/utils/__tests__/reads.test.d.ts.map +1 -0
  371. package/dist/state/utils/__tests__/reads.test.js +792 -0
  372. package/dist/state/utils/__tests__/reads.test.js.map +1 -0
  373. package/dist/state/utils/atomic.d.ts +89 -0
  374. package/dist/state/utils/atomic.d.ts.map +1 -0
  375. package/dist/state/utils/atomic.js +157 -0
  376. package/dist/state/utils/atomic.js.map +1 -0
  377. package/dist/state/utils/index.d.ts +6 -0
  378. package/dist/state/utils/index.d.ts.map +1 -0
  379. package/dist/state/utils/index.js +6 -0
  380. package/dist/state/utils/index.js.map +1 -0
  381. package/dist/state/utils/reads.d.ts +196 -0
  382. package/dist/state/utils/reads.d.ts.map +1 -0
  383. package/dist/state/utils/reads.js +346 -0
  384. package/dist/state/utils/reads.js.map +1 -0
  385. package/dist/work-sources/__tests__/github.test.d.ts +2 -0
  386. package/dist/work-sources/__tests__/github.test.d.ts.map +1 -0
  387. package/dist/work-sources/__tests__/github.test.js +1334 -0
  388. package/dist/work-sources/__tests__/github.test.js.map +1 -0
  389. package/dist/work-sources/__tests__/manager.test.d.ts +2 -0
  390. package/dist/work-sources/__tests__/manager.test.d.ts.map +1 -0
  391. package/dist/work-sources/__tests__/manager.test.js +424 -0
  392. package/dist/work-sources/__tests__/manager.test.js.map +1 -0
  393. package/dist/work-sources/__tests__/registry.test.d.ts +2 -0
  394. package/dist/work-sources/__tests__/registry.test.d.ts.map +1 -0
  395. package/dist/work-sources/__tests__/registry.test.js +381 -0
  396. package/dist/work-sources/__tests__/registry.test.js.map +1 -0
  397. package/dist/work-sources/__tests__/types.test.d.ts +2 -0
  398. package/dist/work-sources/__tests__/types.test.d.ts.map +1 -0
  399. package/dist/work-sources/__tests__/types.test.js +406 -0
  400. package/dist/work-sources/__tests__/types.test.js.map +1 -0
  401. package/dist/work-sources/adapters/github.d.ts +290 -0
  402. package/dist/work-sources/adapters/github.d.ts.map +1 -0
  403. package/dist/work-sources/adapters/github.js +803 -0
  404. package/dist/work-sources/adapters/github.js.map +1 -0
  405. package/dist/work-sources/adapters/index.d.ts +10 -0
  406. package/dist/work-sources/adapters/index.d.ts.map +1 -0
  407. package/dist/work-sources/adapters/index.js +31 -0
  408. package/dist/work-sources/adapters/index.js.map +1 -0
  409. package/dist/work-sources/errors.d.ts +40 -0
  410. package/dist/work-sources/errors.d.ts.map +1 -0
  411. package/dist/work-sources/errors.js +54 -0
  412. package/dist/work-sources/errors.js.map +1 -0
  413. package/dist/work-sources/index.d.ts +105 -0
  414. package/dist/work-sources/index.d.ts.map +1 -0
  415. package/dist/work-sources/index.js +24 -0
  416. package/dist/work-sources/index.js.map +1 -0
  417. package/dist/work-sources/manager.d.ts +370 -0
  418. package/dist/work-sources/manager.d.ts.map +1 -0
  419. package/dist/work-sources/manager.js +61 -0
  420. package/dist/work-sources/manager.js.map +1 -0
  421. package/dist/work-sources/registry.d.ts +128 -0
  422. package/dist/work-sources/registry.d.ts.map +1 -0
  423. package/dist/work-sources/registry.js +132 -0
  424. package/dist/work-sources/registry.js.map +1 -0
  425. package/dist/work-sources/types.d.ts +127 -0
  426. package/dist/work-sources/types.d.ts.map +1 -0
  427. package/dist/work-sources/types.js +8 -0
  428. package/dist/work-sources/types.js.map +1 -0
  429. package/package.json +23 -0
  430. package/src/config/__tests__/agent.test.ts +864 -0
  431. package/src/config/__tests__/interpolate.test.ts +644 -0
  432. package/src/config/__tests__/loader.test.ts +784 -0
  433. package/src/config/__tests__/merge.test.ts +751 -0
  434. package/src/config/__tests__/parser.test.ts +533 -0
  435. package/src/config/__tests__/schema.test.ts +873 -0
  436. package/src/config/index.ts +119 -0
  437. package/src/config/interpolate.ts +189 -0
  438. package/src/config/loader.ts +472 -0
  439. package/src/config/merge.ts +246 -0
  440. package/src/config/parser.ts +376 -0
  441. package/src/config/schema.ts +346 -0
  442. package/src/fleet-manager/__tests__/coverage.test.ts +2869 -0
  443. package/src/fleet-manager/__tests__/errors.test.ts +660 -0
  444. package/src/fleet-manager/__tests__/event-helpers.test.ts +448 -0
  445. package/src/fleet-manager/__tests__/integration.test.ts +1209 -0
  446. package/src/fleet-manager/__tests__/job-control.test.ts +283 -0
  447. package/src/fleet-manager/__tests__/job-manager.test.ts +869 -0
  448. package/src/fleet-manager/__tests__/job-queue.test.ts +401 -0
  449. package/src/fleet-manager/__tests__/reload.test.ts +751 -0
  450. package/src/fleet-manager/__tests__/status-queries.test.ts +595 -0
  451. package/src/fleet-manager/__tests__/trigger.test.ts +601 -0
  452. package/src/fleet-manager/errors.ts +747 -0
  453. package/src/fleet-manager/event-types.ts +378 -0
  454. package/src/fleet-manager/fleet-manager.ts +2315 -0
  455. package/src/fleet-manager/index.ts +128 -0
  456. package/src/fleet-manager/job-manager.ts +663 -0
  457. package/src/fleet-manager/job-queue.ts +798 -0
  458. package/src/fleet-manager/types.ts +839 -0
  459. package/src/index.ts +32 -0
  460. package/src/runner/__tests__/errors.test.ts +382 -0
  461. package/src/runner/__tests__/job-executor.test.ts +1708 -0
  462. package/src/runner/__tests__/message-processor.test.ts +960 -0
  463. package/src/runner/__tests__/sdk-adapter.test.ts +626 -0
  464. package/src/runner/errors.ts +307 -0
  465. package/src/runner/index.ts +57 -0
  466. package/src/runner/job-executor.ts +448 -0
  467. package/src/runner/message-processor.ts +355 -0
  468. package/src/runner/sdk-adapter.ts +191 -0
  469. package/src/runner/types.ts +158 -0
  470. package/src/scheduler/__tests__/errors.test.ts +159 -0
  471. package/src/scheduler/__tests__/interval.test.ts +515 -0
  472. package/src/scheduler/__tests__/schedule-runner.test.ts +798 -0
  473. package/src/scheduler/__tests__/schedule-state.test.ts +671 -0
  474. package/src/scheduler/__tests__/scheduler.test.ts +1280 -0
  475. package/src/scheduler/errors.ts +101 -0
  476. package/src/scheduler/index.ts +53 -0
  477. package/src/scheduler/interval.ts +189 -0
  478. package/src/scheduler/schedule-runner.ts +442 -0
  479. package/src/scheduler/schedule-state.ts +211 -0
  480. package/src/scheduler/scheduler.ts +570 -0
  481. package/src/scheduler/types.ts +216 -0
  482. package/src/state/__tests__/directory.test.ts +595 -0
  483. package/src/state/__tests__/fleet-state.test.ts +868 -0
  484. package/src/state/__tests__/job-metadata-schema.test.ts +414 -0
  485. package/src/state/__tests__/job-metadata.test.ts +831 -0
  486. package/src/state/__tests__/job-output.test.ts +856 -0
  487. package/src/state/__tests__/session-schema.test.ts +378 -0
  488. package/src/state/__tests__/session.test.ts +604 -0
  489. package/src/state/directory.ts +217 -0
  490. package/src/state/errors.ts +97 -0
  491. package/src/state/fleet-state.ts +284 -0
  492. package/src/state/index.ts +79 -0
  493. package/src/state/job-metadata.ts +445 -0
  494. package/src/state/job-output.ts +316 -0
  495. package/src/state/schemas/__tests__/job-output.test.ts +338 -0
  496. package/src/state/schemas/fleet-state.ts +120 -0
  497. package/src/state/schemas/index.ts +67 -0
  498. package/src/state/schemas/job-metadata.ts +181 -0
  499. package/src/state/schemas/job-output.ts +177 -0
  500. package/src/state/schemas/session-info.ts +92 -0
  501. package/src/state/session.ts +253 -0
  502. package/src/state/types.ts +59 -0
  503. package/src/state/utils/__tests__/atomic.test.ts +723 -0
  504. package/src/state/utils/__tests__/reads.test.ts +1071 -0
  505. package/src/state/utils/atomic.ts +221 -0
  506. package/src/state/utils/index.ts +6 -0
  507. package/src/state/utils/reads.ts +512 -0
  508. package/src/work-sources/__tests__/github.test.ts +1800 -0
  509. package/src/work-sources/__tests__/manager.test.ts +529 -0
  510. package/src/work-sources/__tests__/registry.test.ts +477 -0
  511. package/src/work-sources/__tests__/types.test.ts +479 -0
  512. package/src/work-sources/adapters/github.ts +1166 -0
  513. package/src/work-sources/adapters/index.ts +64 -0
  514. package/src/work-sources/errors.ts +71 -0
  515. package/src/work-sources/index.ts +148 -0
  516. package/src/work-sources/manager.ts +413 -0
  517. package/src/work-sources/registry.ts +178 -0
  518. package/src/work-sources/types.ts +161 -0
  519. package/tsconfig.json +9 -0
  520. package/vitest.config.ts +19 -0
@@ -0,0 +1,1209 @@
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
+ });