@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,949 @@
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
+ import { describe, it, expect, beforeEach, afterEach, vi } from "vitest";
11
+ import { mkdtemp, rm, mkdir, writeFile, readFile } from "fs/promises";
12
+ import { tmpdir } from "os";
13
+ import { join } from "path";
14
+ import { FleetManager } from "../fleet-manager.js";
15
+ import { InvalidStateError, AgentNotFoundError, ScheduleNotFoundError, } from "../errors.js";
16
+ describe("FleetManager Integration Tests (US-13)", () => {
17
+ let tempDir;
18
+ let configDir;
19
+ let stateDir;
20
+ beforeEach(async () => {
21
+ tempDir = await mkdtemp(join(tmpdir(), "fleet-integration-test-"));
22
+ configDir = join(tempDir, "config");
23
+ stateDir = join(tempDir, ".herdctl");
24
+ await mkdir(configDir, { recursive: true });
25
+ });
26
+ afterEach(async () => {
27
+ await rm(tempDir, { recursive: true, force: true });
28
+ });
29
+ // Helper to create a test config file
30
+ async function createConfig(config) {
31
+ const configPath = join(configDir, "herdctl.yaml");
32
+ const yaml = await import("yaml");
33
+ await writeFile(configPath, yaml.stringify(config));
34
+ return configPath;
35
+ }
36
+ // Helper to create an agent config file
37
+ async function createAgentConfig(name, config) {
38
+ const agentDir = join(configDir, "agents");
39
+ await mkdir(agentDir, { recursive: true });
40
+ const agentPath = join(agentDir, `${name}.yaml`);
41
+ const yaml = await import("yaml");
42
+ await writeFile(agentPath, yaml.stringify(config));
43
+ return agentPath;
44
+ }
45
+ // Create a silent logger for tests
46
+ function createSilentLogger() {
47
+ return {
48
+ debug: vi.fn(),
49
+ info: vi.fn(),
50
+ warn: vi.fn(),
51
+ error: vi.fn(),
52
+ };
53
+ }
54
+ // Create a test manager with common options
55
+ function createTestManager(configPath, options = {}) {
56
+ return new FleetManager({
57
+ configPath,
58
+ stateDir,
59
+ checkInterval: options.checkInterval ?? 10000, // Long interval by default to avoid unexpected triggers
60
+ logger: createSilentLogger(),
61
+ });
62
+ }
63
+ // ==========================================================================
64
+ // Full Flow Integration Tests
65
+ // ==========================================================================
66
+ describe("Full Flow: initialize → start → trigger → complete → stop", () => {
67
+ it("completes a full lifecycle with manual trigger", async () => {
68
+ // Setup: Create agent config
69
+ await createAgentConfig("workflow-agent", {
70
+ name: "workflow-agent",
71
+ description: "Agent for testing full workflow",
72
+ schedules: {
73
+ hourly: {
74
+ type: "interval",
75
+ interval: "1h",
76
+ prompt: "Check hourly tasks",
77
+ },
78
+ },
79
+ });
80
+ const configPath = await createConfig({
81
+ version: 1,
82
+ agents: [{ path: "./agents/workflow-agent.yaml" }],
83
+ });
84
+ // Create manager
85
+ const manager = createTestManager(configPath);
86
+ const events = [];
87
+ // Track all events
88
+ manager.on("initialized", () => events.push("initialized"));
89
+ manager.on("started", () => events.push("started"));
90
+ manager.on("stopped", () => events.push("stopped"));
91
+ manager.on("job:created", () => events.push("job:created"));
92
+ // 1. Verify initial state
93
+ expect(manager.state.status).toBe("uninitialized");
94
+ // 2. Initialize
95
+ await manager.initialize();
96
+ expect(manager.state.status).toBe("initialized");
97
+ expect(manager.state.agentCount).toBe(1);
98
+ expect(events).toContain("initialized");
99
+ // 3. Start
100
+ await manager.start();
101
+ await new Promise((resolve) => setTimeout(resolve, 50)); // Wait for async start
102
+ expect(manager.state.status).toBe("running");
103
+ expect(events).toContain("started");
104
+ // 4. Trigger agent
105
+ const result = await manager.trigger("workflow-agent", "hourly");
106
+ expect(result.agentName).toBe("workflow-agent");
107
+ expect(result.scheduleName).toBe("hourly");
108
+ expect(result.prompt).toBe("Check hourly tasks");
109
+ expect(result.jobId).toMatch(/^job-\d{4}-\d{2}-\d{2}-[a-z0-9]{6}$/);
110
+ expect(events).toContain("job:created");
111
+ // 5. Verify fleet status while running
112
+ const fleetStatus = await manager.getFleetStatus();
113
+ expect(fleetStatus.state).toBe("running");
114
+ expect(fleetStatus.counts.totalAgents).toBe(1);
115
+ expect(fleetStatus.scheduler.status).toBe("running");
116
+ // 6. Stop
117
+ await manager.stop();
118
+ expect(manager.state.status).toBe("stopped");
119
+ expect(events).toContain("stopped");
120
+ // 7. Verify final state
121
+ const finalStatus = await manager.getFleetStatus();
122
+ expect(finalStatus.state).toBe("stopped");
123
+ expect(finalStatus.stoppedAt).not.toBeNull();
124
+ });
125
+ it("emits events in correct order during lifecycle", async () => {
126
+ await createAgentConfig("event-order-agent", {
127
+ name: "event-order-agent",
128
+ });
129
+ const configPath = await createConfig({
130
+ version: 1,
131
+ agents: [{ path: "./agents/event-order-agent.yaml" }],
132
+ });
133
+ const manager = createTestManager(configPath);
134
+ const eventOrder = [];
135
+ manager.on("initialized", () => eventOrder.push("initialized"));
136
+ manager.on("started", () => eventOrder.push("started"));
137
+ manager.on("job:created", () => eventOrder.push("job:created"));
138
+ manager.on("stopped", () => eventOrder.push("stopped"));
139
+ await manager.initialize();
140
+ await manager.start();
141
+ await new Promise((resolve) => setTimeout(resolve, 50));
142
+ await manager.trigger("event-order-agent");
143
+ await manager.stop();
144
+ expect(eventOrder).toEqual([
145
+ "initialized",
146
+ "started",
147
+ "job:created",
148
+ "stopped",
149
+ ]);
150
+ });
151
+ it("handles multiple agents in full workflow", async () => {
152
+ await createAgentConfig("agent-alpha", {
153
+ name: "agent-alpha",
154
+ description: "First agent",
155
+ });
156
+ await createAgentConfig("agent-beta", {
157
+ name: "agent-beta",
158
+ description: "Second agent",
159
+ });
160
+ await createAgentConfig("agent-gamma", {
161
+ name: "agent-gamma",
162
+ description: "Third agent",
163
+ });
164
+ const configPath = await createConfig({
165
+ version: 1,
166
+ agents: [
167
+ { path: "./agents/agent-alpha.yaml" },
168
+ { path: "./agents/agent-beta.yaml" },
169
+ { path: "./agents/agent-gamma.yaml" },
170
+ ],
171
+ });
172
+ const manager = createTestManager(configPath);
173
+ await manager.initialize();
174
+ expect(manager.state.agentCount).toBe(3);
175
+ await manager.start();
176
+ await new Promise((resolve) => setTimeout(resolve, 50));
177
+ // Trigger all agents
178
+ const results = await Promise.all([
179
+ manager.trigger("agent-alpha"),
180
+ manager.trigger("agent-beta"),
181
+ manager.trigger("agent-gamma"),
182
+ ]);
183
+ expect(results).toHaveLength(3);
184
+ expect(results.map((r) => r.agentName).sort()).toEqual([
185
+ "agent-alpha",
186
+ "agent-beta",
187
+ "agent-gamma",
188
+ ]);
189
+ // Verify all have unique job IDs
190
+ const jobIds = results.map((r) => r.jobId);
191
+ expect(new Set(jobIds).size).toBe(3);
192
+ // Verify fleet status
193
+ const status = await manager.getFleetStatus();
194
+ expect(status.counts.totalAgents).toBe(3);
195
+ await manager.stop();
196
+ });
197
+ it("correctly tracks timing through lifecycle", async () => {
198
+ await createAgentConfig("timing-agent", {
199
+ name: "timing-agent",
200
+ });
201
+ const configPath = await createConfig({
202
+ version: 1,
203
+ agents: [{ path: "./agents/timing-agent.yaml" }],
204
+ });
205
+ const manager = createTestManager(configPath);
206
+ // Before init
207
+ const beforeInit = new Date().toISOString();
208
+ expect(manager.state.initializedAt).toBeNull();
209
+ expect(manager.state.startedAt).toBeNull();
210
+ expect(manager.state.stoppedAt).toBeNull();
211
+ await manager.initialize();
212
+ const afterInit = new Date().toISOString();
213
+ expect(manager.state.initializedAt).not.toBeNull();
214
+ expect(manager.state.initializedAt >= beforeInit).toBe(true);
215
+ expect(manager.state.initializedAt <= afterInit).toBe(true);
216
+ const beforeStart = new Date().toISOString();
217
+ await manager.start();
218
+ await new Promise((resolve) => setTimeout(resolve, 50));
219
+ const afterStart = new Date().toISOString();
220
+ expect(manager.state.startedAt).not.toBeNull();
221
+ expect(manager.state.startedAt >= beforeStart).toBe(true);
222
+ expect(manager.state.startedAt <= afterStart).toBe(true);
223
+ // Check uptime
224
+ const status = await manager.getFleetStatus();
225
+ expect(status.uptimeSeconds).toBeGreaterThanOrEqual(0);
226
+ const beforeStop = new Date().toISOString();
227
+ await manager.stop();
228
+ const afterStop = new Date().toISOString();
229
+ expect(manager.state.stoppedAt).not.toBeNull();
230
+ expect(manager.state.stoppedAt >= beforeStop).toBe(true);
231
+ expect(manager.state.stoppedAt <= afterStop).toBe(true);
232
+ });
233
+ });
234
+ // ==========================================================================
235
+ // Scheduler Integration Tests
236
+ // ==========================================================================
237
+ describe("Scheduler Integration", () => {
238
+ it("scheduler triggers jobs on schedule", async () => {
239
+ await createAgentConfig("scheduled-agent", {
240
+ name: "scheduled-agent",
241
+ schedules: {
242
+ frequent: {
243
+ type: "interval",
244
+ interval: "100ms", // Very short for testing
245
+ prompt: "Scheduled prompt",
246
+ },
247
+ },
248
+ });
249
+ const configPath = await createConfig({
250
+ version: 1,
251
+ agents: [{ path: "./agents/scheduled-agent.yaml" }],
252
+ });
253
+ const manager = new FleetManager({
254
+ configPath,
255
+ stateDir,
256
+ checkInterval: 50, // Check frequently for testing
257
+ logger: createSilentLogger(),
258
+ });
259
+ const scheduleTriggers = [];
260
+ manager.on("schedule:triggered", (payload) => {
261
+ scheduleTriggers.push(payload);
262
+ });
263
+ await manager.initialize();
264
+ await manager.start();
265
+ // Wait for at least one scheduled trigger
266
+ await new Promise((resolve) => setTimeout(resolve, 300));
267
+ await manager.stop();
268
+ // Scheduler should have triggered at least once
269
+ expect(scheduleTriggers.length).toBeGreaterThanOrEqual(1);
270
+ expect(scheduleTriggers[0].agentName).toBe("scheduled-agent");
271
+ expect(scheduleTriggers[0].scheduleName).toBe("frequent");
272
+ });
273
+ it("scheduler respects disabled schedules", async () => {
274
+ await createAgentConfig("disabled-schedule-agent", {
275
+ name: "disabled-schedule-agent",
276
+ schedules: {
277
+ active: {
278
+ type: "interval",
279
+ interval: "5s", // Longer interval to avoid race conditions
280
+ },
281
+ },
282
+ });
283
+ const configPath = await createConfig({
284
+ version: 1,
285
+ agents: [{ path: "./agents/disabled-schedule-agent.yaml" }],
286
+ });
287
+ const manager = new FleetManager({
288
+ configPath,
289
+ stateDir,
290
+ checkInterval: 50,
291
+ logger: createSilentLogger(),
292
+ });
293
+ await manager.initialize();
294
+ // Disable the schedule BEFORE starting
295
+ await manager.disableSchedule("disabled-schedule-agent", "active");
296
+ // Track all triggers
297
+ const triggers = [];
298
+ manager.on("schedule:triggered", () => {
299
+ triggers.push("triggered");
300
+ });
301
+ await manager.start();
302
+ // Wait a bit - disabled schedule should not trigger
303
+ await new Promise((resolve) => setTimeout(resolve, 150));
304
+ await manager.stop();
305
+ // No triggers should have occurred since schedule was disabled
306
+ expect(triggers.length).toBe(0);
307
+ });
308
+ it("getSchedules returns correct schedule information", async () => {
309
+ await createAgentConfig("multi-schedule-agent", {
310
+ name: "multi-schedule-agent",
311
+ schedules: {
312
+ hourly: {
313
+ type: "interval",
314
+ interval: "1h",
315
+ },
316
+ daily: {
317
+ type: "interval",
318
+ interval: "24h",
319
+ },
320
+ },
321
+ });
322
+ const configPath = await createConfig({
323
+ version: 1,
324
+ agents: [{ path: "./agents/multi-schedule-agent.yaml" }],
325
+ });
326
+ const manager = createTestManager(configPath);
327
+ await manager.initialize();
328
+ const schedules = await manager.getSchedules();
329
+ expect(schedules).toHaveLength(2);
330
+ expect(schedules.map((s) => s.name).sort()).toEqual(["daily", "hourly"]);
331
+ expect(schedules.every((s) => s.agentName === "multi-schedule-agent")).toBe(true);
332
+ expect(schedules.every((s) => s.status === "idle")).toBe(true);
333
+ });
334
+ it("getSchedule returns specific schedule", async () => {
335
+ await createAgentConfig("specific-schedule-agent", {
336
+ name: "specific-schedule-agent",
337
+ schedules: {
338
+ target: {
339
+ type: "interval",
340
+ interval: "30m",
341
+ },
342
+ },
343
+ });
344
+ const configPath = await createConfig({
345
+ version: 1,
346
+ agents: [{ path: "./agents/specific-schedule-agent.yaml" }],
347
+ });
348
+ const manager = createTestManager(configPath);
349
+ await manager.initialize();
350
+ const schedule = await manager.getSchedule("specific-schedule-agent", "target");
351
+ expect(schedule.name).toBe("target");
352
+ expect(schedule.agentName).toBe("specific-schedule-agent");
353
+ expect(schedule.type).toBe("interval");
354
+ expect(schedule.interval).toBe("30m");
355
+ });
356
+ it("enableSchedule and disableSchedule toggle schedule status", async () => {
357
+ await createAgentConfig("toggle-agent", {
358
+ name: "toggle-agent",
359
+ schedules: {
360
+ toggleable: {
361
+ type: "interval",
362
+ interval: "1h",
363
+ },
364
+ },
365
+ });
366
+ const configPath = await createConfig({
367
+ version: 1,
368
+ agents: [{ path: "./agents/toggle-agent.yaml" }],
369
+ });
370
+ const manager = createTestManager(configPath);
371
+ await manager.initialize();
372
+ // Initially idle
373
+ let schedule = await manager.getSchedule("toggle-agent", "toggleable");
374
+ expect(schedule.status).toBe("idle");
375
+ // Disable
376
+ await manager.disableSchedule("toggle-agent", "toggleable");
377
+ schedule = await manager.getSchedule("toggle-agent", "toggleable");
378
+ expect(schedule.status).toBe("disabled");
379
+ // Re-enable
380
+ await manager.enableSchedule("toggle-agent", "toggleable");
381
+ schedule = await manager.getSchedule("toggle-agent", "toggleable");
382
+ expect(schedule.status).toBe("idle");
383
+ });
384
+ });
385
+ // ==========================================================================
386
+ // State Persistence Tests
387
+ // ==========================================================================
388
+ describe("State Persistence", () => {
389
+ it("persists fleet state across restart", async () => {
390
+ await createAgentConfig("persistent-agent", {
391
+ name: "persistent-agent",
392
+ schedules: {
393
+ check: {
394
+ type: "interval",
395
+ interval: "1h",
396
+ },
397
+ },
398
+ });
399
+ const configPath = await createConfig({
400
+ version: 1,
401
+ agents: [{ path: "./agents/persistent-agent.yaml" }],
402
+ });
403
+ // First manager instance - trigger a job
404
+ const manager1 = createTestManager(configPath);
405
+ await manager1.initialize();
406
+ await manager1.start();
407
+ await new Promise((resolve) => setTimeout(resolve, 50));
408
+ const triggerResult = await manager1.trigger("persistent-agent", "check");
409
+ const jobId = triggerResult.jobId;
410
+ await manager1.stop();
411
+ // Second manager instance - verify state was persisted
412
+ const manager2 = createTestManager(configPath);
413
+ await manager2.initialize();
414
+ // State should show agent info correctly
415
+ const agentInfo = await manager2.getAgentInfoByName("persistent-agent");
416
+ expect(agentInfo.name).toBe("persistent-agent");
417
+ // Verify the job was created with metadata (stored as YAML)
418
+ const yaml = await import("yaml");
419
+ const jobFilePath = join(stateDir, "jobs", `${jobId}.yaml`);
420
+ const jobContent = await readFile(jobFilePath, "utf-8");
421
+ const metadata = yaml.parse(jobContent);
422
+ expect(metadata.id).toBe(jobId);
423
+ expect(metadata.agent).toBe("persistent-agent");
424
+ });
425
+ it("schedule state survives restart", async () => {
426
+ await createAgentConfig("schedule-persist-agent", {
427
+ name: "schedule-persist-agent",
428
+ schedules: {
429
+ persist: {
430
+ type: "interval",
431
+ interval: "1h",
432
+ },
433
+ },
434
+ });
435
+ const configPath = await createConfig({
436
+ version: 1,
437
+ agents: [{ path: "./agents/schedule-persist-agent.yaml" }],
438
+ });
439
+ // First instance - disable schedule
440
+ const manager1 = createTestManager(configPath);
441
+ await manager1.initialize();
442
+ await manager1.disableSchedule("schedule-persist-agent", "persist");
443
+ let schedule = await manager1.getSchedule("schedule-persist-agent", "persist");
444
+ expect(schedule.status).toBe("disabled");
445
+ // Note: The current implementation may or may not persist schedule disabled state
446
+ // This test documents the expected behavior
447
+ await manager1.stop();
448
+ // Second instance - check if schedule state was preserved
449
+ const manager2 = createTestManager(configPath);
450
+ await manager2.initialize();
451
+ // Get schedule state
452
+ schedule = await manager2.getSchedule("schedule-persist-agent", "persist");
453
+ // Schedule status after restart - depends on implementation
454
+ // Currently schedules start fresh as "idle" on restart
455
+ expect(["idle", "disabled"]).toContain(schedule.status);
456
+ });
457
+ it("job metadata persists to disk", async () => {
458
+ await createAgentConfig("job-persist-agent", {
459
+ name: "job-persist-agent",
460
+ });
461
+ const configPath = await createConfig({
462
+ version: 1,
463
+ agents: [{ path: "./agents/job-persist-agent.yaml" }],
464
+ });
465
+ const manager = createTestManager(configPath);
466
+ await manager.initialize();
467
+ await manager.start();
468
+ await new Promise((resolve) => setTimeout(resolve, 50));
469
+ const result = await manager.trigger("job-persist-agent", undefined, {
470
+ prompt: "Persisted prompt",
471
+ });
472
+ // Verify job metadata was written to disk (stored as YAML)
473
+ const yaml = await import("yaml");
474
+ const jobFilePath = join(stateDir, "jobs", `${result.jobId}.yaml`);
475
+ const metadataContent = await readFile(jobFilePath, "utf-8");
476
+ const metadata = yaml.parse(metadataContent);
477
+ expect(metadata.id).toBe(result.jobId);
478
+ expect(metadata.agent).toBe("job-persist-agent");
479
+ expect(metadata.prompt).toBe("Persisted prompt");
480
+ expect(metadata.trigger_type).toBe("manual");
481
+ await manager.stop();
482
+ });
483
+ it("state directory is created if it does not exist", async () => {
484
+ await createAgentConfig("state-dir-agent", {
485
+ name: "state-dir-agent",
486
+ });
487
+ const configPath = await createConfig({
488
+ version: 1,
489
+ agents: [{ path: "./agents/state-dir-agent.yaml" }],
490
+ });
491
+ // Use a new state directory that doesn't exist
492
+ const newStateDir = join(tempDir, "new-state-dir");
493
+ const manager = new FleetManager({
494
+ configPath,
495
+ stateDir: newStateDir,
496
+ logger: createSilentLogger(),
497
+ });
498
+ await manager.initialize();
499
+ await manager.start();
500
+ await new Promise((resolve) => setTimeout(resolve, 50));
501
+ // Trigger a job to ensure state is persisted
502
+ const result = await manager.trigger("state-dir-agent");
503
+ // Verify job file was created in state directory (stored as YAML)
504
+ const yaml = await import("yaml");
505
+ const jobFilePath = join(newStateDir, "jobs", `${result.jobId}.yaml`);
506
+ const content = await readFile(jobFilePath, "utf-8");
507
+ expect(yaml.parse(content)).toHaveProperty("id", result.jobId);
508
+ await manager.stop();
509
+ });
510
+ });
511
+ // ==========================================================================
512
+ // Edge Case Tests
513
+ // ==========================================================================
514
+ describe("Edge Cases", () => {
515
+ describe("start() edge cases", () => {
516
+ it("throws InvalidStateError when calling start before initialize", async () => {
517
+ const configPath = await createConfig({
518
+ version: 1,
519
+ agents: [],
520
+ });
521
+ const manager = createTestManager(configPath);
522
+ await expect(manager.start()).rejects.toThrow(InvalidStateError);
523
+ await expect(manager.start()).rejects.toMatchObject({
524
+ operation: "start",
525
+ currentState: "uninitialized",
526
+ });
527
+ });
528
+ it("handles start when already running (idempotent)", async () => {
529
+ await createAgentConfig("idempotent-start", {
530
+ name: "idempotent-start",
531
+ });
532
+ const configPath = await createConfig({
533
+ version: 1,
534
+ agents: [{ path: "./agents/idempotent-start.yaml" }],
535
+ });
536
+ const manager = createTestManager(configPath);
537
+ await manager.initialize();
538
+ await manager.start();
539
+ await new Promise((resolve) => setTimeout(resolve, 50));
540
+ expect(manager.state.status).toBe("running");
541
+ // Second start should be safe (idempotent or throw)
542
+ // Based on implementation, may throw InvalidStateError or be no-op
543
+ try {
544
+ await manager.start();
545
+ // If no error, should still be running
546
+ expect(manager.state.status).toBe("running");
547
+ }
548
+ catch (error) {
549
+ expect(error).toBeInstanceOf(InvalidStateError);
550
+ }
551
+ await manager.stop();
552
+ });
553
+ it("requires re-initialization to restart after stop", async () => {
554
+ await createAgentConfig("restart-agent", {
555
+ name: "restart-agent",
556
+ });
557
+ const configPath = await createConfig({
558
+ version: 1,
559
+ agents: [{ path: "./agents/restart-agent.yaml" }],
560
+ });
561
+ const manager = createTestManager(configPath);
562
+ await manager.initialize();
563
+ // First start/stop cycle
564
+ await manager.start();
565
+ await new Promise((resolve) => setTimeout(resolve, 50));
566
+ expect(manager.state.status).toBe("running");
567
+ await manager.stop();
568
+ expect(manager.state.status).toBe("stopped");
569
+ // Cannot restart without re-initialization
570
+ // This documents the current behavior - must create new manager instance
571
+ await expect(manager.start()).rejects.toThrow();
572
+ });
573
+ });
574
+ describe("stop() edge cases", () => {
575
+ it("handles stop when already stopped (idempotent)", async () => {
576
+ await createAgentConfig("idempotent-stop", {
577
+ name: "idempotent-stop",
578
+ });
579
+ const configPath = await createConfig({
580
+ version: 1,
581
+ agents: [{ path: "./agents/idempotent-stop.yaml" }],
582
+ });
583
+ const manager = createTestManager(configPath);
584
+ await manager.initialize();
585
+ await manager.start();
586
+ await new Promise((resolve) => setTimeout(resolve, 50));
587
+ await manager.stop();
588
+ expect(manager.state.status).toBe("stopped");
589
+ // Second stop should be safe
590
+ await manager.stop();
591
+ expect(manager.state.status).toBe("stopped");
592
+ });
593
+ it("handles stop when never started (no-op)", async () => {
594
+ await createAgentConfig("never-started", {
595
+ name: "never-started",
596
+ });
597
+ const configPath = await createConfig({
598
+ version: 1,
599
+ agents: [{ path: "./agents/never-started.yaml" }],
600
+ });
601
+ const manager = createTestManager(configPath);
602
+ await manager.initialize();
603
+ // Stop without ever starting is a no-op - stays in initialized state
604
+ await manager.stop();
605
+ expect(manager.state.status).toBe("initialized");
606
+ });
607
+ it("stop respects timeout option", async () => {
608
+ await createAgentConfig("timeout-agent", {
609
+ name: "timeout-agent",
610
+ });
611
+ const configPath = await createConfig({
612
+ version: 1,
613
+ agents: [{ path: "./agents/timeout-agent.yaml" }],
614
+ });
615
+ const manager = createTestManager(configPath);
616
+ await manager.initialize();
617
+ await manager.start();
618
+ await new Promise((resolve) => setTimeout(resolve, 50));
619
+ // Stop with short timeout
620
+ const beforeStop = Date.now();
621
+ await manager.stop({ timeout: 100 });
622
+ const afterStop = Date.now();
623
+ expect(manager.state.status).toBe("stopped");
624
+ // Stop should complete quickly
625
+ expect(afterStop - beforeStop).toBeLessThan(1000);
626
+ });
627
+ });
628
+ describe("initialize() edge cases", () => {
629
+ it("throws error for invalid config", async () => {
630
+ const configPath = join(configDir, "herdctl.yaml");
631
+ await writeFile(configPath, "invalid: yaml: content:");
632
+ const manager = new FleetManager({
633
+ configPath,
634
+ stateDir,
635
+ logger: createSilentLogger(),
636
+ });
637
+ await expect(manager.initialize()).rejects.toThrow();
638
+ });
639
+ it("throws error for non-existent config", async () => {
640
+ const manager = new FleetManager({
641
+ configPath: "/nonexistent/path/config.yaml",
642
+ stateDir,
643
+ logger: createSilentLogger(),
644
+ });
645
+ await expect(manager.initialize()).rejects.toThrow();
646
+ });
647
+ it("handles re-initialization (idempotent or error)", async () => {
648
+ await createAgentConfig("reinit-agent", {
649
+ name: "reinit-agent",
650
+ });
651
+ const configPath = await createConfig({
652
+ version: 1,
653
+ agents: [{ path: "./agents/reinit-agent.yaml" }],
654
+ });
655
+ const manager = createTestManager(configPath);
656
+ await manager.initialize();
657
+ // Second initialize - should be idempotent or throw
658
+ try {
659
+ await manager.initialize();
660
+ expect(manager.state.status).toBe("initialized");
661
+ }
662
+ catch (error) {
663
+ expect(error).toBeInstanceOf(InvalidStateError);
664
+ }
665
+ });
666
+ });
667
+ describe("trigger() edge cases", () => {
668
+ it("throws AgentNotFoundError for non-existent agent", async () => {
669
+ await createAgentConfig("existing-agent", {
670
+ name: "existing-agent",
671
+ });
672
+ const configPath = await createConfig({
673
+ version: 1,
674
+ agents: [{ path: "./agents/existing-agent.yaml" }],
675
+ });
676
+ const manager = createTestManager(configPath);
677
+ await manager.initialize();
678
+ await expect(manager.trigger("nonexistent-agent")).rejects.toThrow(AgentNotFoundError);
679
+ await expect(manager.trigger("nonexistent-agent")).rejects.toMatchObject({
680
+ agentName: "nonexistent-agent",
681
+ availableAgents: ["existing-agent"],
682
+ });
683
+ });
684
+ it("throws ScheduleNotFoundError for non-existent schedule", async () => {
685
+ await createAgentConfig("schedule-edge-agent", {
686
+ name: "schedule-edge-agent",
687
+ schedules: {
688
+ existing: {
689
+ type: "interval",
690
+ interval: "1h",
691
+ },
692
+ },
693
+ });
694
+ const configPath = await createConfig({
695
+ version: 1,
696
+ agents: [{ path: "./agents/schedule-edge-agent.yaml" }],
697
+ });
698
+ const manager = createTestManager(configPath);
699
+ await manager.initialize();
700
+ await expect(manager.trigger("schedule-edge-agent", "nonexistent")).rejects.toThrow(ScheduleNotFoundError);
701
+ await expect(manager.trigger("schedule-edge-agent", "nonexistent")).rejects.toMatchObject({
702
+ scheduleName: "nonexistent",
703
+ availableSchedules: ["existing"],
704
+ });
705
+ });
706
+ it("throws InvalidStateError before initialize", async () => {
707
+ const configPath = await createConfig({
708
+ version: 1,
709
+ agents: [],
710
+ });
711
+ const manager = createTestManager(configPath);
712
+ await expect(manager.trigger("any-agent")).rejects.toThrow(InvalidStateError);
713
+ });
714
+ it("trigger works after stop", async () => {
715
+ await createAgentConfig("trigger-after-stop", {
716
+ name: "trigger-after-stop",
717
+ });
718
+ const configPath = await createConfig({
719
+ version: 1,
720
+ agents: [{ path: "./agents/trigger-after-stop.yaml" }],
721
+ });
722
+ const manager = createTestManager(configPath);
723
+ await manager.initialize();
724
+ await manager.start();
725
+ await new Promise((resolve) => setTimeout(resolve, 50));
726
+ await manager.stop();
727
+ // Should still be able to trigger after stop
728
+ const result = await manager.trigger("trigger-after-stop");
729
+ expect(result.agentName).toBe("trigger-after-stop");
730
+ });
731
+ });
732
+ describe("getAgentInfoByName() edge cases", () => {
733
+ it("throws AgentNotFoundError before initialize", async () => {
734
+ const configPath = await createConfig({
735
+ version: 1,
736
+ agents: [],
737
+ });
738
+ const manager = createTestManager(configPath);
739
+ await expect(manager.getAgentInfoByName("any")).rejects.toThrow(AgentNotFoundError);
740
+ });
741
+ it("throws AgentNotFoundError for unknown agent", async () => {
742
+ await createAgentConfig("known", {
743
+ name: "known",
744
+ });
745
+ const configPath = await createConfig({
746
+ version: 1,
747
+ agents: [{ path: "./agents/known.yaml" }],
748
+ });
749
+ const manager = createTestManager(configPath);
750
+ await manager.initialize();
751
+ await expect(manager.getAgentInfoByName("unknown")).rejects.toThrow(AgentNotFoundError);
752
+ });
753
+ });
754
+ describe("reload() edge cases", () => {
755
+ it("throws InvalidStateError before initialize", async () => {
756
+ const configPath = await createConfig({
757
+ version: 1,
758
+ agents: [],
759
+ });
760
+ const manager = createTestManager(configPath);
761
+ await expect(manager.reload()).rejects.toThrow(InvalidStateError);
762
+ });
763
+ it("reload works in all valid states", async () => {
764
+ await createAgentConfig("reload-states", {
765
+ name: "reload-states",
766
+ });
767
+ const configPath = await createConfig({
768
+ version: 1,
769
+ agents: [{ path: "./agents/reload-states.yaml" }],
770
+ });
771
+ const manager = createTestManager(configPath);
772
+ // Test in initialized state
773
+ await manager.initialize();
774
+ let result = await manager.reload();
775
+ expect(result.agentCount).toBe(1);
776
+ // Test in running state
777
+ await manager.start();
778
+ await new Promise((resolve) => setTimeout(resolve, 50));
779
+ result = await manager.reload();
780
+ expect(result.agentCount).toBe(1);
781
+ // Test in stopped state
782
+ await manager.stop();
783
+ result = await manager.reload();
784
+ expect(result.agentCount).toBe(1);
785
+ });
786
+ });
787
+ describe("getSchedule() edge cases", () => {
788
+ it("throws AgentNotFoundError for unknown agent", async () => {
789
+ await createAgentConfig("schedule-agent", {
790
+ name: "schedule-agent",
791
+ schedules: {
792
+ test: { type: "interval", interval: "1h" },
793
+ },
794
+ });
795
+ const configPath = await createConfig({
796
+ version: 1,
797
+ agents: [{ path: "./agents/schedule-agent.yaml" }],
798
+ });
799
+ const manager = createTestManager(configPath);
800
+ await manager.initialize();
801
+ await expect(manager.getSchedule("unknown-agent", "test")).rejects.toThrow(AgentNotFoundError);
802
+ });
803
+ it("throws ScheduleNotFoundError for unknown schedule", async () => {
804
+ await createAgentConfig("schedule-not-found", {
805
+ name: "schedule-not-found",
806
+ schedules: {
807
+ exists: { type: "interval", interval: "1h" },
808
+ },
809
+ });
810
+ const configPath = await createConfig({
811
+ version: 1,
812
+ agents: [{ path: "./agents/schedule-not-found.yaml" }],
813
+ });
814
+ const manager = createTestManager(configPath);
815
+ await manager.initialize();
816
+ await expect(manager.getSchedule("schedule-not-found", "does-not-exist")).rejects.toThrow(ScheduleNotFoundError);
817
+ });
818
+ });
819
+ describe("empty fleet edge cases", () => {
820
+ it("handles fleet with no agents", async () => {
821
+ const configPath = await createConfig({
822
+ version: 1,
823
+ agents: [],
824
+ });
825
+ const manager = createTestManager(configPath);
826
+ await manager.initialize();
827
+ expect(manager.state.agentCount).toBe(0);
828
+ const status = await manager.getFleetStatus();
829
+ expect(status.counts.totalAgents).toBe(0);
830
+ expect(status.counts.totalSchedules).toBe(0);
831
+ const agents = await manager.getAgentInfo();
832
+ expect(agents).toEqual([]);
833
+ const schedules = await manager.getSchedules();
834
+ expect(schedules).toEqual([]);
835
+ await manager.start();
836
+ await new Promise((resolve) => setTimeout(resolve, 50));
837
+ expect(manager.state.status).toBe("running");
838
+ await manager.stop();
839
+ expect(manager.state.status).toBe("stopped");
840
+ });
841
+ it("handles agent with no schedules", async () => {
842
+ await createAgentConfig("no-schedules", {
843
+ name: "no-schedules",
844
+ description: "Agent without schedules",
845
+ });
846
+ const configPath = await createConfig({
847
+ version: 1,
848
+ agents: [{ path: "./agents/no-schedules.yaml" }],
849
+ });
850
+ const manager = createTestManager(configPath);
851
+ await manager.initialize();
852
+ const agentInfo = await manager.getAgentInfoByName("no-schedules");
853
+ expect(agentInfo.scheduleCount).toBe(0);
854
+ expect(agentInfo.schedules).toEqual([]);
855
+ // Trigger should still work
856
+ const result = await manager.trigger("no-schedules");
857
+ expect(result.agentName).toBe("no-schedules");
858
+ expect(result.scheduleName).toBeNull();
859
+ });
860
+ });
861
+ describe("concurrency edge cases", () => {
862
+ it("single start/stop cycle completes correctly", async () => {
863
+ await createAgentConfig("rapid-cycle", {
864
+ name: "rapid-cycle",
865
+ });
866
+ const configPath = await createConfig({
867
+ version: 1,
868
+ agents: [{ path: "./agents/rapid-cycle.yaml" }],
869
+ });
870
+ const manager = createTestManager(configPath);
871
+ await manager.initialize();
872
+ // Single start/stop cycle
873
+ await manager.start();
874
+ await new Promise((resolve) => setTimeout(resolve, 20));
875
+ await manager.stop();
876
+ expect(manager.state.status).toBe("stopped");
877
+ });
878
+ it("concurrent triggers to same agent", async () => {
879
+ await createAgentConfig("concurrent-agent", {
880
+ name: "concurrent-agent",
881
+ instances: { max_concurrent: 5 },
882
+ });
883
+ const configPath = await createConfig({
884
+ version: 1,
885
+ agents: [{ path: "./agents/concurrent-agent.yaml" }],
886
+ });
887
+ const manager = createTestManager(configPath);
888
+ await manager.initialize();
889
+ // Trigger multiple jobs concurrently
890
+ const triggers = await Promise.all([
891
+ manager.trigger("concurrent-agent", undefined, { prompt: "Job 1" }),
892
+ manager.trigger("concurrent-agent", undefined, { prompt: "Job 2" }),
893
+ manager.trigger("concurrent-agent", undefined, { prompt: "Job 3" }),
894
+ ]);
895
+ expect(triggers).toHaveLength(3);
896
+ const jobIds = triggers.map((t) => t.jobId);
897
+ expect(new Set(jobIds).size).toBe(3); // All unique IDs
898
+ });
899
+ });
900
+ describe("stop with cancelOnTimeout", () => {
901
+ it("stops with cancelOnTimeout option", async () => {
902
+ await createAgentConfig("cancel-agent", {
903
+ name: "cancel-agent",
904
+ });
905
+ const configPath = await createConfig({
906
+ version: 1,
907
+ agents: [{ path: "./agents/cancel-agent.yaml" }],
908
+ });
909
+ const manager = createTestManager(configPath);
910
+ await manager.initialize();
911
+ await manager.start();
912
+ await new Promise((resolve) => setTimeout(resolve, 50));
913
+ // Stop with cancelOnTimeout enabled (doesn't matter here since no running jobs)
914
+ await manager.stop({
915
+ timeout: 100,
916
+ cancelOnTimeout: true,
917
+ cancelTimeout: 50,
918
+ });
919
+ expect(manager.state.status).toBe("stopped");
920
+ });
921
+ });
922
+ describe("error state handling", () => {
923
+ it("tracks errors during initialization", async () => {
924
+ // Create invalid config with missing agent file
925
+ const configPath = await createConfig({
926
+ version: 1,
927
+ agents: [{ path: "./agents/nonexistent.yaml" }],
928
+ });
929
+ const logger = createSilentLogger();
930
+ const manager = new FleetManager({
931
+ configPath,
932
+ stateDir,
933
+ logger,
934
+ });
935
+ // Initialize should fail
936
+ try {
937
+ await manager.initialize();
938
+ }
939
+ catch {
940
+ // Expected
941
+ }
942
+ // Status should be error
943
+ expect(manager.state.status).toBe("error");
944
+ expect(manager.state.lastError).toBeDefined();
945
+ });
946
+ });
947
+ });
948
+ });
949
+ //# sourceMappingURL=integration.test.js.map