@specforge/mcp 2.6.0 → 3.0.0

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 (237) hide show
  1. package/README.md +73 -0
  2. package/bin/{specforge-mcp → specforge} +0 -5
  3. package/dist/ai-provider/circuit-breaker.d.ts +63 -0
  4. package/dist/ai-provider/circuit-breaker.d.ts.map +1 -0
  5. package/dist/ai-provider/circuit-breaker.js +160 -0
  6. package/dist/ai-provider/circuit-breaker.js.map +1 -0
  7. package/dist/ai-provider/cli-version.d.ts +50 -0
  8. package/dist/ai-provider/cli-version.d.ts.map +1 -0
  9. package/dist/ai-provider/cli-version.js +141 -0
  10. package/dist/ai-provider/cli-version.js.map +1 -0
  11. package/dist/ai-provider/config-loader.d.ts +45 -0
  12. package/dist/ai-provider/config-loader.d.ts.map +1 -0
  13. package/dist/ai-provider/config-loader.js +106 -0
  14. package/dist/ai-provider/config-loader.js.map +1 -0
  15. package/dist/ai-provider/errors.d.ts +48 -0
  16. package/dist/ai-provider/errors.d.ts.map +1 -0
  17. package/dist/ai-provider/errors.js +102 -0
  18. package/dist/ai-provider/errors.js.map +1 -0
  19. package/dist/ai-provider/events.d.ts +73 -0
  20. package/dist/ai-provider/events.d.ts.map +1 -0
  21. package/dist/ai-provider/events.js +75 -0
  22. package/dist/ai-provider/events.js.map +1 -0
  23. package/dist/ai-provider/factory.d.ts +31 -0
  24. package/dist/ai-provider/factory.d.ts.map +1 -0
  25. package/dist/ai-provider/factory.js +100 -0
  26. package/dist/ai-provider/factory.js.map +1 -0
  27. package/dist/ai-provider/index.d.ts +24 -0
  28. package/dist/ai-provider/index.d.ts.map +1 -0
  29. package/dist/ai-provider/index.js +46 -0
  30. package/dist/ai-provider/index.js.map +1 -0
  31. package/dist/ai-provider/instance-coordinator.d.ts +54 -0
  32. package/dist/ai-provider/instance-coordinator.d.ts.map +1 -0
  33. package/dist/ai-provider/instance-coordinator.js +199 -0
  34. package/dist/ai-provider/instance-coordinator.js.map +1 -0
  35. package/dist/ai-provider/jsonl-parser.d.ts +43 -0
  36. package/dist/ai-provider/jsonl-parser.d.ts.map +1 -0
  37. package/dist/ai-provider/jsonl-parser.js +107 -0
  38. package/dist/ai-provider/jsonl-parser.js.map +1 -0
  39. package/dist/ai-provider/lifecycle.d.ts +50 -0
  40. package/dist/ai-provider/lifecycle.d.ts.map +1 -0
  41. package/dist/ai-provider/lifecycle.js +145 -0
  42. package/dist/ai-provider/lifecycle.js.map +1 -0
  43. package/dist/ai-provider/logger.d.ts +69 -0
  44. package/dist/ai-provider/logger.d.ts.map +1 -0
  45. package/dist/ai-provider/logger.js +161 -0
  46. package/dist/ai-provider/logger.js.map +1 -0
  47. package/dist/ai-provider/metrics.d.ts +91 -0
  48. package/dist/ai-provider/metrics.d.ts.map +1 -0
  49. package/dist/ai-provider/metrics.js +187 -0
  50. package/dist/ai-provider/metrics.js.map +1 -0
  51. package/dist/ai-provider/process-manager.d.ts +97 -0
  52. package/dist/ai-provider/process-manager.d.ts.map +1 -0
  53. package/dist/ai-provider/process-manager.js +477 -0
  54. package/dist/ai-provider/process-manager.js.map +1 -0
  55. package/dist/ai-provider/providers/claude-code.d.ts +64 -0
  56. package/dist/ai-provider/providers/claude-code.d.ts.map +1 -0
  57. package/dist/ai-provider/providers/claude-code.js +205 -0
  58. package/dist/ai-provider/providers/claude-code.js.map +1 -0
  59. package/dist/ai-provider/retry-executor.d.ts +52 -0
  60. package/dist/ai-provider/retry-executor.d.ts.map +1 -0
  61. package/dist/ai-provider/retry-executor.js +138 -0
  62. package/dist/ai-provider/retry-executor.js.map +1 -0
  63. package/dist/ai-provider/safe-args.d.ts +58 -0
  64. package/dist/ai-provider/safe-args.d.ts.map +1 -0
  65. package/dist/ai-provider/safe-args.js +176 -0
  66. package/dist/ai-provider/safe-args.js.map +1 -0
  67. package/dist/ai-provider/semaphore.d.ts +50 -0
  68. package/dist/ai-provider/semaphore.d.ts.map +1 -0
  69. package/dist/ai-provider/semaphore.js +97 -0
  70. package/dist/ai-provider/semaphore.js.map +1 -0
  71. package/dist/ai-provider/tracer.d.ts +67 -0
  72. package/dist/ai-provider/tracer.d.ts.map +1 -0
  73. package/dist/ai-provider/tracer.js +209 -0
  74. package/dist/ai-provider/tracer.js.map +1 -0
  75. package/dist/ai-provider/types.d.ts +181 -0
  76. package/dist/ai-provider/types.d.ts.map +1 -0
  77. package/dist/ai-provider/types.js +8 -0
  78. package/dist/ai-provider/types.js.map +1 -0
  79. package/dist/autopilot/agents/agent-runner.d.ts +109 -0
  80. package/dist/autopilot/agents/agent-runner.d.ts.map +1 -0
  81. package/dist/autopilot/agents/agent-runner.js +731 -0
  82. package/dist/autopilot/agents/agent-runner.js.map +1 -0
  83. package/dist/autopilot/agents/agent-selector.d.ts +59 -0
  84. package/dist/autopilot/agents/agent-selector.d.ts.map +1 -0
  85. package/dist/autopilot/agents/agent-selector.js +234 -0
  86. package/dist/autopilot/agents/agent-selector.js.map +1 -0
  87. package/dist/autopilot/agents/model-selector.d.ts +49 -0
  88. package/dist/autopilot/agents/model-selector.d.ts.map +1 -0
  89. package/dist/autopilot/agents/model-selector.js +62 -0
  90. package/dist/autopilot/agents/model-selector.js.map +1 -0
  91. package/dist/autopilot/agents/profiles/builtin.d.ts +55 -0
  92. package/dist/autopilot/agents/profiles/builtin.d.ts.map +1 -0
  93. package/dist/autopilot/agents/profiles/builtin.js +323 -0
  94. package/dist/autopilot/agents/profiles/builtin.js.map +1 -0
  95. package/dist/autopilot/agents/profiles/types.d.ts +98 -0
  96. package/dist/autopilot/agents/profiles/types.d.ts.map +1 -0
  97. package/dist/autopilot/agents/profiles/types.js +17 -0
  98. package/dist/autopilot/agents/profiles/types.js.map +1 -0
  99. package/dist/autopilot/api/autopilot-api-client.d.ts +217 -0
  100. package/dist/autopilot/api/autopilot-api-client.d.ts.map +1 -0
  101. package/dist/autopilot/api/autopilot-api-client.js +402 -0
  102. package/dist/autopilot/api/autopilot-api-client.js.map +1 -0
  103. package/dist/autopilot/cli/abort.d.ts +20 -0
  104. package/dist/autopilot/cli/abort.d.ts.map +1 -0
  105. package/dist/autopilot/cli/abort.js +201 -0
  106. package/dist/autopilot/cli/abort.js.map +1 -0
  107. package/dist/autopilot/cli/display.d.ts +63 -0
  108. package/dist/autopilot/cli/display.d.ts.map +1 -0
  109. package/dist/autopilot/cli/display.js +260 -0
  110. package/dist/autopilot/cli/display.js.map +1 -0
  111. package/dist/autopilot/cli/index.d.ts +24 -0
  112. package/dist/autopilot/cli/index.d.ts.map +1 -0
  113. package/dist/autopilot/cli/index.js +79 -0
  114. package/dist/autopilot/cli/index.js.map +1 -0
  115. package/dist/autopilot/cli/pause.d.ts +18 -0
  116. package/dist/autopilot/cli/pause.d.ts.map +1 -0
  117. package/dist/autopilot/cli/pause.js +110 -0
  118. package/dist/autopilot/cli/pause.js.map +1 -0
  119. package/dist/autopilot/cli/resume.d.ts +22 -0
  120. package/dist/autopilot/cli/resume.d.ts.map +1 -0
  121. package/dist/autopilot/cli/resume.js +172 -0
  122. package/dist/autopilot/cli/resume.js.map +1 -0
  123. package/dist/autopilot/cli/run.d.ts +25 -0
  124. package/dist/autopilot/cli/run.d.ts.map +1 -0
  125. package/dist/autopilot/cli/run.js +220 -0
  126. package/dist/autopilot/cli/run.js.map +1 -0
  127. package/dist/autopilot/cli/status.d.ts +20 -0
  128. package/dist/autopilot/cli/status.d.ts.map +1 -0
  129. package/dist/autopilot/cli/status.js +217 -0
  130. package/dist/autopilot/cli/status.js.map +1 -0
  131. package/dist/autopilot/config.d.ts +45 -0
  132. package/dist/autopilot/config.d.ts.map +1 -0
  133. package/dist/autopilot/config.js +269 -0
  134. package/dist/autopilot/config.js.map +1 -0
  135. package/dist/autopilot/core/dependency-resolver.d.ts +108 -0
  136. package/dist/autopilot/core/dependency-resolver.d.ts.map +1 -0
  137. package/dist/autopilot/core/dependency-resolver.js +394 -0
  138. package/dist/autopilot/core/dependency-resolver.js.map +1 -0
  139. package/dist/autopilot/core/dispatcher.d.ts +215 -0
  140. package/dist/autopilot/core/dispatcher.d.ts.map +1 -0
  141. package/dist/autopilot/core/dispatcher.js +594 -0
  142. package/dist/autopilot/core/dispatcher.js.map +1 -0
  143. package/dist/autopilot/core/failure-handler.d.ts +145 -0
  144. package/dist/autopilot/core/failure-handler.d.ts.map +1 -0
  145. package/dist/autopilot/core/failure-handler.js +308 -0
  146. package/dist/autopilot/core/failure-handler.js.map +1 -0
  147. package/dist/autopilot/core/rate-limit-handler.d.ts +108 -0
  148. package/dist/autopilot/core/rate-limit-handler.d.ts.map +1 -0
  149. package/dist/autopilot/core/rate-limit-handler.js +195 -0
  150. package/dist/autopilot/core/rate-limit-handler.js.map +1 -0
  151. package/dist/autopilot/core/state-manager.d.ts +160 -0
  152. package/dist/autopilot/core/state-manager.d.ts.map +1 -0
  153. package/dist/autopilot/core/state-manager.js +393 -0
  154. package/dist/autopilot/core/state-manager.js.map +1 -0
  155. package/dist/autopilot/core/timeout-manager.d.ts +95 -0
  156. package/dist/autopilot/core/timeout-manager.d.ts.map +1 -0
  157. package/dist/autopilot/core/timeout-manager.js +188 -0
  158. package/dist/autopilot/core/timeout-manager.js.map +1 -0
  159. package/dist/autopilot/git/branch-manager.d.ts +117 -0
  160. package/dist/autopilot/git/branch-manager.d.ts.map +1 -0
  161. package/dist/autopilot/git/branch-manager.js +238 -0
  162. package/dist/autopilot/git/branch-manager.js.map +1 -0
  163. package/dist/autopilot/git/index.d.ts +9 -0
  164. package/dist/autopilot/git/index.d.ts.map +1 -0
  165. package/dist/autopilot/git/index.js +9 -0
  166. package/dist/autopilot/git/index.js.map +1 -0
  167. package/dist/autopilot/git/merge-manager.d.ts +118 -0
  168. package/dist/autopilot/git/merge-manager.d.ts.map +1 -0
  169. package/dist/autopilot/git/merge-manager.js +304 -0
  170. package/dist/autopilot/git/merge-manager.js.map +1 -0
  171. package/dist/autopilot/git/worktree-manager.d.ts +128 -0
  172. package/dist/autopilot/git/worktree-manager.d.ts.map +1 -0
  173. package/dist/autopilot/git/worktree-manager.js +298 -0
  174. package/dist/autopilot/git/worktree-manager.js.map +1 -0
  175. package/dist/autopilot/index.d.ts +30 -0
  176. package/dist/autopilot/index.d.ts.map +1 -0
  177. package/dist/autopilot/index.js +55 -0
  178. package/dist/autopilot/index.js.map +1 -0
  179. package/dist/autopilot/sync/index.d.ts +7 -0
  180. package/dist/autopilot/sync/index.d.ts.map +1 -0
  181. package/dist/autopilot/sync/index.js +7 -0
  182. package/dist/autopilot/sync/index.js.map +1 -0
  183. package/dist/autopilot/sync/sync-manager.d.ts +168 -0
  184. package/dist/autopilot/sync/sync-manager.d.ts.map +1 -0
  185. package/dist/autopilot/sync/sync-manager.js +303 -0
  186. package/dist/autopilot/sync/sync-manager.js.map +1 -0
  187. package/dist/autopilot/types.d.ts +454 -0
  188. package/dist/autopilot/types.d.ts.map +1 -0
  189. package/dist/autopilot/types.js +26 -0
  190. package/dist/autopilot/types.js.map +1 -0
  191. package/dist/autopilot/utils/audit-logger.d.ts +176 -0
  192. package/dist/autopilot/utils/audit-logger.d.ts.map +1 -0
  193. package/dist/autopilot/utils/audit-logger.js +308 -0
  194. package/dist/autopilot/utils/audit-logger.js.map +1 -0
  195. package/dist/autopilot/utils/cost-tracker.d.ts +162 -0
  196. package/dist/autopilot/utils/cost-tracker.d.ts.map +1 -0
  197. package/dist/autopilot/utils/cost-tracker.js +269 -0
  198. package/dist/autopilot/utils/cost-tracker.js.map +1 -0
  199. package/dist/autopilot/utils/index.d.ts +9 -0
  200. package/dist/autopilot/utils/index.d.ts.map +1 -0
  201. package/dist/autopilot/utils/index.js +9 -0
  202. package/dist/autopilot/utils/index.js.map +1 -0
  203. package/dist/autopilot/utils/progress-reporter.d.ts +132 -0
  204. package/dist/autopilot/utils/progress-reporter.d.ts.map +1 -0
  205. package/dist/autopilot/utils/progress-reporter.js +290 -0
  206. package/dist/autopilot/utils/progress-reporter.js.map +1 -0
  207. package/dist/autopilot/worker/worker-pool.d.ts +179 -0
  208. package/dist/autopilot/worker/worker-pool.d.ts.map +1 -0
  209. package/dist/autopilot/worker/worker-pool.js +331 -0
  210. package/dist/autopilot/worker/worker-pool.js.map +1 -0
  211. package/dist/autopilot/worker/worker-session.d.ts +171 -0
  212. package/dist/autopilot/worker/worker-session.d.ts.map +1 -0
  213. package/dist/autopilot/worker/worker-session.js +295 -0
  214. package/dist/autopilot/worker/worker-session.js.map +1 -0
  215. package/dist/cli/index.d.ts +1 -1
  216. package/dist/cli/index.d.ts.map +1 -1
  217. package/dist/cli/index.js +4 -1
  218. package/dist/cli/index.js.map +1 -1
  219. package/dist/index.js +0 -1
  220. package/dist/index.js.map +1 -1
  221. package/dist/tools/core/epic.js +1 -1
  222. package/dist/tools/core/epic.js.map +1 -1
  223. package/dist/tools/core/lookup.d.ts.map +1 -1
  224. package/dist/tools/core/lookup.js +3 -2
  225. package/dist/tools/core/lookup.js.map +1 -1
  226. package/dist/tools/core/specification.js +1 -1
  227. package/dist/tools/core/specification.js.map +1 -1
  228. package/dist/tools/core/ticket.d.ts.map +1 -1
  229. package/dist/tools/core/ticket.js +4 -6
  230. package/dist/tools/core/ticket.js.map +1 -1
  231. package/dist/tools/index.d.ts.map +1 -1
  232. package/dist/tools/index.js +60 -0
  233. package/dist/tools/index.js.map +1 -1
  234. package/dist/validation/index.d.ts.map +1 -1
  235. package/dist/validation/index.js +4 -1
  236. package/dist/validation/index.js.map +1 -1
  237. package/package.json +8 -4
@@ -0,0 +1,393 @@
1
+ /**
2
+ * Autopilot State Manager
3
+ *
4
+ * Manages autopilot state, persistence, and state change events.
5
+ */
6
+ import { EventEmitter } from 'events';
7
+ import * as fs from 'fs';
8
+ import * as path from 'path';
9
+ import { randomUUID } from 'crypto';
10
+ /**
11
+ * State file name
12
+ */
13
+ export const STATE_FILE_NAME = 'autopilot-state.json';
14
+ const STATE_FILE = `.specforge/${STATE_FILE_NAME}`;
15
+ /**
16
+ * Typed event emitter for autopilot events
17
+ */
18
+ class AutopilotEventEmitter extends EventEmitter {
19
+ emit(event, payload) {
20
+ return super.emit(event, payload);
21
+ }
22
+ on(event, listener) {
23
+ return super.on(event, listener);
24
+ }
25
+ once(event, listener) {
26
+ return super.once(event, listener);
27
+ }
28
+ }
29
+ /**
30
+ * Autopilot State Manager
31
+ */
32
+ export class StateManager {
33
+ state;
34
+ events;
35
+ persistPath;
36
+ persistEnabled;
37
+ constructor(options) {
38
+ this.events = new AutopilotEventEmitter();
39
+ this.persistPath = path.join(options.workingDirectory || process.cwd(), STATE_FILE);
40
+ this.persistEnabled = options.persistState ?? true;
41
+ // Initialize state
42
+ this.state = {
43
+ runId: randomUUID(),
44
+ status: 'idle',
45
+ specificationId: options.specificationId,
46
+ projectId: options.projectId,
47
+ startedAt: new Date(),
48
+ workers: [],
49
+ completedTickets: [],
50
+ failedTickets: [],
51
+ skippedTickets: [],
52
+ totalCostUsd: 0,
53
+ totalTokens: { inputTokens: 0, outputTokens: 0 },
54
+ };
55
+ }
56
+ /**
57
+ * Get current state
58
+ */
59
+ getState() {
60
+ return { ...this.state };
61
+ }
62
+ /**
63
+ * Get run ID
64
+ */
65
+ getRunId() {
66
+ return this.state.runId;
67
+ }
68
+ /**
69
+ * Get event emitter for subscriptions
70
+ */
71
+ getEvents() {
72
+ return this.events;
73
+ }
74
+ /**
75
+ * Update status
76
+ */
77
+ setStatus(status, options) {
78
+ const previousStatus = this.state.status;
79
+ this.state.status = status;
80
+ if (status === 'paused' && options?.reason) {
81
+ this.state.pauseReason = options.reason;
82
+ }
83
+ if (status === 'aborted' && options?.reason) {
84
+ this.state.errorMessage = options.reason;
85
+ }
86
+ if (status === 'completed' || status === 'aborted') {
87
+ this.state.endedAt = new Date();
88
+ }
89
+ this.emitStateChange();
90
+ this.persist();
91
+ // Emit specific events
92
+ if (status === 'paused') {
93
+ this.events.emit('paused', { reason: options?.reason || 'Unknown' });
94
+ }
95
+ else if (status === 'aborted') {
96
+ this.events.emit('aborted', { reason: options?.reason || 'Unknown' });
97
+ }
98
+ else if (status === 'running' && previousStatus === 'paused') {
99
+ this.events.emit('resumed', undefined);
100
+ }
101
+ }
102
+ /**
103
+ * Set current epic
104
+ */
105
+ setCurrentEpic(epicId) {
106
+ this.state.currentEpicId = epicId;
107
+ this.events.emit('epic:started', { epicId });
108
+ this.emitStateChange();
109
+ this.persist();
110
+ }
111
+ /**
112
+ * Clear current epic
113
+ */
114
+ clearCurrentEpic(summary) {
115
+ if (this.state.currentEpicId) {
116
+ this.events.emit('epic:completed', {
117
+ epicId: this.state.currentEpicId,
118
+ completedTickets: summary.completedTickets,
119
+ failedTickets: summary.failedTickets,
120
+ });
121
+ }
122
+ this.state.currentEpicId = undefined;
123
+ this.emitStateChange();
124
+ this.persist();
125
+ }
126
+ /**
127
+ * Initialize workers
128
+ */
129
+ initializeWorkers(workers) {
130
+ this.state.workers = workers;
131
+ this.emitStateChange();
132
+ this.persist();
133
+ }
134
+ /**
135
+ * Update worker state
136
+ */
137
+ updateWorker(workerId, updates) {
138
+ const worker = this.state.workers.find(w => w.workerId === workerId);
139
+ if (worker) {
140
+ Object.assign(worker, updates);
141
+ this.emitStateChange();
142
+ this.persist();
143
+ }
144
+ }
145
+ /**
146
+ * Get worker by ID
147
+ */
148
+ getWorker(workerId) {
149
+ return this.state.workers.find(w => w.workerId === workerId);
150
+ }
151
+ /**
152
+ * Record ticket started
153
+ */
154
+ ticketStarted(ticketId, workerId) {
155
+ const worker = this.getWorker(workerId);
156
+ if (worker) {
157
+ worker.currentTicketId = ticketId;
158
+ worker.ticketStartedAt = new Date();
159
+ worker.status = 'implementing';
160
+ worker.retryCount = 0;
161
+ }
162
+ this.events.emit('ticket:started', { ticketId, workerId });
163
+ this.events.emit('worker:started', { workerId, ticketId });
164
+ this.emitStateChange();
165
+ this.persist();
166
+ }
167
+ /**
168
+ * Record ticket completed
169
+ */
170
+ ticketCompleted(ticketId, workerId, result) {
171
+ this.state.completedTickets.push(ticketId);
172
+ this.state.totalCostUsd += result.costUsd;
173
+ this.state.totalTokens.inputTokens += result.tokens.inputTokens;
174
+ this.state.totalTokens.outputTokens += result.tokens.outputTokens;
175
+ const worker = this.getWorker(workerId);
176
+ if (worker) {
177
+ worker.completedCount++;
178
+ worker.costUsd += result.costUsd;
179
+ worker.currentTicketId = undefined;
180
+ worker.ticketStartedAt = undefined;
181
+ worker.status = 'idle';
182
+ }
183
+ this.events.emit('ticket:completed', {
184
+ ticketId,
185
+ result: 'success',
186
+ costUsd: result.costUsd,
187
+ });
188
+ this.events.emit('worker:completed', {
189
+ workerId,
190
+ ticketId,
191
+ result: 'success',
192
+ });
193
+ this.emitStateChange();
194
+ this.persist();
195
+ }
196
+ /**
197
+ * Record ticket failed
198
+ */
199
+ ticketFailed(ticketId, workerId, error, retryable) {
200
+ const existingFailure = this.state.failedTickets.find(f => f.ticketId === ticketId);
201
+ if (existingFailure) {
202
+ existingFailure.attempts++;
203
+ existingFailure.retryCount++;
204
+ existingFailure.error = error;
205
+ existingFailure.lastAttemptAt = new Date();
206
+ }
207
+ else {
208
+ this.state.failedTickets.push({
209
+ ticketId,
210
+ error,
211
+ attempts: 1,
212
+ retryCount: 0,
213
+ lastAttemptAt: new Date(),
214
+ });
215
+ }
216
+ const worker = this.getWorker(workerId);
217
+ if (worker) {
218
+ worker.failedCount++;
219
+ worker.lastError = error;
220
+ worker.currentTicketId = undefined;
221
+ worker.ticketStartedAt = undefined;
222
+ worker.status = 'idle';
223
+ }
224
+ this.events.emit('ticket:failed', { ticketId, error, retryable });
225
+ this.events.emit('worker:failed', { workerId, ticketId, error });
226
+ this.emitStateChange();
227
+ this.persist();
228
+ }
229
+ /**
230
+ * Record ticket skipped
231
+ */
232
+ ticketSkipped(ticketId, reason) {
233
+ if (!this.state.skippedTickets.includes(ticketId)) {
234
+ this.state.skippedTickets.push(ticketId);
235
+ }
236
+ this.emitStateChange();
237
+ this.persist();
238
+ }
239
+ /**
240
+ * Set rate limit info
241
+ */
242
+ setRateLimited(info) {
243
+ this.state.rateLimitInfo = info;
244
+ this.state.status = 'rate_limited';
245
+ this.events.emit('rate_limit:hit', { message: info.message });
246
+ this.emitStateChange();
247
+ this.persist();
248
+ }
249
+ /**
250
+ * Clear rate limit
251
+ */
252
+ clearRateLimit() {
253
+ this.state.rateLimitInfo = undefined;
254
+ if (this.state.status === 'rate_limited') {
255
+ this.state.status = 'running';
256
+ }
257
+ this.events.emit('rate_limit:cleared', undefined);
258
+ this.emitStateChange();
259
+ this.persist();
260
+ }
261
+ /**
262
+ * Add cost
263
+ */
264
+ addCost(costUsd, tokens) {
265
+ this.state.totalCostUsd += costUsd;
266
+ this.state.totalTokens.inputTokens += tokens.inputTokens;
267
+ this.state.totalTokens.outputTokens += tokens.outputTokens;
268
+ this.emitStateChange();
269
+ this.persist();
270
+ }
271
+ /**
272
+ * Get completed ticket count
273
+ */
274
+ getCompletedCount() {
275
+ return this.state.completedTickets.length;
276
+ }
277
+ /**
278
+ * Get failed ticket count
279
+ */
280
+ getFailedCount() {
281
+ return this.state.failedTickets.length;
282
+ }
283
+ /**
284
+ * Check if ticket was completed
285
+ */
286
+ isTicketCompleted(ticketId) {
287
+ return this.state.completedTickets.includes(ticketId);
288
+ }
289
+ /**
290
+ * Check if ticket was failed
291
+ */
292
+ isTicketFailed(ticketId) {
293
+ return this.state.failedTickets.some(f => f.ticketId === ticketId);
294
+ }
295
+ /**
296
+ * Get failure info for ticket
297
+ */
298
+ getFailureInfo(ticketId) {
299
+ return this.state.failedTickets.find(f => f.ticketId === ticketId);
300
+ }
301
+ /**
302
+ * Emit state change event
303
+ */
304
+ emitStateChange() {
305
+ this.events.emit('state:changed', this.getState());
306
+ }
307
+ /**
308
+ * Persist state to disk
309
+ */
310
+ persist() {
311
+ if (!this.persistEnabled)
312
+ return;
313
+ try {
314
+ const dir = path.dirname(this.persistPath);
315
+ if (!fs.existsSync(dir)) {
316
+ fs.mkdirSync(dir, { recursive: true });
317
+ }
318
+ fs.writeFileSync(this.persistPath, JSON.stringify(this.state, null, 2), 'utf-8');
319
+ }
320
+ catch (error) {
321
+ console.warn('Failed to persist autopilot state:', error);
322
+ }
323
+ }
324
+ /**
325
+ * Load state from disk
326
+ */
327
+ static load(workingDirectory = process.cwd()) {
328
+ const statePath = path.join(workingDirectory, STATE_FILE);
329
+ try {
330
+ if (!fs.existsSync(statePath)) {
331
+ return null;
332
+ }
333
+ const content = fs.readFileSync(statePath, 'utf-8');
334
+ const state = JSON.parse(content);
335
+ // Convert date strings back to Date objects
336
+ state.startedAt = new Date(state.startedAt);
337
+ if (state.endedAt) {
338
+ state.endedAt = new Date(state.endedAt);
339
+ }
340
+ if (state.rateLimitInfo?.hitAt) {
341
+ state.rateLimitInfo.hitAt = new Date(state.rateLimitInfo.hitAt);
342
+ }
343
+ for (const failure of state.failedTickets) {
344
+ failure.lastAttemptAt = new Date(failure.lastAttemptAt);
345
+ }
346
+ for (const worker of state.workers) {
347
+ if (worker.ticketStartedAt) {
348
+ worker.ticketStartedAt = new Date(worker.ticketStartedAt);
349
+ }
350
+ }
351
+ return state;
352
+ }
353
+ catch (error) {
354
+ console.warn('Failed to load autopilot state:', error);
355
+ return null;
356
+ }
357
+ }
358
+ /**
359
+ * Clear persisted state
360
+ */
361
+ static clear(workingDirectory = process.cwd()) {
362
+ const statePath = path.join(workingDirectory, STATE_FILE);
363
+ try {
364
+ if (fs.existsSync(statePath)) {
365
+ fs.unlinkSync(statePath);
366
+ }
367
+ }
368
+ catch (error) {
369
+ console.warn('Failed to clear autopilot state:', error);
370
+ }
371
+ }
372
+ /**
373
+ * Resume from existing state
374
+ */
375
+ static resume(state, options) {
376
+ const manager = new StateManager({
377
+ specificationId: state.specificationId,
378
+ projectId: state.projectId,
379
+ workingDirectory: options?.workingDirectory,
380
+ persistState: options?.persistState,
381
+ });
382
+ // Restore state (preserving run ID)
383
+ manager.state = { ...state };
384
+ return manager;
385
+ }
386
+ }
387
+ /**
388
+ * Create a state manager
389
+ */
390
+ export function createStateManager(options) {
391
+ return new StateManager(options);
392
+ }
393
+ //# sourceMappingURL=state-manager.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"state-manager.js","sourceRoot":"","sources":["../../../src/autopilot/core/state-manager.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,95 @@
1
+ /**
2
+ * Timeout Manager
3
+ *
4
+ * Tracks per-ticket timeouts and emits warnings.
5
+ */
6
+ import { EventEmitter } from 'events';
7
+ import type { AutopilotConfig } from '../types.js';
8
+ /**
9
+ * Timeout Manager Events
10
+ */
11
+ export interface TimeoutManagerEvents {
12
+ 'timeout:warning': {
13
+ ticketId: string;
14
+ percentUsed: number;
15
+ };
16
+ 'timeout:exceeded': {
17
+ ticketId: string;
18
+ };
19
+ }
20
+ /**
21
+ * Timeout Manager
22
+ */
23
+ export declare class TimeoutManager extends EventEmitter {
24
+ private timeouts;
25
+ private checkInterval?;
26
+ private config;
27
+ constructor(config: AutopilotConfig);
28
+ /**
29
+ * Start the timeout manager
30
+ */
31
+ start(): void;
32
+ /**
33
+ * Stop the timeout manager
34
+ */
35
+ stop(): void;
36
+ /**
37
+ * Start tracking a ticket
38
+ */
39
+ startTicket(ticketId: string, workerId: number, customTimeoutMs?: number): void;
40
+ /**
41
+ * Cancel timeout tracking for a ticket
42
+ */
43
+ cancelTicket(ticketId: string): void;
44
+ /**
45
+ * Check all timeouts and emit warnings
46
+ */
47
+ private checkTimeouts;
48
+ /**
49
+ * Handle timeout exceeded
50
+ */
51
+ private handleTimeout;
52
+ /**
53
+ * Get remaining time for a ticket
54
+ */
55
+ getRemainingTime(ticketId: string): number | null;
56
+ /**
57
+ * Get elapsed time for a ticket
58
+ */
59
+ getElapsedTime(ticketId: string): number | null;
60
+ /**
61
+ * Get percent used for a ticket
62
+ */
63
+ getPercentUsed(ticketId: string): number | null;
64
+ /**
65
+ * Check if ticket is being tracked
66
+ */
67
+ isTracking(ticketId: string): boolean;
68
+ /**
69
+ * Get all active timeouts
70
+ */
71
+ getActiveTimeouts(): Array<{
72
+ ticketId: string;
73
+ workerId: number;
74
+ elapsedMs: number;
75
+ remainingMs: number;
76
+ percentUsed: number;
77
+ }>;
78
+ /**
79
+ * Update default timeout
80
+ */
81
+ setDefaultTimeout(timeoutMs: number): void;
82
+ /**
83
+ * Update warning threshold
84
+ */
85
+ setWarningThreshold(percent: number): void;
86
+ /**
87
+ * Get count of active timeouts
88
+ */
89
+ getActiveCount(): number;
90
+ }
91
+ /**
92
+ * Create a timeout manager
93
+ */
94
+ export declare function createTimeoutManager(config: AutopilotConfig): TimeoutManager;
95
+ //# sourceMappingURL=timeout-manager.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"timeout-manager.d.ts","sourceRoot":"","sources":["../../../src/autopilot/core/timeout-manager.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AAEH,OAAO,EAAE,YAAY,EAAE,MAAM,QAAQ,CAAC;AACtC,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,aAAa,CAAC;AAcnD;;GAEG;AACH,MAAM,WAAW,oBAAoB;IACnC,iBAAiB,EAAE;QAAE,QAAQ,EAAE,MAAM,CAAC;QAAC,WAAW,EAAE,MAAM,CAAA;KAAE,CAAC;IAC7D,kBAAkB,EAAE;QAAE,QAAQ,EAAE,MAAM,CAAA;KAAE,CAAC;CAC1C;AAED;;GAEG;AACH,qBAAa,cAAe,SAAQ,YAAY;IAC9C,OAAO,CAAC,QAAQ,CAAyC;IACzD,OAAO,CAAC,aAAa,CAAC,CAAiB;IACvC,OAAO,CAAC,MAAM,CAGZ;gBAEU,MAAM,EAAE,eAAe;IAQnC;;OAEG;IACH,KAAK,IAAI,IAAI;IASb;;OAEG;IACH,IAAI,IAAI,IAAI;IAeZ;;OAEG;IACH,WAAW,CACT,QAAQ,EAAE,MAAM,EAChB,QAAQ,EAAE,MAAM,EAChB,eAAe,CAAC,EAAE,MAAM,GACvB,IAAI;IAuBP;;OAEG;IACH,YAAY,CAAC,QAAQ,EAAE,MAAM,GAAG,IAAI;IAUpC;;OAEG;IACH,OAAO,CAAC,aAAa;IAqBrB;;OAEG;IACH,OAAO,CAAC,aAAa;IAQrB;;OAEG;IACH,gBAAgB,CAAC,QAAQ,EAAE,MAAM,GAAG,MAAM,GAAG,IAAI;IAQjD;;OAEG;IACH,cAAc,CAAC,QAAQ,EAAE,MAAM,GAAG,MAAM,GAAG,IAAI;IAO/C;;OAEG;IACH,cAAc,CAAC,QAAQ,EAAE,MAAM,GAAG,MAAM,GAAG,IAAI;IAQ/C;;OAEG;IACH,UAAU,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO;IAIrC;;OAEG;IACH,iBAAiB,IAAI,KAAK,CAAC;QACzB,QAAQ,EAAE,MAAM,CAAC;QACjB,QAAQ,EAAE,MAAM,CAAC;QACjB,SAAS,EAAE,MAAM,CAAC;QAClB,WAAW,EAAE,MAAM,CAAC;QACpB,WAAW,EAAE,MAAM,CAAC;KACrB,CAAC;IAwBF;;OAEG;IACH,iBAAiB,CAAC,SAAS,EAAE,MAAM,GAAG,IAAI;IAI1C;;OAEG;IACH,mBAAmB,CAAC,OAAO,EAAE,MAAM,GAAG,IAAI;IAI1C;;OAEG;IACH,cAAc,IAAI,MAAM;CAGzB;AAGD;;GAEG;AACH,wBAAgB,oBAAoB,CAAC,MAAM,EAAE,eAAe,GAAG,cAAc,CAE5E"}
@@ -0,0 +1,188 @@
1
+ /**
2
+ * Timeout Manager
3
+ *
4
+ * Tracks per-ticket timeouts and emits warnings.
5
+ */
6
+ import { EventEmitter } from 'events';
7
+ /**
8
+ * Timeout Manager
9
+ */
10
+ export class TimeoutManager extends EventEmitter {
11
+ timeouts = new Map();
12
+ checkInterval;
13
+ config;
14
+ constructor(config) {
15
+ super();
16
+ this.config = {
17
+ ticketTimeoutMs: config.ticketTimeoutMs,
18
+ warnAtPercent: config.warnAtTimeoutPercent,
19
+ };
20
+ }
21
+ /**
22
+ * Start the timeout manager
23
+ */
24
+ start() {
25
+ if (this.checkInterval)
26
+ return;
27
+ // Check timeouts every 5 seconds
28
+ this.checkInterval = setInterval(() => {
29
+ this.checkTimeouts();
30
+ }, 5000);
31
+ }
32
+ /**
33
+ * Stop the timeout manager
34
+ */
35
+ stop() {
36
+ if (this.checkInterval) {
37
+ clearInterval(this.checkInterval);
38
+ this.checkInterval = undefined;
39
+ }
40
+ // Clear all timeouts
41
+ for (const timeout of this.timeouts.values()) {
42
+ if (timeout.timeoutHandle) {
43
+ clearTimeout(timeout.timeoutHandle);
44
+ }
45
+ }
46
+ this.timeouts.clear();
47
+ }
48
+ /**
49
+ * Start tracking a ticket
50
+ */
51
+ startTicket(ticketId, workerId, customTimeoutMs) {
52
+ // Clear existing timeout if any
53
+ this.cancelTicket(ticketId);
54
+ const timeoutMs = customTimeoutMs || this.config.ticketTimeoutMs;
55
+ const now = Date.now();
56
+ const timeoutInfo = {
57
+ ticketId,
58
+ workerId,
59
+ startedAt: now,
60
+ timeoutMs,
61
+ warningEmitted: false,
62
+ };
63
+ // Set timeout for exceeded
64
+ timeoutInfo.timeoutHandle = setTimeout(() => {
65
+ this.handleTimeout(ticketId);
66
+ }, timeoutMs);
67
+ this.timeouts.set(ticketId, timeoutInfo);
68
+ }
69
+ /**
70
+ * Cancel timeout tracking for a ticket
71
+ */
72
+ cancelTicket(ticketId) {
73
+ const timeout = this.timeouts.get(ticketId);
74
+ if (timeout) {
75
+ if (timeout.timeoutHandle) {
76
+ clearTimeout(timeout.timeoutHandle);
77
+ }
78
+ this.timeouts.delete(ticketId);
79
+ }
80
+ }
81
+ /**
82
+ * Check all timeouts and emit warnings
83
+ */
84
+ checkTimeouts() {
85
+ const now = Date.now();
86
+ for (const timeout of this.timeouts.values()) {
87
+ const elapsed = now - timeout.startedAt;
88
+ const percentUsed = (elapsed / timeout.timeoutMs) * 100;
89
+ // Emit warning at threshold
90
+ if (!timeout.warningEmitted &&
91
+ percentUsed >= this.config.warnAtPercent) {
92
+ timeout.warningEmitted = true;
93
+ this.emit('timeout:warning', {
94
+ ticketId: timeout.ticketId,
95
+ percentUsed: Math.round(percentUsed),
96
+ });
97
+ }
98
+ }
99
+ }
100
+ /**
101
+ * Handle timeout exceeded
102
+ */
103
+ handleTimeout(ticketId) {
104
+ const timeout = this.timeouts.get(ticketId);
105
+ if (timeout) {
106
+ this.timeouts.delete(ticketId);
107
+ this.emit('timeout:exceeded', { ticketId });
108
+ }
109
+ }
110
+ /**
111
+ * Get remaining time for a ticket
112
+ */
113
+ getRemainingTime(ticketId) {
114
+ const timeout = this.timeouts.get(ticketId);
115
+ if (!timeout)
116
+ return null;
117
+ const elapsed = Date.now() - timeout.startedAt;
118
+ return Math.max(0, timeout.timeoutMs - elapsed);
119
+ }
120
+ /**
121
+ * Get elapsed time for a ticket
122
+ */
123
+ getElapsedTime(ticketId) {
124
+ const timeout = this.timeouts.get(ticketId);
125
+ if (!timeout)
126
+ return null;
127
+ return Date.now() - timeout.startedAt;
128
+ }
129
+ /**
130
+ * Get percent used for a ticket
131
+ */
132
+ getPercentUsed(ticketId) {
133
+ const timeout = this.timeouts.get(ticketId);
134
+ if (!timeout)
135
+ return null;
136
+ const elapsed = Date.now() - timeout.startedAt;
137
+ return Math.round((elapsed / timeout.timeoutMs) * 100);
138
+ }
139
+ /**
140
+ * Check if ticket is being tracked
141
+ */
142
+ isTracking(ticketId) {
143
+ return this.timeouts.has(ticketId);
144
+ }
145
+ /**
146
+ * Get all active timeouts
147
+ */
148
+ getActiveTimeouts() {
149
+ const now = Date.now();
150
+ const result = [];
151
+ for (const timeout of this.timeouts.values()) {
152
+ const elapsed = now - timeout.startedAt;
153
+ result.push({
154
+ ticketId: timeout.ticketId,
155
+ workerId: timeout.workerId,
156
+ elapsedMs: elapsed,
157
+ remainingMs: Math.max(0, timeout.timeoutMs - elapsed),
158
+ percentUsed: Math.round((elapsed / timeout.timeoutMs) * 100),
159
+ });
160
+ }
161
+ return result;
162
+ }
163
+ /**
164
+ * Update default timeout
165
+ */
166
+ setDefaultTimeout(timeoutMs) {
167
+ this.config.ticketTimeoutMs = timeoutMs;
168
+ }
169
+ /**
170
+ * Update warning threshold
171
+ */
172
+ setWarningThreshold(percent) {
173
+ this.config.warnAtPercent = percent;
174
+ }
175
+ /**
176
+ * Get count of active timeouts
177
+ */
178
+ getActiveCount() {
179
+ return this.timeouts.size;
180
+ }
181
+ }
182
+ /**
183
+ * Create a timeout manager
184
+ */
185
+ export function createTimeoutManager(config) {
186
+ return new TimeoutManager(config);
187
+ }
188
+ //# sourceMappingURL=timeout-manager.js.map