@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,798 @@
1
+ import {
2
+ describe,
3
+ it,
4
+ expect,
5
+ beforeEach,
6
+ afterEach,
7
+ vi,
8
+ } from "vitest";
9
+ import { mkdir, rm, realpath } from "node:fs/promises";
10
+ import { join } from "node:path";
11
+ import { tmpdir } from "node:os";
12
+ import {
13
+ runSchedule,
14
+ buildSchedulePrompt,
15
+ type RunScheduleOptions,
16
+ type ScheduleRunnerLogger,
17
+ } from "../schedule-runner.js";
18
+ import type { ResolvedAgent, Schedule } from "../../config/index.js";
19
+ import type { ScheduleState } from "../../state/schemas/fleet-state.js";
20
+ import type { SDKQueryFunction, SDKMessage } from "../../runner/index.js";
21
+ import type {
22
+ WorkSourceManager,
23
+ WorkItem,
24
+ GetNextWorkItemResult,
25
+ WorkResult,
26
+ } from "../../work-sources/index.js";
27
+ import { readFleetState } from "../../state/fleet-state.js";
28
+
29
+ // Helper to create a temp directory
30
+ async function createTempDir(): Promise<string> {
31
+ const baseDir = join(
32
+ tmpdir(),
33
+ `herdctl-schedule-runner-test-${Date.now()}-${Math.random().toString(36).slice(2)}`
34
+ );
35
+ await mkdir(baseDir, { recursive: true });
36
+ return await realpath(baseDir);
37
+ }
38
+
39
+ // Helper to create a mock logger
40
+ function createMockLogger(): ScheduleRunnerLogger & {
41
+ debugs: string[];
42
+ infos: string[];
43
+ warnings: string[];
44
+ errors: string[];
45
+ } {
46
+ const debugs: string[] = [];
47
+ const infos: string[] = [];
48
+ const warnings: string[] = [];
49
+ const errors: string[] = [];
50
+ return {
51
+ debugs,
52
+ infos,
53
+ warnings,
54
+ errors,
55
+ debug: (message: string) => debugs.push(message),
56
+ info: (message: string) => infos.push(message),
57
+ warn: (message: string) => warnings.push(message),
58
+ error: (message: string) => errors.push(message),
59
+ };
60
+ }
61
+
62
+ // Helper to create a test agent
63
+ function createTestAgent(
64
+ name: string,
65
+ overrides?: Partial<ResolvedAgent>
66
+ ): ResolvedAgent {
67
+ return {
68
+ name,
69
+ configPath: `/fake/path/${name}.yaml`,
70
+ ...overrides,
71
+ } as ResolvedAgent;
72
+ }
73
+
74
+ // Helper to create a test schedule
75
+ function createTestSchedule(overrides?: Partial<Schedule>): Schedule {
76
+ return {
77
+ type: "interval",
78
+ interval: "1h",
79
+ prompt: "Default test prompt",
80
+ ...overrides,
81
+ } as Schedule;
82
+ }
83
+
84
+ // Helper to create a test schedule state
85
+ function createTestScheduleState(overrides?: Partial<ScheduleState>): ScheduleState {
86
+ return {
87
+ status: "idle",
88
+ last_run_at: null,
89
+ next_run_at: null,
90
+ last_error: null,
91
+ ...overrides,
92
+ };
93
+ }
94
+
95
+ // Helper to create a test work item
96
+ function createTestWorkItem(overrides?: Partial<WorkItem>): WorkItem {
97
+ return {
98
+ id: "github:123",
99
+ source: "github",
100
+ externalId: "123",
101
+ title: "Fix authentication bug",
102
+ description: "Users are getting logged out unexpectedly",
103
+ priority: "high",
104
+ labels: ["bug", "auth"],
105
+ metadata: {},
106
+ url: "https://github.com/org/repo/issues/123",
107
+ createdAt: new Date("2024-01-01"),
108
+ updatedAt: new Date("2024-01-02"),
109
+ ...overrides,
110
+ };
111
+ }
112
+
113
+ // Helper to create a mock SDK query function
114
+ function createMockSDKQuery(
115
+ messages: SDKMessage[] = []
116
+ ): SDKQueryFunction {
117
+ return async function* mockQuery() {
118
+ // Emit init message with session_id
119
+ yield {
120
+ type: "system" as const,
121
+ subtype: "init",
122
+ session_id: "test-session-123",
123
+ };
124
+
125
+ // Emit any provided messages
126
+ for (const message of messages) {
127
+ yield message;
128
+ }
129
+
130
+ // Emit result message
131
+ yield {
132
+ type: "assistant" as const,
133
+ content: "Task completed successfully",
134
+ };
135
+ };
136
+ }
137
+
138
+ // Helper to create a mock work source manager
139
+ function createMockWorkSourceManager(
140
+ nextWorkResult?: GetNextWorkItemResult
141
+ ): WorkSourceManager & {
142
+ getNextWorkItemCalls: Array<{ agent: ResolvedAgent }>;
143
+ reportOutcomeCalls: Array<{ taskId: string; result: WorkResult }>;
144
+ releaseWorkItemCalls: Array<{ taskId: string; reason?: string }>;
145
+ } {
146
+ const getNextWorkItemCalls: Array<{ agent: ResolvedAgent }> = [];
147
+ const reportOutcomeCalls: Array<{ taskId: string; result: WorkResult }> = [];
148
+ const releaseWorkItemCalls: Array<{ taskId: string; reason?: string }> = [];
149
+
150
+ return {
151
+ getNextWorkItemCalls,
152
+ reportOutcomeCalls,
153
+ releaseWorkItemCalls,
154
+ getNextWorkItem: vi.fn(async (agent) => {
155
+ getNextWorkItemCalls.push({ agent });
156
+ return nextWorkResult ?? { item: null, claimed: false };
157
+ }),
158
+ reportOutcome: vi.fn(async (taskId, result) => {
159
+ reportOutcomeCalls.push({ taskId, result });
160
+ }),
161
+ releaseWorkItem: vi.fn(async (taskId, options) => {
162
+ releaseWorkItemCalls.push({ taskId, reason: options?.reason });
163
+ return { success: true };
164
+ }),
165
+ getAdapter: vi.fn(async () => null),
166
+ clearCache: vi.fn(),
167
+ };
168
+ }
169
+
170
+ describe("buildSchedulePrompt", () => {
171
+ describe("without work item", () => {
172
+ it("returns schedule prompt when configured", () => {
173
+ const schedule = createTestSchedule({ prompt: "Check for updates" });
174
+ const result = buildSchedulePrompt(schedule);
175
+ expect(result).toBe("Check for updates");
176
+ });
177
+
178
+ it("returns default prompt when no schedule prompt", () => {
179
+ const schedule = createTestSchedule({ prompt: undefined });
180
+ const result = buildSchedulePrompt(schedule);
181
+ expect(result).toBe("Execute scheduled task.");
182
+ });
183
+ });
184
+
185
+ describe("with work item", () => {
186
+ it("combines schedule prompt and work item", () => {
187
+ const schedule = createTestSchedule({ prompt: "Process this issue:" });
188
+ const workItem = createTestWorkItem();
189
+
190
+ const result = buildSchedulePrompt(schedule, workItem);
191
+
192
+ expect(result).toContain("Process this issue:");
193
+ expect(result).toContain("## Work Item: Fix authentication bug");
194
+ expect(result).toContain("Users are getting logged out unexpectedly");
195
+ expect(result).toContain("**Source:** github");
196
+ expect(result).toContain("**Priority:** high");
197
+ expect(result).toContain("**Labels:** bug, auth");
198
+ });
199
+
200
+ it("works with work item alone (no schedule prompt)", () => {
201
+ const schedule = createTestSchedule({ prompt: undefined });
202
+ const workItem = createTestWorkItem({
203
+ title: "Add new feature",
204
+ description: "Implement the widget",
205
+ });
206
+
207
+ const result = buildSchedulePrompt(schedule, workItem);
208
+
209
+ expect(result).toContain("## Work Item: Add new feature");
210
+ expect(result).toContain("Implement the widget");
211
+ expect(result).not.toContain("undefined");
212
+ });
213
+
214
+ it("includes work item URL", () => {
215
+ const schedule = createTestSchedule();
216
+ const workItem = createTestWorkItem({
217
+ url: "https://github.com/org/repo/issues/456",
218
+ });
219
+
220
+ const result = buildSchedulePrompt(schedule, workItem);
221
+
222
+ expect(result).toContain("**URL:** https://github.com/org/repo/issues/456");
223
+ });
224
+
225
+ it("handles work item without labels", () => {
226
+ const schedule = createTestSchedule();
227
+ const workItem = createTestWorkItem({ labels: [] });
228
+
229
+ const result = buildSchedulePrompt(schedule, workItem);
230
+
231
+ expect(result).not.toContain("**Labels:**");
232
+ });
233
+ });
234
+ });
235
+
236
+ describe("runSchedule", () => {
237
+ let tempDir: string;
238
+ let mockLogger: ReturnType<typeof createMockLogger>;
239
+
240
+ beforeEach(async () => {
241
+ tempDir = await createTempDir();
242
+ // Create jobs and sessions directories for job executor
243
+ await mkdir(join(tempDir, "jobs"), { recursive: true });
244
+ await mkdir(join(tempDir, "sessions"), { recursive: true });
245
+ mockLogger = createMockLogger();
246
+ });
247
+
248
+ afterEach(async () => {
249
+ await rm(tempDir, { recursive: true, force: true });
250
+ });
251
+
252
+ describe("basic execution", () => {
253
+ it("executes a schedule successfully", async () => {
254
+ const agent = createTestAgent("test-agent");
255
+ const schedule = createTestSchedule({ prompt: "Do the thing" });
256
+ const sdkQuery = createMockSDKQuery();
257
+
258
+ const result = await runSchedule({
259
+ agent,
260
+ scheduleName: "hourly",
261
+ schedule,
262
+ scheduleState: createTestScheduleState(),
263
+ stateDir: tempDir,
264
+ sdkQuery,
265
+ logger: mockLogger,
266
+ });
267
+
268
+ expect(result.success).toBe(true);
269
+ expect(result.jobId).toBeDefined();
270
+ expect(result.sessionId).toBe("test-session-123");
271
+ expect(result.processedWorkItem).toBe(false);
272
+ expect(result.workItem).toBeUndefined();
273
+ });
274
+
275
+ it("updates schedule state to running during execution", async () => {
276
+ const agent = createTestAgent("test-agent");
277
+ const schedule = createTestSchedule();
278
+
279
+ // Track state during execution
280
+ let stateWhileRunning: ScheduleState | undefined;
281
+
282
+ const sdkQuery: SDKQueryFunction = async function* () {
283
+ // Read state during execution
284
+ const fleetState = await readFleetState(join(tempDir, "state.yaml"));
285
+ stateWhileRunning = fleetState.agents["test-agent"]?.schedules?.hourly;
286
+
287
+ yield { type: "system" as const, subtype: "init", session_id: "test" };
288
+ yield { type: "assistant" as const, content: "Done" };
289
+ };
290
+
291
+ await runSchedule({
292
+ agent,
293
+ scheduleName: "hourly",
294
+ schedule,
295
+ scheduleState: createTestScheduleState(),
296
+ stateDir: tempDir,
297
+ sdkQuery,
298
+ logger: mockLogger,
299
+ });
300
+
301
+ expect(stateWhileRunning?.status).toBe("running");
302
+ });
303
+
304
+ it("updates schedule state with last_run_at after completion", async () => {
305
+ const agent = createTestAgent("test-agent");
306
+ const schedule = createTestSchedule({ interval: "1h" });
307
+ const sdkQuery = createMockSDKQuery();
308
+
309
+ const beforeRun = new Date();
310
+
311
+ await runSchedule({
312
+ agent,
313
+ scheduleName: "hourly",
314
+ schedule,
315
+ scheduleState: createTestScheduleState(),
316
+ stateDir: tempDir,
317
+ sdkQuery,
318
+ logger: mockLogger,
319
+ });
320
+
321
+ const fleetState = await readFleetState(join(tempDir, "state.yaml"));
322
+ const scheduleState = fleetState.agents["test-agent"]?.schedules?.hourly;
323
+
324
+ expect(scheduleState?.status).toBe("idle");
325
+ expect(scheduleState?.last_run_at).toBeDefined();
326
+ expect(new Date(scheduleState!.last_run_at!).getTime()).toBeGreaterThanOrEqual(
327
+ beforeRun.getTime()
328
+ );
329
+ });
330
+
331
+ it("calculates next_run_at based on interval", async () => {
332
+ const agent = createTestAgent("test-agent");
333
+ const schedule = createTestSchedule({ interval: "1h" });
334
+ const sdkQuery = createMockSDKQuery();
335
+
336
+ await runSchedule({
337
+ agent,
338
+ scheduleName: "hourly",
339
+ schedule,
340
+ scheduleState: createTestScheduleState(),
341
+ stateDir: tempDir,
342
+ sdkQuery,
343
+ logger: mockLogger,
344
+ });
345
+
346
+ const fleetState = await readFleetState(join(tempDir, "state.yaml"));
347
+ const scheduleState = fleetState.agents["test-agent"]?.schedules?.hourly;
348
+
349
+ expect(scheduleState?.next_run_at).toBeDefined();
350
+
351
+ // Next run should be approximately 1 hour from now
352
+ const nextRun = new Date(scheduleState!.next_run_at!);
353
+ const now = new Date();
354
+ const diffMs = nextRun.getTime() - now.getTime();
355
+ const diffHours = diffMs / (1000 * 60 * 60);
356
+
357
+ expect(diffHours).toBeGreaterThan(0.9);
358
+ expect(diffHours).toBeLessThan(1.1);
359
+ });
360
+
361
+ it("passes correct trigger type and schedule name to executor", async () => {
362
+ const agent = createTestAgent("test-agent");
363
+ const schedule = createTestSchedule();
364
+
365
+ // We can't easily inspect what was passed to the executor,
366
+ // but we can verify the job was created correctly by checking logs
367
+ const sdkQuery = createMockSDKQuery();
368
+
369
+ const result = await runSchedule({
370
+ agent,
371
+ scheduleName: "my-schedule",
372
+ schedule,
373
+ scheduleState: createTestScheduleState(),
374
+ stateDir: tempDir,
375
+ sdkQuery,
376
+ logger: mockLogger,
377
+ });
378
+
379
+ expect(result.success).toBe(true);
380
+ expect(mockLogger.infos.some((m) => m.includes("my-schedule"))).toBe(true);
381
+ });
382
+ });
383
+
384
+ describe("with work source", () => {
385
+ it("fetches work item when schedule has work_source", async () => {
386
+ const agent = createTestAgent("test-agent");
387
+ const schedule = createTestSchedule({
388
+ prompt: "Process issue:",
389
+ work_source: { type: "github", repo: "org/repo" } as const,
390
+ });
391
+ const workItem = createTestWorkItem();
392
+ const workSourceManager = createMockWorkSourceManager({
393
+ item: workItem,
394
+ claimed: true,
395
+ claimResult: { success: true, workItem },
396
+ });
397
+ const sdkQuery = createMockSDKQuery();
398
+
399
+ const result = await runSchedule({
400
+ agent,
401
+ scheduleName: "hourly",
402
+ schedule,
403
+ scheduleState: createTestScheduleState(),
404
+ stateDir: tempDir,
405
+ sdkQuery,
406
+ workSourceManager,
407
+ logger: mockLogger,
408
+ });
409
+
410
+ expect(result.success).toBe(true);
411
+ expect(result.processedWorkItem).toBe(true);
412
+ expect(result.workItem).toBe(workItem);
413
+ expect(workSourceManager.getNextWorkItemCalls).toHaveLength(1);
414
+ });
415
+
416
+ it("reports outcome to work source after successful execution", async () => {
417
+ const agent = createTestAgent("test-agent");
418
+ const schedule = createTestSchedule({
419
+ work_source: { type: "github", repo: "org/repo" } as const,
420
+ });
421
+ const workItem = createTestWorkItem();
422
+ const workSourceManager = createMockWorkSourceManager({
423
+ item: workItem,
424
+ claimed: true,
425
+ claimResult: { success: true, workItem },
426
+ });
427
+ const sdkQuery = createMockSDKQuery();
428
+
429
+ await runSchedule({
430
+ agent,
431
+ scheduleName: "hourly",
432
+ schedule,
433
+ scheduleState: createTestScheduleState(),
434
+ stateDir: tempDir,
435
+ sdkQuery,
436
+ workSourceManager,
437
+ logger: mockLogger,
438
+ });
439
+
440
+ expect(workSourceManager.reportOutcomeCalls).toHaveLength(1);
441
+ expect(workSourceManager.reportOutcomeCalls[0].taskId).toBe(workItem.id);
442
+ expect(workSourceManager.reportOutcomeCalls[0].result.outcome).toBe("success");
443
+ });
444
+
445
+ it("reports failure outcome when job fails", async () => {
446
+ const agent = createTestAgent("test-agent");
447
+ const schedule = createTestSchedule({
448
+ work_source: { type: "github", repo: "org/repo" } as const,
449
+ });
450
+ const workItem = createTestWorkItem();
451
+ const workSourceManager = createMockWorkSourceManager({
452
+ item: workItem,
453
+ claimed: true,
454
+ claimResult: { success: true, workItem },
455
+ });
456
+
457
+ // Create SDK query that produces an error
458
+ const sdkQuery: SDKQueryFunction = async function* () {
459
+ yield { type: "system" as const, subtype: "init", session_id: "test" };
460
+ yield { type: "error" as const, message: "API error", code: "API_ERROR" };
461
+ };
462
+
463
+ await runSchedule({
464
+ agent,
465
+ scheduleName: "hourly",
466
+ schedule,
467
+ scheduleState: createTestScheduleState(),
468
+ stateDir: tempDir,
469
+ sdkQuery,
470
+ workSourceManager,
471
+ logger: mockLogger,
472
+ });
473
+
474
+ expect(workSourceManager.reportOutcomeCalls).toHaveLength(1);
475
+ expect(workSourceManager.reportOutcomeCalls[0].result.outcome).toBe("failure");
476
+ });
477
+
478
+ it("continues when no work is available", async () => {
479
+ const agent = createTestAgent("test-agent");
480
+ const schedule = createTestSchedule({
481
+ work_source: { type: "github", repo: "org/repo" } as const,
482
+ });
483
+ const workSourceManager = createMockWorkSourceManager({
484
+ item: null,
485
+ claimed: false,
486
+ });
487
+ const sdkQuery = createMockSDKQuery();
488
+
489
+ const result = await runSchedule({
490
+ agent,
491
+ scheduleName: "hourly",
492
+ schedule,
493
+ scheduleState: createTestScheduleState(),
494
+ stateDir: tempDir,
495
+ sdkQuery,
496
+ workSourceManager,
497
+ logger: mockLogger,
498
+ });
499
+
500
+ // Still runs with schedule prompt, just no work item
501
+ expect(result.success).toBe(true);
502
+ expect(result.processedWorkItem).toBe(false);
503
+ expect(result.workItem).toBeUndefined();
504
+ expect(workSourceManager.reportOutcomeCalls).toHaveLength(0);
505
+ });
506
+
507
+ it("handles claim failure gracefully", async () => {
508
+ const agent = createTestAgent("test-agent");
509
+ const schedule = createTestSchedule({
510
+ work_source: { type: "github", repo: "org/repo" } as const,
511
+ });
512
+ const workItem = createTestWorkItem();
513
+ const workSourceManager = createMockWorkSourceManager({
514
+ item: workItem,
515
+ claimed: false,
516
+ claimResult: { success: false, reason: "already_claimed" },
517
+ });
518
+ const sdkQuery = createMockSDKQuery();
519
+
520
+ const result = await runSchedule({
521
+ agent,
522
+ scheduleName: "hourly",
523
+ schedule,
524
+ scheduleState: createTestScheduleState(),
525
+ stateDir: tempDir,
526
+ sdkQuery,
527
+ workSourceManager,
528
+ logger: mockLogger,
529
+ });
530
+
531
+ // Should still run, just without work item
532
+ expect(result.success).toBe(true);
533
+ expect(result.processedWorkItem).toBe(false);
534
+ expect(mockLogger.warnings.some((m) => m.includes("claim failed"))).toBe(true);
535
+ });
536
+
537
+ it("skips work source when no manager provided", async () => {
538
+ const agent = createTestAgent("test-agent");
539
+ const schedule = createTestSchedule({
540
+ prompt: "Test",
541
+ work_source: { type: "github", repo: "org/repo" } as const,
542
+ });
543
+ const sdkQuery = createMockSDKQuery();
544
+
545
+ const result = await runSchedule({
546
+ agent,
547
+ scheduleName: "hourly",
548
+ schedule,
549
+ scheduleState: createTestScheduleState(),
550
+ stateDir: tempDir,
551
+ sdkQuery,
552
+ // No workSourceManager provided
553
+ logger: mockLogger,
554
+ });
555
+
556
+ expect(result.success).toBe(true);
557
+ expect(result.processedWorkItem).toBe(false);
558
+ });
559
+ });
560
+
561
+ describe("error handling", () => {
562
+ it("reports failure outcome when SDK throws", async () => {
563
+ const agent = createTestAgent("test-agent");
564
+ const schedule = createTestSchedule({
565
+ work_source: { type: "github", repo: "org/repo" } as const,
566
+ });
567
+ const workItem = createTestWorkItem();
568
+ const workSourceManager = createMockWorkSourceManager({
569
+ item: workItem,
570
+ claimed: true,
571
+ claimResult: { success: true, workItem },
572
+ });
573
+
574
+ // Create SDK query that throws - JobExecutor catches this and returns failed result
575
+ const sdkQuery: SDKQueryFunction = async function* () {
576
+ throw new Error("Unexpected SDK failure");
577
+ };
578
+
579
+ const result = await runSchedule({
580
+ agent,
581
+ scheduleName: "hourly",
582
+ schedule,
583
+ scheduleState: createTestScheduleState(),
584
+ stateDir: tempDir,
585
+ sdkQuery,
586
+ workSourceManager,
587
+ logger: mockLogger,
588
+ });
589
+
590
+ // JobExecutor catches SDK errors and returns failed result
591
+ expect(result.success).toBe(false);
592
+ expect(result.error).toBeDefined();
593
+
594
+ // Work outcome should still be reported as failure
595
+ expect(workSourceManager.reportOutcomeCalls).toHaveLength(1);
596
+ expect(workSourceManager.reportOutcomeCalls[0].result.outcome).toBe("failure");
597
+ });
598
+
599
+ it("records error in schedule state on SDK failure", async () => {
600
+ const agent = createTestAgent("test-agent");
601
+ const schedule = createTestSchedule();
602
+
603
+ // Create SDK query that throws - JobExecutor catches this
604
+ const sdkQuery: SDKQueryFunction = async function* () {
605
+ throw new Error("Execution failed");
606
+ };
607
+
608
+ const result = await runSchedule({
609
+ agent,
610
+ scheduleName: "hourly",
611
+ schedule,
612
+ scheduleState: createTestScheduleState(),
613
+ stateDir: tempDir,
614
+ sdkQuery,
615
+ logger: mockLogger,
616
+ });
617
+
618
+ expect(result.success).toBe(false);
619
+
620
+ const fleetState = await readFleetState(join(tempDir, "state.yaml"));
621
+ const scheduleState = fleetState.agents["test-agent"]?.schedules?.hourly;
622
+
623
+ expect(scheduleState?.status).toBe("idle");
624
+ // Error message contains the original error but may be wrapped
625
+ expect(scheduleState?.last_error).toContain("Execution failed");
626
+ });
627
+
628
+ it("still calculates next_run_at on SDK failure", async () => {
629
+ const agent = createTestAgent("test-agent");
630
+ const schedule = createTestSchedule({ interval: "30m" });
631
+
632
+ const sdkQuery: SDKQueryFunction = async function* () {
633
+ throw new Error("Execution failed");
634
+ };
635
+
636
+ const result = await runSchedule({
637
+ agent,
638
+ scheduleName: "hourly",
639
+ schedule,
640
+ scheduleState: createTestScheduleState(),
641
+ stateDir: tempDir,
642
+ sdkQuery,
643
+ logger: mockLogger,
644
+ });
645
+
646
+ expect(result.success).toBe(false);
647
+
648
+ const fleetState = await readFleetState(join(tempDir, "state.yaml"));
649
+ const scheduleState = fleetState.agents["test-agent"]?.schedules?.hourly;
650
+
651
+ // Should still have calculated next_run_at
652
+ expect(scheduleState?.next_run_at).toBeDefined();
653
+ });
654
+
655
+ it("continues if reporting outcome fails", async () => {
656
+ const agent = createTestAgent("test-agent");
657
+ const schedule = createTestSchedule({
658
+ work_source: { type: "github", repo: "org/repo" } as const,
659
+ });
660
+ const workItem = createTestWorkItem();
661
+ const workSourceManager = createMockWorkSourceManager({
662
+ item: workItem,
663
+ claimed: true,
664
+ claimResult: { success: true, workItem },
665
+ });
666
+
667
+ // Make reportOutcome throw
668
+ (workSourceManager.reportOutcome as ReturnType<typeof vi.fn>).mockRejectedValue(
669
+ new Error("Report failed")
670
+ );
671
+
672
+ const sdkQuery = createMockSDKQuery();
673
+
674
+ const result = await runSchedule({
675
+ agent,
676
+ scheduleName: "hourly",
677
+ schedule,
678
+ scheduleState: createTestScheduleState(),
679
+ stateDir: tempDir,
680
+ sdkQuery,
681
+ workSourceManager,
682
+ logger: mockLogger,
683
+ });
684
+
685
+ // Should still succeed overall
686
+ expect(result.success).toBe(true);
687
+ expect(mockLogger.errors.some((m) => m.includes("Report failed"))).toBe(true);
688
+ });
689
+ });
690
+
691
+ describe("logging", () => {
692
+ it("logs schedule start", async () => {
693
+ const agent = createTestAgent("my-agent");
694
+ const schedule = createTestSchedule();
695
+ const sdkQuery = createMockSDKQuery();
696
+
697
+ await runSchedule({
698
+ agent,
699
+ scheduleName: "my-schedule",
700
+ schedule,
701
+ scheduleState: createTestScheduleState(),
702
+ stateDir: tempDir,
703
+ sdkQuery,
704
+ logger: mockLogger,
705
+ });
706
+
707
+ expect(
708
+ mockLogger.infos.some(
709
+ (m) => m.includes("Running") && m.includes("my-agent/my-schedule")
710
+ )
711
+ ).toBe(true);
712
+ });
713
+
714
+ it("logs schedule completion", async () => {
715
+ const agent = createTestAgent("my-agent");
716
+ const schedule = createTestSchedule();
717
+ const sdkQuery = createMockSDKQuery();
718
+
719
+ await runSchedule({
720
+ agent,
721
+ scheduleName: "my-schedule",
722
+ schedule,
723
+ scheduleState: createTestScheduleState(),
724
+ stateDir: tempDir,
725
+ sdkQuery,
726
+ logger: mockLogger,
727
+ });
728
+
729
+ expect(
730
+ mockLogger.infos.some(
731
+ (m) => m.includes("Completed") && m.includes("my-agent/my-schedule")
732
+ )
733
+ ).toBe(true);
734
+ });
735
+
736
+ it("logs work item claim", async () => {
737
+ const agent = createTestAgent("test-agent");
738
+ const schedule = createTestSchedule({
739
+ work_source: { type: "github", repo: "org/repo" } as const,
740
+ });
741
+ const workItem = createTestWorkItem({ title: "Important task" });
742
+ const workSourceManager = createMockWorkSourceManager({
743
+ item: workItem,
744
+ claimed: true,
745
+ claimResult: { success: true, workItem },
746
+ });
747
+ const sdkQuery = createMockSDKQuery();
748
+
749
+ await runSchedule({
750
+ agent,
751
+ scheduleName: "hourly",
752
+ schedule,
753
+ scheduleState: createTestScheduleState(),
754
+ stateDir: tempDir,
755
+ sdkQuery,
756
+ workSourceManager,
757
+ logger: mockLogger,
758
+ });
759
+
760
+ expect(
761
+ mockLogger.infos.some(
762
+ (m) => m.includes("Claimed") && m.includes("Important task")
763
+ )
764
+ ).toBe(true);
765
+ });
766
+
767
+ it("logs outcome reporting", async () => {
768
+ const agent = createTestAgent("test-agent");
769
+ const schedule = createTestSchedule({
770
+ work_source: { type: "github", repo: "org/repo" } as const,
771
+ });
772
+ const workItem = createTestWorkItem();
773
+ const workSourceManager = createMockWorkSourceManager({
774
+ item: workItem,
775
+ claimed: true,
776
+ claimResult: { success: true, workItem },
777
+ });
778
+ const sdkQuery = createMockSDKQuery();
779
+
780
+ await runSchedule({
781
+ agent,
782
+ scheduleName: "hourly",
783
+ schedule,
784
+ scheduleState: createTestScheduleState(),
785
+ stateDir: tempDir,
786
+ sdkQuery,
787
+ workSourceManager,
788
+ logger: mockLogger,
789
+ });
790
+
791
+ expect(
792
+ mockLogger.infos.some(
793
+ (m) => m.includes("Reported outcome") && m.includes(workItem.id)
794
+ )
795
+ ).toBe(true);
796
+ });
797
+ });
798
+ });