@herdctl/core 0.0.1 → 0.0.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (275) hide show
  1. package/dist/config/__tests__/agent.test.js +31 -13
  2. package/dist/config/__tests__/agent.test.js.map +1 -1
  3. package/dist/config/__tests__/merge.test.js +9 -2
  4. package/dist/config/__tests__/merge.test.js.map +1 -1
  5. package/dist/config/__tests__/schema.test.js +350 -1
  6. package/dist/config/__tests__/schema.test.js.map +1 -1
  7. package/dist/config/index.d.ts +1 -1
  8. package/dist/config/index.d.ts.map +1 -1
  9. package/dist/config/index.js +3 -1
  10. package/dist/config/index.js.map +1 -1
  11. package/dist/config/schema.d.ts +828 -24
  12. package/dist/config/schema.d.ts.map +1 -1
  13. package/dist/config/schema.js +118 -6
  14. package/dist/config/schema.js.map +1 -1
  15. package/dist/fleet-manager/__tests__/coverage.test.js +11 -332
  16. package/dist/fleet-manager/__tests__/coverage.test.js.map +1 -1
  17. package/dist/fleet-manager/__tests__/errors.test.js +1 -49
  18. package/dist/fleet-manager/__tests__/errors.test.js.map +1 -1
  19. package/dist/fleet-manager/__tests__/integration.test.js +109 -0
  20. package/dist/fleet-manager/__tests__/integration.test.js.map +1 -1
  21. package/dist/fleet-manager/__tests__/reload.test.js +1 -1
  22. package/dist/fleet-manager/__tests__/reload.test.js.map +1 -1
  23. package/dist/fleet-manager/config-reload.d.ts +164 -0
  24. package/dist/fleet-manager/config-reload.d.ts.map +1 -0
  25. package/dist/fleet-manager/config-reload.js +445 -0
  26. package/dist/fleet-manager/config-reload.js.map +1 -0
  27. package/dist/fleet-manager/context.d.ts +76 -0
  28. package/dist/fleet-manager/context.d.ts.map +1 -0
  29. package/dist/fleet-manager/context.js +11 -0
  30. package/dist/fleet-manager/context.js.map +1 -0
  31. package/dist/fleet-manager/errors.d.ts +0 -25
  32. package/dist/fleet-manager/errors.d.ts.map +1 -1
  33. package/dist/fleet-manager/errors.js +0 -38
  34. package/dist/fleet-manager/errors.js.map +1 -1
  35. package/dist/fleet-manager/event-emitters.d.ts +123 -0
  36. package/dist/fleet-manager/event-emitters.d.ts.map +1 -0
  37. package/dist/fleet-manager/event-emitters.js +136 -0
  38. package/dist/fleet-manager/event-emitters.js.map +1 -0
  39. package/dist/fleet-manager/event-types.d.ts +0 -15
  40. package/dist/fleet-manager/event-types.d.ts.map +1 -1
  41. package/dist/fleet-manager/fleet-manager.d.ts +40 -653
  42. package/dist/fleet-manager/fleet-manager.d.ts.map +1 -1
  43. package/dist/fleet-manager/fleet-manager.js +95 -1720
  44. package/dist/fleet-manager/fleet-manager.js.map +1 -1
  45. package/dist/fleet-manager/index.d.ts +13 -2
  46. package/dist/fleet-manager/index.d.ts.map +1 -1
  47. package/dist/fleet-manager/index.js +19 -6
  48. package/dist/fleet-manager/index.js.map +1 -1
  49. package/dist/fleet-manager/job-control.d.ts +64 -0
  50. package/dist/fleet-manager/job-control.d.ts.map +1 -0
  51. package/dist/fleet-manager/job-control.js +296 -0
  52. package/dist/fleet-manager/job-control.js.map +1 -0
  53. package/dist/fleet-manager/log-streaming.d.ts +171 -0
  54. package/dist/fleet-manager/log-streaming.d.ts.map +1 -0
  55. package/dist/fleet-manager/log-streaming.js +503 -0
  56. package/dist/fleet-manager/log-streaming.js.map +1 -0
  57. package/dist/fleet-manager/schedule-executor.d.ts +63 -0
  58. package/dist/fleet-manager/schedule-executor.d.ts.map +1 -0
  59. package/dist/fleet-manager/schedule-executor.js +209 -0
  60. package/dist/fleet-manager/schedule-executor.js.map +1 -0
  61. package/dist/fleet-manager/schedule-management.d.ts +71 -0
  62. package/dist/fleet-manager/schedule-management.d.ts.map +1 -0
  63. package/dist/fleet-manager/schedule-management.js +171 -0
  64. package/dist/fleet-manager/schedule-management.js.map +1 -0
  65. package/dist/fleet-manager/status-queries.d.ts +105 -0
  66. package/dist/fleet-manager/status-queries.d.ts.map +1 -0
  67. package/dist/fleet-manager/status-queries.js +247 -0
  68. package/dist/fleet-manager/status-queries.js.map +1 -0
  69. package/dist/fleet-manager/types.d.ts +0 -39
  70. package/dist/fleet-manager/types.d.ts.map +1 -1
  71. package/dist/runner/__tests__/job-executor.test.js +206 -1
  72. package/dist/runner/__tests__/job-executor.test.js.map +1 -1
  73. package/dist/runner/job-executor.d.ts +9 -0
  74. package/dist/runner/job-executor.d.ts.map +1 -1
  75. package/dist/runner/job-executor.js +78 -4
  76. package/dist/runner/job-executor.js.map +1 -1
  77. package/dist/runner/types.d.ts +2 -0
  78. package/dist/runner/types.d.ts.map +1 -1
  79. package/dist/scheduler/__tests__/cron.test.d.ts +2 -0
  80. package/dist/scheduler/__tests__/cron.test.d.ts.map +1 -0
  81. package/dist/scheduler/__tests__/cron.test.js +867 -0
  82. package/dist/scheduler/__tests__/cron.test.js.map +1 -0
  83. package/dist/scheduler/__tests__/scheduler.test.js +164 -5
  84. package/dist/scheduler/__tests__/scheduler.test.js.map +1 -1
  85. package/dist/scheduler/cron.d.ts +126 -0
  86. package/dist/scheduler/cron.d.ts.map +1 -0
  87. package/dist/scheduler/cron.js +390 -0
  88. package/dist/scheduler/cron.js.map +1 -0
  89. package/dist/scheduler/errors.d.ts +81 -1
  90. package/dist/scheduler/errors.d.ts.map +1 -1
  91. package/dist/scheduler/errors.js +81 -6
  92. package/dist/scheduler/errors.js.map +1 -1
  93. package/dist/scheduler/index.d.ts +1 -0
  94. package/dist/scheduler/index.d.ts.map +1 -1
  95. package/dist/scheduler/index.js +2 -0
  96. package/dist/scheduler/index.js.map +1 -1
  97. package/dist/scheduler/schedule-runner.d.ts +2 -2
  98. package/dist/scheduler/schedule-runner.d.ts.map +1 -1
  99. package/dist/scheduler/schedule-runner.js +20 -8
  100. package/dist/scheduler/schedule-runner.js.map +1 -1
  101. package/dist/scheduler/scheduler.d.ts +4 -4
  102. package/dist/scheduler/scheduler.d.ts.map +1 -1
  103. package/dist/scheduler/scheduler.js +86 -20
  104. package/dist/scheduler/scheduler.js.map +1 -1
  105. package/dist/scheduler/types.d.ts +1 -1
  106. package/dist/scheduler/types.d.ts.map +1 -1
  107. package/dist/state/schemas/job-metadata.d.ts +2 -2
  108. package/package.json +33 -8
  109. package/.turbo/turbo-build.log +0 -4
  110. package/.turbo/turbo-test.log +0 -219
  111. package/.turbo/turbo-typecheck.log +0 -4
  112. package/coverage/base.css +0 -224
  113. package/coverage/block-navigation.js +0 -87
  114. package/coverage/coverage-final.json +0 -51
  115. package/coverage/favicon.png +0 -0
  116. package/coverage/index.html +0 -251
  117. package/coverage/prettify.css +0 -1
  118. package/coverage/prettify.js +0 -2
  119. package/coverage/sort-arrow-sprite.png +0 -0
  120. package/coverage/sorter.js +0 -210
  121. package/coverage/src/config/index.html +0 -191
  122. package/coverage/src/config/index.ts.html +0 -442
  123. package/coverage/src/config/interpolate.ts.html +0 -652
  124. package/coverage/src/config/loader.ts.html +0 -1501
  125. package/coverage/src/config/merge.ts.html +0 -823
  126. package/coverage/src/config/parser.ts.html +0 -1213
  127. package/coverage/src/config/schema.ts.html +0 -1123
  128. package/coverage/src/fleet-manager/errors.ts.html +0 -2326
  129. package/coverage/src/fleet-manager/event-types.ts.html +0 -1219
  130. package/coverage/src/fleet-manager/fleet-manager.ts.html +0 -7030
  131. package/coverage/src/fleet-manager/index.html +0 -206
  132. package/coverage/src/fleet-manager/index.ts.html +0 -469
  133. package/coverage/src/fleet-manager/job-manager.ts.html +0 -2074
  134. package/coverage/src/fleet-manager/job-queue.ts.html +0 -2479
  135. package/coverage/src/fleet-manager/types.ts.html +0 -2602
  136. package/coverage/src/index.html +0 -116
  137. package/coverage/src/index.ts.html +0 -181
  138. package/coverage/src/runner/errors.ts.html +0 -1006
  139. package/coverage/src/runner/index.html +0 -191
  140. package/coverage/src/runner/index.ts.html +0 -256
  141. package/coverage/src/runner/job-executor.ts.html +0 -1429
  142. package/coverage/src/runner/message-processor.ts.html +0 -1150
  143. package/coverage/src/runner/sdk-adapter.ts.html +0 -658
  144. package/coverage/src/runner/types.ts.html +0 -559
  145. package/coverage/src/scheduler/errors.ts.html +0 -388
  146. package/coverage/src/scheduler/index.html +0 -206
  147. package/coverage/src/scheduler/index.ts.html +0 -244
  148. package/coverage/src/scheduler/interval.ts.html +0 -652
  149. package/coverage/src/scheduler/schedule-runner.ts.html +0 -1411
  150. package/coverage/src/scheduler/schedule-state.ts.html +0 -718
  151. package/coverage/src/scheduler/scheduler.ts.html +0 -1795
  152. package/coverage/src/scheduler/types.ts.html +0 -733
  153. package/coverage/src/state/directory.ts.html +0 -736
  154. package/coverage/src/state/errors.ts.html +0 -376
  155. package/coverage/src/state/fleet-state.ts.html +0 -937
  156. package/coverage/src/state/index.html +0 -221
  157. package/coverage/src/state/index.ts.html +0 -322
  158. package/coverage/src/state/job-metadata.ts.html +0 -1420
  159. package/coverage/src/state/job-output.ts.html +0 -1033
  160. package/coverage/src/state/schemas/fleet-state.ts.html +0 -445
  161. package/coverage/src/state/schemas/index.html +0 -176
  162. package/coverage/src/state/schemas/index.ts.html +0 -286
  163. package/coverage/src/state/schemas/job-metadata.ts.html +0 -628
  164. package/coverage/src/state/schemas/job-output.ts.html +0 -616
  165. package/coverage/src/state/schemas/session-info.ts.html +0 -361
  166. package/coverage/src/state/session.ts.html +0 -844
  167. package/coverage/src/state/types.ts.html +0 -262
  168. package/coverage/src/state/utils/atomic.ts.html +0 -748
  169. package/coverage/src/state/utils/index.html +0 -146
  170. package/coverage/src/state/utils/index.ts.html +0 -103
  171. package/coverage/src/state/utils/reads.ts.html +0 -1621
  172. package/coverage/src/work-sources/adapters/github.ts.html +0 -3583
  173. package/coverage/src/work-sources/adapters/index.html +0 -131
  174. package/coverage/src/work-sources/adapters/index.ts.html +0 -277
  175. package/coverage/src/work-sources/errors.ts.html +0 -298
  176. package/coverage/src/work-sources/index.html +0 -176
  177. package/coverage/src/work-sources/index.ts.html +0 -529
  178. package/coverage/src/work-sources/manager.ts.html +0 -1324
  179. package/coverage/src/work-sources/registry.ts.html +0 -619
  180. package/coverage/src/work-sources/types.ts.html +0 -568
  181. package/dist/fleet-manager/__tests__/event-helpers.test.d.ts +0 -7
  182. package/dist/fleet-manager/__tests__/event-helpers.test.d.ts.map +0 -1
  183. package/dist/fleet-manager/__tests__/event-helpers.test.js +0 -368
  184. package/dist/fleet-manager/__tests__/event-helpers.test.js.map +0 -1
  185. package/src/config/__tests__/agent.test.ts +0 -864
  186. package/src/config/__tests__/interpolate.test.ts +0 -644
  187. package/src/config/__tests__/loader.test.ts +0 -784
  188. package/src/config/__tests__/merge.test.ts +0 -751
  189. package/src/config/__tests__/parser.test.ts +0 -533
  190. package/src/config/__tests__/schema.test.ts +0 -873
  191. package/src/config/index.ts +0 -119
  192. package/src/config/interpolate.ts +0 -189
  193. package/src/config/loader.ts +0 -472
  194. package/src/config/merge.ts +0 -246
  195. package/src/config/parser.ts +0 -376
  196. package/src/config/schema.ts +0 -346
  197. package/src/fleet-manager/__tests__/coverage.test.ts +0 -2869
  198. package/src/fleet-manager/__tests__/errors.test.ts +0 -660
  199. package/src/fleet-manager/__tests__/event-helpers.test.ts +0 -448
  200. package/src/fleet-manager/__tests__/integration.test.ts +0 -1209
  201. package/src/fleet-manager/__tests__/job-control.test.ts +0 -283
  202. package/src/fleet-manager/__tests__/job-manager.test.ts +0 -869
  203. package/src/fleet-manager/__tests__/job-queue.test.ts +0 -401
  204. package/src/fleet-manager/__tests__/reload.test.ts +0 -751
  205. package/src/fleet-manager/__tests__/status-queries.test.ts +0 -595
  206. package/src/fleet-manager/__tests__/trigger.test.ts +0 -601
  207. package/src/fleet-manager/errors.ts +0 -747
  208. package/src/fleet-manager/event-types.ts +0 -378
  209. package/src/fleet-manager/fleet-manager.ts +0 -2315
  210. package/src/fleet-manager/index.ts +0 -128
  211. package/src/fleet-manager/job-manager.ts +0 -663
  212. package/src/fleet-manager/job-queue.ts +0 -798
  213. package/src/fleet-manager/types.ts +0 -839
  214. package/src/index.ts +0 -32
  215. package/src/runner/__tests__/errors.test.ts +0 -382
  216. package/src/runner/__tests__/job-executor.test.ts +0 -1708
  217. package/src/runner/__tests__/message-processor.test.ts +0 -960
  218. package/src/runner/__tests__/sdk-adapter.test.ts +0 -626
  219. package/src/runner/errors.ts +0 -307
  220. package/src/runner/index.ts +0 -57
  221. package/src/runner/job-executor.ts +0 -448
  222. package/src/runner/message-processor.ts +0 -355
  223. package/src/runner/sdk-adapter.ts +0 -191
  224. package/src/runner/types.ts +0 -158
  225. package/src/scheduler/__tests__/errors.test.ts +0 -159
  226. package/src/scheduler/__tests__/interval.test.ts +0 -515
  227. package/src/scheduler/__tests__/schedule-runner.test.ts +0 -798
  228. package/src/scheduler/__tests__/schedule-state.test.ts +0 -671
  229. package/src/scheduler/__tests__/scheduler.test.ts +0 -1280
  230. package/src/scheduler/errors.ts +0 -101
  231. package/src/scheduler/index.ts +0 -53
  232. package/src/scheduler/interval.ts +0 -189
  233. package/src/scheduler/schedule-runner.ts +0 -442
  234. package/src/scheduler/schedule-state.ts +0 -211
  235. package/src/scheduler/scheduler.ts +0 -570
  236. package/src/scheduler/types.ts +0 -216
  237. package/src/state/__tests__/directory.test.ts +0 -595
  238. package/src/state/__tests__/fleet-state.test.ts +0 -868
  239. package/src/state/__tests__/job-metadata-schema.test.ts +0 -414
  240. package/src/state/__tests__/job-metadata.test.ts +0 -831
  241. package/src/state/__tests__/job-output.test.ts +0 -856
  242. package/src/state/__tests__/session-schema.test.ts +0 -378
  243. package/src/state/__tests__/session.test.ts +0 -604
  244. package/src/state/directory.ts +0 -217
  245. package/src/state/errors.ts +0 -97
  246. package/src/state/fleet-state.ts +0 -284
  247. package/src/state/index.ts +0 -79
  248. package/src/state/job-metadata.ts +0 -445
  249. package/src/state/job-output.ts +0 -316
  250. package/src/state/schemas/__tests__/job-output.test.ts +0 -338
  251. package/src/state/schemas/fleet-state.ts +0 -120
  252. package/src/state/schemas/index.ts +0 -67
  253. package/src/state/schemas/job-metadata.ts +0 -181
  254. package/src/state/schemas/job-output.ts +0 -177
  255. package/src/state/schemas/session-info.ts +0 -92
  256. package/src/state/session.ts +0 -253
  257. package/src/state/types.ts +0 -59
  258. package/src/state/utils/__tests__/atomic.test.ts +0 -723
  259. package/src/state/utils/__tests__/reads.test.ts +0 -1071
  260. package/src/state/utils/atomic.ts +0 -221
  261. package/src/state/utils/index.ts +0 -6
  262. package/src/state/utils/reads.ts +0 -512
  263. package/src/work-sources/__tests__/github.test.ts +0 -1800
  264. package/src/work-sources/__tests__/manager.test.ts +0 -529
  265. package/src/work-sources/__tests__/registry.test.ts +0 -477
  266. package/src/work-sources/__tests__/types.test.ts +0 -479
  267. package/src/work-sources/adapters/github.ts +0 -1166
  268. package/src/work-sources/adapters/index.ts +0 -64
  269. package/src/work-sources/errors.ts +0 -71
  270. package/src/work-sources/index.ts +0 -148
  271. package/src/work-sources/manager.ts +0 -413
  272. package/src/work-sources/registry.ts +0 -178
  273. package/src/work-sources/types.ts +0 -161
  274. package/tsconfig.json +0 -9
  275. package/vitest.config.ts +0 -19
@@ -1,448 +0,0 @@
1
- /**
2
- * Job executor for running agents with streaming output to job logs
3
- *
4
- * Manages the lifecycle of agent execution including:
5
- * - Creating job records before execution
6
- * - Streaming all SDK messages to job output in real-time
7
- * - Updating job status and metadata on completion
8
- */
9
-
10
- import { join } from "node:path";
11
- import type {
12
- RunnerOptions,
13
- RunnerOptionsWithCallbacks,
14
- RunnerResult,
15
- RunnerErrorDetails,
16
- SDKMessage,
17
- } from "./types.js";
18
- import {
19
- RunnerError,
20
- SDKInitializationError,
21
- SDKStreamingError,
22
- MalformedResponseError,
23
- wrapError,
24
- classifyError,
25
- buildErrorMessage,
26
- } from "./errors.js";
27
- import { toSDKOptions } from "./sdk-adapter.js";
28
- import {
29
- processSDKMessage,
30
- isTerminalMessage,
31
- extractSummary,
32
- } from "./message-processor.js";
33
- import {
34
- createJob,
35
- updateJob,
36
- appendJobOutput,
37
- getJobOutputPath,
38
- updateSessionInfo,
39
- getSessionInfo,
40
- type JobMetadata,
41
- type TriggerType,
42
- type SessionInfo,
43
- } from "../state/index.js";
44
-
45
- // =============================================================================
46
- // Types
47
- // =============================================================================
48
-
49
- /**
50
- * Logger interface for job executor
51
- */
52
- export interface JobExecutorLogger {
53
- warn: (message: string) => void;
54
- error: (message: string) => void;
55
- info?: (message: string) => void;
56
- }
57
-
58
- /**
59
- * Options for job executor
60
- */
61
- export interface JobExecutorOptions {
62
- /** Logger for warnings and errors */
63
- logger?: JobExecutorLogger;
64
- }
65
-
66
- /**
67
- * SDK query function type (for dependency injection)
68
- */
69
- export type SDKQueryFunction = (params: {
70
- prompt: string;
71
- options?: Record<string, unknown>;
72
- abortController?: AbortController;
73
- }) => AsyncIterable<SDKMessage>;
74
-
75
- // =============================================================================
76
- // Default Logger
77
- // =============================================================================
78
-
79
- const defaultLogger: JobExecutorLogger = {
80
- warn: (message: string) => console.warn(`[herdctl] ${message}`),
81
- error: (message: string) => console.error(`[herdctl] ${message}`),
82
- info: (message: string) => console.info(`[herdctl] ${message}`),
83
- };
84
-
85
- // =============================================================================
86
- // Job Executor Class
87
- // =============================================================================
88
-
89
- /**
90
- * Executes agents with streaming output to job logs
91
- *
92
- * This class manages the complete lifecycle of agent execution:
93
- * 1. Creates a job record before starting
94
- * 2. Updates job status to 'running'
95
- * 3. Streams all SDK messages to job output in real-time
96
- * 4. Updates job with final status on completion
97
- *
98
- * @example
99
- * ```typescript
100
- * const executor = new JobExecutor(sdkQuery);
101
- *
102
- * const result = await executor.execute({
103
- * agent: resolvedAgent,
104
- * prompt: "Fix the bug in auth.ts",
105
- * stateDir: "/path/to/.herdctl",
106
- * triggerType: "manual",
107
- * });
108
- *
109
- * console.log(`Job ${result.jobId} completed: ${result.success}`);
110
- * ```
111
- */
112
- export class JobExecutor {
113
- private sdkQuery: SDKQueryFunction;
114
- private logger: JobExecutorLogger;
115
-
116
- /**
117
- * Create a new job executor
118
- *
119
- * @param sdkQuery - The SDK query function to use for agent execution
120
- * @param options - Optional configuration
121
- */
122
- constructor(sdkQuery: SDKQueryFunction, options: JobExecutorOptions = {}) {
123
- this.sdkQuery = sdkQuery;
124
- this.logger = options.logger ?? defaultLogger;
125
- }
126
-
127
- /**
128
- * Execute an agent and stream output to job log
129
- *
130
- * @param options - Runner options including agent config and prompt
131
- * @returns Result of the execution including job ID and status
132
- */
133
- async execute(options: RunnerOptionsWithCallbacks): Promise<RunnerResult> {
134
- const { agent, prompt, stateDir, triggerType, schedule, onMessage } =
135
- options;
136
-
137
- const jobsDir = join(stateDir, "jobs");
138
- let job: JobMetadata;
139
- let sessionId: string | undefined;
140
- let summary: string | undefined;
141
- let lastError: RunnerError | undefined;
142
- let errorDetails: RunnerErrorDetails | undefined;
143
- let messagesReceived = 0;
144
-
145
- // Determine trigger type: use 'fork' if forking, otherwise use provided or default to 'manual'
146
- const effectiveTriggerType: TriggerType = options.fork
147
- ? "fork"
148
- : ((triggerType ?? "manual") as TriggerType);
149
-
150
- // Step 1: Create job record
151
- try {
152
- job = await createJob(jobsDir, {
153
- agent: agent.name,
154
- trigger_type: effectiveTriggerType,
155
- prompt,
156
- schedule,
157
- forked_from: options.fork ? options.forkedFrom : undefined,
158
- });
159
-
160
- this.logger.info?.(`Created job ${job.id} for agent ${agent.name}`);
161
- } catch (error) {
162
- this.logger.error(`Failed to create job: ${(error as Error).message}`);
163
- throw error;
164
- }
165
-
166
- // Step 2: Update job status to 'running'
167
- try {
168
- await updateJob(jobsDir, job.id, {
169
- status: "running",
170
- });
171
- } catch (error) {
172
- this.logger.warn(
173
- `Failed to update job status to running: ${(error as Error).message}`
174
- );
175
- // Continue execution - job was created
176
- }
177
-
178
- // Step 3: Build SDK options
179
- const sdkOptions = toSDKOptions(agent, {
180
- resume: options.resume,
181
- fork: options.fork ? true : undefined,
182
- });
183
-
184
- // Step 4: Execute agent and stream output
185
- try {
186
- let messages: AsyncIterable<SDKMessage>;
187
-
188
- // Catch SDK initialization errors (e.g., missing API key)
189
- try {
190
- messages = this.sdkQuery({
191
- prompt,
192
- options: sdkOptions as Record<string, unknown>,
193
- });
194
- } catch (initError) {
195
- // Wrap initialization errors with context
196
- throw new SDKInitializationError(
197
- buildErrorMessage((initError as Error).message, {
198
- jobId: job.id,
199
- agentName: agent.name,
200
- }),
201
- {
202
- jobId: job.id,
203
- agentName: agent.name,
204
- cause: initError as Error,
205
- }
206
- );
207
- }
208
-
209
- for await (const sdkMessage of messages) {
210
- messagesReceived++;
211
-
212
- // Process the message safely (handles malformed responses)
213
- let processed;
214
- try {
215
- processed = processSDKMessage(sdkMessage);
216
- } catch (processError) {
217
- // Log but don't crash on malformed messages
218
- this.logger.warn(
219
- `Malformed SDK message received: ${(processError as Error).message}`
220
- );
221
-
222
- // Write a warning to job output
223
- try {
224
- await appendJobOutput(jobsDir, job.id, {
225
- type: "error",
226
- message: `Malformed SDK message: ${(processError as Error).message}`,
227
- code: "MALFORMED_MESSAGE",
228
- });
229
- } catch {
230
- // Ignore output write failures for malformed message warnings
231
- }
232
-
233
- // Continue processing other messages
234
- continue;
235
- }
236
-
237
- // Write to job output immediately (no buffering)
238
- try {
239
- await appendJobOutput(jobsDir, job.id, processed.output);
240
- } catch (outputError) {
241
- this.logger.warn(
242
- `Failed to write job output: ${(outputError as Error).message}`
243
- );
244
- // Continue processing - don't fail execution due to logging issues
245
- }
246
-
247
- // Extract session ID if present
248
- if (processed.sessionId) {
249
- sessionId = processed.sessionId;
250
- }
251
-
252
- // Extract summary if present
253
- const messageSummary = extractSummary(sdkMessage);
254
- if (messageSummary) {
255
- summary = messageSummary;
256
- }
257
-
258
- // Call user's onMessage callback if provided
259
- if (onMessage) {
260
- try {
261
- await onMessage(sdkMessage);
262
- } catch (callbackError) {
263
- this.logger.warn(
264
- `onMessage callback error: ${(callbackError as Error).message}`
265
- );
266
- }
267
- }
268
-
269
- // Check for terminal messages
270
- if (isTerminalMessage(sdkMessage)) {
271
- if (sdkMessage.type === "error") {
272
- const errorMessage =
273
- (sdkMessage.message as string) ?? "Agent execution failed";
274
- lastError = new SDKStreamingError(
275
- buildErrorMessage(errorMessage, {
276
- jobId: job.id,
277
- agentName: agent.name,
278
- }),
279
- {
280
- jobId: job.id,
281
- agentName: agent.name,
282
- code: sdkMessage.code as string | undefined,
283
- messagesReceived,
284
- }
285
- );
286
- }
287
- break;
288
- }
289
- }
290
- } catch (error) {
291
- // Wrap the error with context if not already a RunnerError
292
- lastError = wrapError(error, {
293
- jobId: job.id,
294
- agentName: agent.name,
295
- phase: messagesReceived === 0 ? "init" : "streaming",
296
- });
297
-
298
- // Add messages received count for streaming errors
299
- if (lastError instanceof SDKStreamingError && messagesReceived > 0) {
300
- (lastError as SDKStreamingError & { messagesReceived?: number }).messagesReceived = messagesReceived;
301
- }
302
-
303
- // Log the error with context
304
- this.logger.error(
305
- `${lastError.name}: ${lastError.message}`
306
- );
307
-
308
- // Write error to job output with full context
309
- try {
310
- await appendJobOutput(jobsDir, job.id, {
311
- type: "error",
312
- message: lastError.message,
313
- code: (lastError as SDKStreamingError).code ?? (lastError.cause as NodeJS.ErrnoException)?.code,
314
- stack: lastError.stack,
315
- });
316
- } catch (outputError) {
317
- this.logger.warn(
318
- `Failed to write error to job output: ${(outputError as Error).message}`
319
- );
320
- }
321
- }
322
-
323
- // Build error details for programmatic access
324
- if (lastError) {
325
- errorDetails = {
326
- message: lastError.message,
327
- code:
328
- (lastError as SDKStreamingError).code ??
329
- (lastError.cause as NodeJS.ErrnoException)?.code,
330
- stack: lastError.stack,
331
- };
332
-
333
- // Determine error type
334
- if (lastError instanceof SDKInitializationError) {
335
- errorDetails.type = "initialization";
336
- errorDetails.recoverable = lastError.isNetworkError();
337
- } else if (lastError instanceof SDKStreamingError) {
338
- errorDetails.type = "streaming";
339
- errorDetails.recoverable = lastError.isRecoverable();
340
- errorDetails.messagesReceived = lastError.messagesReceived;
341
- } else if (lastError instanceof MalformedResponseError) {
342
- errorDetails.type = "malformed_response";
343
- errorDetails.recoverable = false;
344
- } else {
345
- errorDetails.type = "unknown";
346
- errorDetails.recoverable = false;
347
- }
348
- }
349
-
350
- // Step 5: Update job with final status
351
- const success = !lastError;
352
- const finishedAt = new Date().toISOString();
353
-
354
- // Determine exit reason based on error classification
355
- const exitReason = success ? "success" : classifyError(lastError!);
356
-
357
- try {
358
- await updateJob(jobsDir, job.id, {
359
- status: success ? "completed" : "failed",
360
- finished_at: finishedAt,
361
- session_id: sessionId,
362
- summary,
363
- exit_reason: exitReason,
364
- output_file: getJobOutputPath(jobsDir, job.id),
365
- });
366
- } catch (error) {
367
- this.logger.warn(
368
- `Failed to update job final status: ${(error as Error).message}`
369
- );
370
- }
371
-
372
- // Step 6: Persist session info for resume capability
373
- if (sessionId) {
374
- try {
375
- const sessionsDir = join(stateDir, "sessions");
376
-
377
- // Get existing session to determine if updating or creating
378
- const existingSession = await getSessionInfo(sessionsDir, agent.name);
379
-
380
- await updateSessionInfo(sessionsDir, agent.name, {
381
- session_id: sessionId,
382
- job_count: (existingSession?.job_count ?? 0) + 1,
383
- mode: existingSession?.mode ?? "autonomous",
384
- });
385
-
386
- this.logger.info?.(
387
- `Persisted session ${sessionId} for agent ${agent.name}`
388
- );
389
- } catch (sessionError) {
390
- this.logger.warn(
391
- `Failed to persist session info: ${(sessionError as Error).message}`
392
- );
393
- // Continue - session persistence is non-fatal
394
- }
395
- }
396
-
397
- // Calculate duration
398
- const startTime = new Date(job.started_at).getTime();
399
- const endTime = new Date(finishedAt).getTime();
400
- const durationSeconds = Math.round((endTime - startTime) / 1000);
401
-
402
- return {
403
- success,
404
- jobId: job.id,
405
- sessionId,
406
- summary,
407
- error: lastError,
408
- errorDetails,
409
- durationSeconds,
410
- };
411
- }
412
- }
413
-
414
- // =============================================================================
415
- // Convenience Function
416
- // =============================================================================
417
-
418
- /**
419
- * Execute an agent with streaming output to job log
420
- *
421
- * This is a convenience function that creates a JobExecutor and runs
422
- * a single execution. For multiple executions, prefer creating a
423
- * JobExecutor instance directly.
424
- *
425
- * @param sdkQuery - The SDK query function
426
- * @param options - Runner options including agent config and prompt
427
- * @param executorOptions - Optional executor configuration
428
- * @returns Result of the execution
429
- *
430
- * @example
431
- * ```typescript
432
- * import { query } from "@anthropic-ai/claude-agent-sdk";
433
- *
434
- * const result = await executeJob(query, {
435
- * agent: resolvedAgent,
436
- * prompt: "Fix the bug",
437
- * stateDir: "/path/to/.herdctl",
438
- * });
439
- * ```
440
- */
441
- export async function executeJob(
442
- sdkQuery: SDKQueryFunction,
443
- options: RunnerOptionsWithCallbacks,
444
- executorOptions: JobExecutorOptions = {}
445
- ): Promise<RunnerResult> {
446
- const executor = new JobExecutor(sdkQuery, executorOptions);
447
- return executor.execute(options);
448
- }