@clinebot/core 0.0.35 → 0.0.37

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 (441) hide show
  1. package/README.md +1 -2
  2. package/dist/ClineCore.d.ts +362 -39
  3. package/dist/ClineCore.d.ts.map +1 -1
  4. package/dist/account/cline-account-service.d.ts.map +1 -1
  5. package/dist/account/index.d.ts +1 -1
  6. package/dist/account/index.d.ts.map +1 -1
  7. package/dist/account/rpc.d.ts +6 -6
  8. package/dist/account/rpc.d.ts.map +1 -1
  9. package/dist/cron/cron-event-ingress.d.ts +38 -0
  10. package/dist/cron/cron-event-ingress.d.ts.map +1 -0
  11. package/dist/cron/cron-materializer.d.ts +36 -0
  12. package/dist/cron/cron-materializer.d.ts.map +1 -0
  13. package/dist/cron/cron-reconciler.d.ts +62 -0
  14. package/dist/cron/cron-reconciler.d.ts.map +1 -0
  15. package/dist/cron/cron-report-writer.d.ts +41 -0
  16. package/dist/cron/cron-report-writer.d.ts.map +1 -0
  17. package/dist/cron/cron-runner.d.ts +43 -0
  18. package/dist/cron/cron-runner.d.ts.map +1 -0
  19. package/dist/cron/cron-schema.d.ts +3 -0
  20. package/dist/cron/cron-schema.d.ts.map +1 -0
  21. package/dist/cron/cron-service.d.ts +57 -0
  22. package/dist/cron/cron-service.d.ts.map +1 -0
  23. package/dist/cron/cron-spec-parser.d.ts +27 -0
  24. package/dist/cron/cron-spec-parser.d.ts.map +1 -0
  25. package/dist/cron/cron-watcher.d.ts +23 -0
  26. package/dist/cron/cron-watcher.d.ts.map +1 -0
  27. package/dist/cron/resource-limiter.d.ts +9 -0
  28. package/dist/cron/resource-limiter.d.ts.map +1 -0
  29. package/dist/cron/schedule-command-service.d.ts +10 -0
  30. package/dist/cron/schedule-command-service.d.ts.map +1 -0
  31. package/dist/cron/schedule-service.d.ts +100 -0
  32. package/dist/cron/schedule-service.d.ts.map +1 -0
  33. package/dist/cron/scheduler.d.ts +68 -0
  34. package/dist/cron/scheduler.d.ts.map +1 -0
  35. package/dist/cron/sqlite-cron-store.d.ts +230 -0
  36. package/dist/cron/sqlite-cron-store.d.ts.map +1 -0
  37. package/dist/cron/sqlite-schedule-store.d.ts +52 -0
  38. package/dist/cron/sqlite-schedule-store.d.ts.map +1 -0
  39. package/dist/extensions/config/agent-config-loader.d.ts +4 -3
  40. package/dist/extensions/config/agent-config-loader.d.ts.map +1 -1
  41. package/dist/extensions/config/runtime-commands.d.ts +1 -0
  42. package/dist/extensions/config/runtime-commands.d.ts.map +1 -1
  43. package/dist/extensions/config/user-instruction-config-loader.d.ts +1 -0
  44. package/dist/extensions/config/user-instruction-config-loader.d.ts.map +1 -1
  45. package/dist/extensions/context/agentic-compaction.d.ts +2 -2
  46. package/dist/extensions/context/agentic-compaction.d.ts.map +1 -1
  47. package/dist/extensions/context/compaction-shared.d.ts +5 -4
  48. package/dist/extensions/context/compaction-shared.d.ts.map +1 -1
  49. package/dist/extensions/context/compaction.d.ts.map +1 -1
  50. package/dist/extensions/plugin/plugin-config-loader.d.ts +15 -2
  51. package/dist/extensions/plugin/plugin-config-loader.d.ts.map +1 -1
  52. package/dist/extensions/plugin/plugin-loader.d.ts +13 -7
  53. package/dist/extensions/plugin/plugin-loader.d.ts.map +1 -1
  54. package/dist/extensions/plugin/plugin-module-import.d.ts.map +1 -1
  55. package/dist/extensions/plugin/plugin-sandbox.d.ts +21 -2
  56. package/dist/extensions/plugin/plugin-sandbox.d.ts.map +1 -1
  57. package/dist/extensions/plugin/plugin-targeting.d.ts +7 -0
  58. package/dist/extensions/plugin/plugin-targeting.d.ts.map +1 -0
  59. package/dist/extensions/plugin-sandbox-bootstrap.js +237 -276
  60. package/dist/extensions/tools/constants.d.ts +1 -0
  61. package/dist/extensions/tools/constants.d.ts.map +1 -1
  62. package/dist/extensions/tools/definitions.d.ts +3 -4
  63. package/dist/extensions/tools/definitions.d.ts.map +1 -1
  64. package/dist/extensions/tools/executors/apply-patch.d.ts +3 -1
  65. package/dist/extensions/tools/executors/apply-patch.d.ts.map +1 -1
  66. package/dist/extensions/tools/executors/editor.d.ts.map +1 -1
  67. package/dist/extensions/tools/executors/search.d.ts +1 -1
  68. package/dist/extensions/tools/executors/search.d.ts.map +1 -1
  69. package/dist/extensions/tools/helpers.d.ts +1 -0
  70. package/dist/extensions/tools/helpers.d.ts.map +1 -1
  71. package/dist/extensions/tools/index.d.ts +3 -2
  72. package/dist/extensions/tools/index.d.ts.map +1 -1
  73. package/dist/extensions/tools/presets.d.ts +27 -44
  74. package/dist/extensions/tools/presets.d.ts.map +1 -1
  75. package/dist/extensions/tools/runtime.d.ts +25 -0
  76. package/dist/extensions/tools/runtime.d.ts.map +1 -0
  77. package/dist/extensions/tools/schemas.d.ts +25 -3
  78. package/dist/extensions/tools/schemas.d.ts.map +1 -1
  79. package/dist/extensions/tools/team/delegated-agent.d.ts +2 -2
  80. package/dist/extensions/tools/team/delegated-agent.d.ts.map +1 -1
  81. package/dist/extensions/tools/team/multi-agent.d.ts +7 -3
  82. package/dist/extensions/tools/team/multi-agent.d.ts.map +1 -1
  83. package/dist/extensions/tools/team/team-tools.d.ts +1 -0
  84. package/dist/extensions/tools/team/team-tools.d.ts.map +1 -1
  85. package/dist/extensions/tools/types.d.ts +0 -5
  86. package/dist/extensions/tools/types.d.ts.map +1 -1
  87. package/dist/hooks/hook-bridge.d.ts +118 -0
  88. package/dist/hooks/hook-bridge.d.ts.map +1 -0
  89. package/dist/hooks/hook-file-hooks.d.ts +6 -2
  90. package/dist/hooks/hook-file-hooks.d.ts.map +1 -1
  91. package/dist/hooks/hook-registry.d.ts +16 -0
  92. package/dist/hooks/hook-registry.d.ts.map +1 -0
  93. package/dist/hooks/index.d.ts +0 -1
  94. package/dist/hooks/index.d.ts.map +1 -1
  95. package/dist/hooks/subprocess.d.ts +8 -1
  96. package/dist/hooks/subprocess.d.ts.map +1 -1
  97. package/dist/hub/browser-websocket.d.ts +18 -0
  98. package/dist/hub/browser-websocket.d.ts.map +1 -0
  99. package/dist/hub/client.d.ts +51 -0
  100. package/dist/hub/client.d.ts.map +1 -0
  101. package/dist/hub/connect.d.ts +15 -0
  102. package/dist/hub/connect.d.ts.map +1 -0
  103. package/dist/hub/daemon-entry.d.ts +2 -0
  104. package/dist/hub/daemon-entry.d.ts.map +1 -0
  105. package/dist/hub/daemon-entry.js +1305 -0
  106. package/dist/hub/daemon.d.ts +5 -0
  107. package/dist/hub/daemon.d.ts.map +1 -0
  108. package/dist/hub/defaults.d.ts +17 -0
  109. package/dist/hub/defaults.d.ts.map +1 -0
  110. package/dist/hub/discovery.d.ts +29 -0
  111. package/dist/hub/discovery.d.ts.map +1 -0
  112. package/dist/hub/index.d.ts +15 -0
  113. package/dist/hub/index.d.ts.map +1 -0
  114. package/dist/hub/index.js +1294 -0
  115. package/dist/hub/native-transport.d.ts +17 -0
  116. package/dist/hub/native-transport.d.ts.map +1 -0
  117. package/dist/hub/runtime-handlers.d.ts +11 -0
  118. package/dist/hub/runtime-handlers.d.ts.map +1 -0
  119. package/dist/hub/server.d.ts +104 -0
  120. package/dist/hub/server.d.ts.map +1 -0
  121. package/dist/hub/session-client.d.ts +90 -0
  122. package/dist/hub/session-client.d.ts.map +1 -0
  123. package/dist/hub/start-shared-server.d.ts +19 -0
  124. package/dist/hub/start-shared-server.d.ts.map +1 -0
  125. package/dist/hub/transport.d.ts +8 -0
  126. package/dist/hub/transport.d.ts.map +1 -0
  127. package/dist/hub/ui-client.d.ts +45 -0
  128. package/dist/hub/ui-client.d.ts.map +1 -0
  129. package/dist/hub/workspace.d.ts +4 -0
  130. package/dist/hub/workspace.d.ts.map +1 -0
  131. package/dist/index.d.ts +29 -16
  132. package/dist/index.d.ts.map +1 -1
  133. package/dist/index.js +782 -471
  134. package/dist/llms/cline-recommended-models.d.ts +20 -0
  135. package/dist/llms/cline-recommended-models.d.ts.map +1 -0
  136. package/dist/llms/configured-provider-registry.d.ts +28 -0
  137. package/dist/llms/configured-provider-registry.d.ts.map +1 -0
  138. package/dist/llms/handler-factory.d.ts +16 -0
  139. package/dist/llms/handler-factory.d.ts.map +1 -0
  140. package/dist/llms/provider-defaults.d.ts +27 -0
  141. package/dist/llms/provider-defaults.d.ts.map +1 -0
  142. package/dist/llms/provider-settings.d.ts +245 -0
  143. package/dist/llms/provider-settings.d.ts.map +1 -0
  144. package/dist/llms/runtime-config.d.ts +4 -0
  145. package/dist/llms/runtime-config.d.ts.map +1 -0
  146. package/dist/llms/runtime-registry.d.ts +20 -0
  147. package/dist/llms/runtime-registry.d.ts.map +1 -0
  148. package/dist/llms/runtime-types.d.ts +85 -0
  149. package/dist/llms/runtime-types.d.ts.map +1 -0
  150. package/dist/runtime/agent-config-adapter.d.ts +148 -0
  151. package/dist/runtime/agent-config-adapter.d.ts.map +1 -0
  152. package/dist/runtime/agent-runtime-config-builder.d.ts +96 -0
  153. package/dist/runtime/agent-runtime-config-builder.d.ts.map +1 -0
  154. package/dist/runtime/history.d.ts +6 -0
  155. package/dist/runtime/history.d.ts.map +1 -1
  156. package/dist/runtime/host.d.ts +1 -2
  157. package/dist/runtime/host.d.ts.map +1 -1
  158. package/dist/runtime/loop-detection.d.ts +59 -0
  159. package/dist/runtime/loop-detection.d.ts.map +1 -0
  160. package/dist/runtime/mistake-tracker.d.ts +69 -0
  161. package/dist/runtime/mistake-tracker.d.ts.map +1 -0
  162. package/dist/runtime/rules.d.ts +1 -0
  163. package/dist/runtime/rules.d.ts.map +1 -1
  164. package/dist/runtime/runtime-builder.d.ts.map +1 -1
  165. package/dist/runtime/runtime-event-adapter.d.ts +102 -0
  166. package/dist/runtime/runtime-event-adapter.d.ts.map +1 -0
  167. package/dist/runtime/runtime-host.d.ts +49 -26
  168. package/dist/runtime/runtime-host.d.ts.map +1 -1
  169. package/dist/runtime/runtime-oauth-token-manager.d.ts.map +1 -1
  170. package/dist/runtime/session-runtime-orchestrator.d.ts +261 -0
  171. package/dist/runtime/session-runtime-orchestrator.d.ts.map +1 -0
  172. package/dist/runtime/session-runtime.d.ts +16 -21
  173. package/dist/runtime/session-runtime.d.ts.map +1 -1
  174. package/dist/runtime/user-input-builder.d.ts +24 -0
  175. package/dist/runtime/user-input-builder.d.ts.map +1 -0
  176. package/dist/services/global-settings.d.ts +12 -0
  177. package/dist/services/global-settings.d.ts.map +1 -0
  178. package/dist/services/index.js +28 -0
  179. package/dist/services/local-runtime-bootstrap.d.ts +9 -3
  180. package/dist/services/local-runtime-bootstrap.d.ts.map +1 -1
  181. package/dist/services/plugin-tools.d.ts +16 -0
  182. package/dist/services/plugin-tools.d.ts.map +1 -0
  183. package/dist/services/providers/local-provider-registry.d.ts +199 -23
  184. package/dist/services/providers/local-provider-registry.d.ts.map +1 -1
  185. package/dist/services/providers/local-provider-service.d.ts +15 -13
  186. package/dist/services/providers/local-provider-service.d.ts.map +1 -1
  187. package/dist/services/session-data.d.ts +1 -1
  188. package/dist/services/session-data.d.ts.map +1 -1
  189. package/dist/services/session-telemetry.d.ts +7 -2
  190. package/dist/services/session-telemetry.d.ts.map +1 -1
  191. package/dist/services/storage/file-team-store.d.ts.map +1 -1
  192. package/dist/services/storage/provider-settings-legacy-migration.d.ts +1 -1
  193. package/dist/services/storage/provider-settings-legacy-migration.d.ts.map +1 -1
  194. package/dist/services/storage/provider-settings-manager.d.ts +1 -0
  195. package/dist/services/storage/provider-settings-manager.d.ts.map +1 -1
  196. package/dist/services/storage/sqlite-team-store.d.ts.map +1 -1
  197. package/dist/services/workspace-manifest.d.ts +11 -0
  198. package/dist/services/workspace-manifest.d.ts.map +1 -1
  199. package/dist/session/conversation-store.d.ts +30 -0
  200. package/dist/session/conversation-store.d.ts.map +1 -0
  201. package/dist/session/message-builder.d.ts +65 -0
  202. package/dist/session/message-builder.d.ts.map +1 -0
  203. package/dist/session/persistence-service.d.ts +11 -23
  204. package/dist/session/persistence-service.d.ts.map +1 -1
  205. package/dist/session/session-manifest-store.d.ts +22 -0
  206. package/dist/session/session-manifest-store.d.ts.map +1 -0
  207. package/dist/session/session-manifest.d.ts +1 -1
  208. package/dist/session/session-row.d.ts +93 -0
  209. package/dist/session/session-row.d.ts.map +1 -0
  210. package/dist/session/session-service.d.ts +2 -102
  211. package/dist/session/session-service.d.ts.map +1 -1
  212. package/dist/session/subagent-session-manager.d.ts +36 -0
  213. package/dist/session/subagent-session-manager.d.ts.map +1 -0
  214. package/dist/session/team-persistence-store.d.ts +24 -0
  215. package/dist/session/team-persistence-store.d.ts.map +1 -0
  216. package/dist/transports/hub.d.ts +58 -0
  217. package/dist/transports/hub.d.ts.map +1 -0
  218. package/dist/transports/local.d.ts +23 -9
  219. package/dist/transports/local.d.ts.map +1 -1
  220. package/dist/transports/remote.d.ts +10 -0
  221. package/dist/transports/remote.d.ts.map +1 -0
  222. package/dist/transports/runtime-host-support.d.ts +3 -2
  223. package/dist/transports/runtime-host-support.d.ts.map +1 -1
  224. package/dist/types/chat-schema.d.ts +15 -17
  225. package/dist/types/chat-schema.d.ts.map +1 -1
  226. package/dist/types/config.d.ts +17 -7
  227. package/dist/types/config.d.ts.map +1 -1
  228. package/dist/types/events.d.ts +7 -6
  229. package/dist/types/events.d.ts.map +1 -1
  230. package/dist/types/provider-settings.d.ts +4 -5
  231. package/dist/types/provider-settings.d.ts.map +1 -1
  232. package/dist/types/session.d.ts +7 -3
  233. package/dist/types/session.d.ts.map +1 -1
  234. package/dist/types.d.ts +11 -4
  235. package/dist/types.d.ts.map +1 -1
  236. package/package.json +20 -6
  237. package/src/ClineCore.ts +757 -44
  238. package/src/account/cline-account-service.ts +44 -6
  239. package/src/account/index.ts +3 -3
  240. package/src/account/rpc.ts +12 -12
  241. package/src/cron/cron-event-ingress.ts +357 -0
  242. package/src/cron/cron-materializer.ts +97 -0
  243. package/src/cron/cron-reconciler.ts +241 -0
  244. package/src/cron/cron-report-writer.ts +153 -0
  245. package/src/cron/cron-runner.ts +495 -0
  246. package/src/cron/cron-schema.ts +127 -0
  247. package/src/cron/cron-service.ts +163 -0
  248. package/src/cron/cron-spec-parser.ts +489 -0
  249. package/src/cron/cron-watcher.ts +102 -0
  250. package/src/cron/index.ts +15 -0
  251. package/src/cron/resource-limiter.ts +46 -0
  252. package/src/cron/schedule-command-service.ts +193 -0
  253. package/src/cron/schedule-service.ts +703 -0
  254. package/src/cron/scheduler.ts +772 -0
  255. package/src/cron/sqlite-cron-store.ts +1286 -0
  256. package/src/cron/sqlite-schedule-store.ts +708 -0
  257. package/src/extensions/config/agent-config-loader.ts +17 -7
  258. package/src/extensions/config/runtime-commands.ts +6 -0
  259. package/src/extensions/config/user-instruction-config-loader.ts +1 -0
  260. package/src/extensions/context/agentic-compaction.ts +3 -3
  261. package/src/extensions/context/basic-compaction.ts +2 -2
  262. package/src/extensions/context/compaction-shared.ts +5 -4
  263. package/src/extensions/context/compaction.ts +3 -3
  264. package/src/extensions/plugin/plugin-config-loader.ts +37 -2
  265. package/src/extensions/plugin/plugin-loader.ts +69 -9
  266. package/src/extensions/plugin/plugin-module-import.ts +0 -2
  267. package/src/extensions/plugin/plugin-sandbox-bootstrap.ts +243 -39
  268. package/src/extensions/plugin/plugin-sandbox.ts +173 -29
  269. package/src/extensions/plugin/plugin-targeting.ts +32 -0
  270. package/src/extensions/tools/constants.ts +2 -0
  271. package/src/extensions/tools/definitions.ts +61 -71
  272. package/src/extensions/tools/executors/apply-patch.ts +69 -80
  273. package/src/extensions/tools/executors/editor.ts +4 -3
  274. package/src/extensions/tools/executors/search.ts +195 -3
  275. package/src/extensions/tools/helpers.ts +24 -0
  276. package/src/extensions/tools/index.ts +11 -2
  277. package/src/extensions/tools/presets.ts +32 -47
  278. package/src/extensions/tools/runtime.ts +261 -0
  279. package/src/extensions/tools/schemas.ts +17 -20
  280. package/src/extensions/tools/team/delegated-agent.ts +8 -3
  281. package/src/extensions/tools/team/multi-agent.ts +135 -19
  282. package/src/extensions/tools/team/team-tools.ts +172 -91
  283. package/src/extensions/tools/types.ts +0 -6
  284. package/src/hooks/hook-bridge.ts +489 -0
  285. package/src/hooks/hook-file-hooks.ts +66 -5
  286. package/src/hooks/hook-registry.ts +257 -0
  287. package/src/hooks/index.ts +0 -7
  288. package/src/hooks/subprocess-runner.ts +1 -1
  289. package/src/hooks/subprocess.ts +9 -0
  290. package/src/hub/browser-websocket.ts +159 -0
  291. package/src/hub/client.ts +633 -0
  292. package/src/hub/connect.ts +156 -0
  293. package/src/hub/daemon-entry.ts +122 -0
  294. package/src/hub/daemon.ts +284 -0
  295. package/src/hub/defaults.ts +70 -0
  296. package/src/hub/discovery.ts +247 -0
  297. package/src/hub/index.ts +14 -0
  298. package/src/hub/native-transport.ts +31 -0
  299. package/src/hub/runtime-handlers.ts +141 -0
  300. package/src/hub/server.ts +2317 -0
  301. package/src/hub/session-client.ts +502 -0
  302. package/src/hub/start-shared-server.ts +61 -0
  303. package/src/hub/transport.ts +14 -0
  304. package/src/hub/ui-client.ts +126 -0
  305. package/src/hub/workspace.ts +19 -0
  306. package/src/index.ts +169 -68
  307. package/src/llms/cline-recommended-models.ts +167 -0
  308. package/src/llms/configured-provider-registry.ts +193 -0
  309. package/src/llms/handler-factory.ts +56 -0
  310. package/src/llms/provider-defaults.ts +653 -0
  311. package/src/llms/provider-settings.ts +310 -0
  312. package/src/llms/runtime-config.ts +43 -0
  313. package/src/llms/runtime-registry.ts +172 -0
  314. package/src/llms/runtime-types.ts +121 -0
  315. package/src/runtime/agent-config-adapter.ts +636 -0
  316. package/src/runtime/agent-runtime-config-builder.ts +205 -0
  317. package/src/runtime/error-feedback.ts +142 -0
  318. package/src/runtime/history.ts +137 -0
  319. package/src/runtime/host.ts +127 -267
  320. package/src/runtime/index.ts +1 -0
  321. package/src/runtime/loop-detection.ts +162 -0
  322. package/src/runtime/mistake-tracker.ts +221 -0
  323. package/src/runtime/rules.ts +12 -0
  324. package/src/runtime/runtime-builder.ts +85 -13
  325. package/src/runtime/runtime-event-adapter.ts +412 -0
  326. package/src/runtime/runtime-host.ts +134 -62
  327. package/src/runtime/runtime-oauth-token-manager.ts +11 -15
  328. package/src/runtime/session-runtime-orchestrator.ts +1253 -0
  329. package/src/runtime/session-runtime.ts +16 -26
  330. package/src/runtime/user-input-builder.ts +167 -0
  331. package/src/services/global-settings.ts +122 -0
  332. package/src/services/local-runtime-bootstrap.ts +175 -31
  333. package/src/services/plugin-tools.ts +86 -0
  334. package/src/services/providers/local-provider-registry.ts +277 -61
  335. package/src/services/providers/local-provider-service.ts +109 -44
  336. package/src/services/session-data.ts +18 -10
  337. package/src/services/session-telemetry.ts +6 -15
  338. package/src/services/storage/file-team-store.ts +1 -5
  339. package/src/services/storage/provider-settings-legacy-migration.ts +14 -51
  340. package/src/services/storage/provider-settings-manager.ts +17 -2
  341. package/src/services/storage/sqlite-team-store.ts +1 -5
  342. package/src/services/workspace-manifest.ts +18 -0
  343. package/src/session/conversation-store.ts +77 -0
  344. package/src/session/file-session-service.ts +1 -1
  345. package/src/session/index.ts +6 -27
  346. package/src/session/message-builder.ts +941 -0
  347. package/src/session/persistence-service.ts +119 -504
  348. package/src/session/session-manifest-store.ts +158 -0
  349. package/src/session/session-row.ts +199 -0
  350. package/src/session/session-service.ts +17 -376
  351. package/src/session/session-team-coordination.ts +1 -1
  352. package/src/session/subagent-session-manager.ts +397 -0
  353. package/src/session/team-persistence-store.ts +176 -0
  354. package/src/transports/hub.ts +1081 -0
  355. package/src/transports/local.ts +419 -93
  356. package/src/transports/remote.ts +27 -0
  357. package/src/transports/runtime-host-support.ts +63 -9
  358. package/src/types/chat-schema.ts +4 -5
  359. package/src/types/config.ts +17 -7
  360. package/src/types/events.ts +8 -6
  361. package/src/types/index.ts +3 -0
  362. package/src/types/provider-settings.ts +18 -7
  363. package/src/types/session.ts +7 -6
  364. package/src/types.ts +42 -2
  365. package/dist/hooks/persistent.d.ts +0 -64
  366. package/dist/hooks/persistent.d.ts.map +0 -1
  367. package/dist/runtime/rpc-runtime-ensure.d.ts +0 -65
  368. package/dist/runtime/rpc-runtime-ensure.d.ts.map +0 -1
  369. package/dist/runtime/rpc-spawn-lease.d.ts +0 -8
  370. package/dist/runtime/rpc-spawn-lease.d.ts.map +0 -1
  371. package/dist/services/telemetry/index.js +0 -15
  372. package/dist/session/rpc-session-service.d.ts +0 -16
  373. package/dist/session/rpc-session-service.d.ts.map +0 -1
  374. package/dist/session/sqlite-rpc-session-backend.d.ts +0 -31
  375. package/dist/session/sqlite-rpc-session-backend.d.ts.map +0 -1
  376. package/dist/transports/rpc.d.ts +0 -51
  377. package/dist/transports/rpc.d.ts.map +0 -1
  378. package/src/ClineCore.test.ts +0 -226
  379. package/src/account/cline-account-service.test.ts +0 -185
  380. package/src/account/featurebase-token.test.ts +0 -175
  381. package/src/account/rpc.test.ts +0 -63
  382. package/src/auth/bounded-ttl-cache.test.ts +0 -38
  383. package/src/auth/client.test.ts +0 -69
  384. package/src/auth/cline.test.ts +0 -267
  385. package/src/auth/codex.test.ts +0 -170
  386. package/src/auth/oca.test.ts +0 -340
  387. package/src/auth/server.test.ts +0 -287
  388. package/src/auth/utils.test.ts +0 -128
  389. package/src/extensions/config/agent-config-loader.test.ts +0 -236
  390. package/src/extensions/config/hooks-config-loader.test.ts +0 -20
  391. package/src/extensions/config/runtime-commands.test.ts +0 -115
  392. package/src/extensions/config/unified-config-file-watcher.test.ts +0 -196
  393. package/src/extensions/config/user-instruction-config-loader.test.ts +0 -246
  394. package/src/extensions/context/compaction.test.ts +0 -483
  395. package/src/extensions/mcp/config-loader.test.ts +0 -238
  396. package/src/extensions/mcp/manager.test.ts +0 -105
  397. package/src/extensions/plugin/plugin-config-loader.test.ts +0 -184
  398. package/src/extensions/plugin/plugin-loader.test.ts +0 -292
  399. package/src/extensions/plugin/plugin-sandbox.test.ts +0 -423
  400. package/src/extensions/tools/definitions.test.ts +0 -780
  401. package/src/extensions/tools/executors/bash.test.ts +0 -87
  402. package/src/extensions/tools/executors/editor.test.ts +0 -35
  403. package/src/extensions/tools/executors/file-read.test.ts +0 -125
  404. package/src/extensions/tools/model-tool-routing.test.ts +0 -86
  405. package/src/extensions/tools/presets.test.ts +0 -70
  406. package/src/extensions/tools/team/multi-agent.lifecycle.test.ts +0 -455
  407. package/src/extensions/tools/team/spawn-agent-tool.test.ts +0 -381
  408. package/src/extensions/tools/team/team-tools.test.ts +0 -918
  409. package/src/hooks/checkpoint-hooks.test.ts +0 -168
  410. package/src/hooks/hook-file-hooks.test.ts +0 -311
  411. package/src/hooks/persistent.ts +0 -661
  412. package/src/runtime/history.test.ts +0 -114
  413. package/src/runtime/host.test.ts +0 -230
  414. package/src/runtime/rpc-runtime-ensure.test.ts +0 -123
  415. package/src/runtime/rpc-runtime-ensure.ts +0 -659
  416. package/src/runtime/rpc-spawn-lease.test.ts +0 -81
  417. package/src/runtime/rpc-spawn-lease.ts +0 -156
  418. package/src/runtime/runtime-builder.team-persistence.test.ts +0 -245
  419. package/src/runtime/runtime-builder.test.ts +0 -615
  420. package/src/runtime/runtime-oauth-token-manager.test.ts +0 -137
  421. package/src/runtime/runtime-parity.test.ts +0 -143
  422. package/src/services/providers/local-provider-service.test.ts +0 -1062
  423. package/src/services/session-data.test.ts +0 -160
  424. package/src/services/storage/provider-settings-legacy-migration.test.ts +0 -424
  425. package/src/services/storage/provider-settings-manager.test.ts +0 -191
  426. package/src/services/telemetry/OpenTelemetryAdapter.test.ts +0 -157
  427. package/src/services/telemetry/OpenTelemetryProvider.test.ts +0 -326
  428. package/src/services/telemetry/TelemetryLoggerSink.test.ts +0 -42
  429. package/src/services/telemetry/TelemetryService.test.ts +0 -134
  430. package/src/services/telemetry/distinct-id.test.ts +0 -57
  431. package/src/services/workspace/file-indexer.d.ts +0 -11
  432. package/src/services/workspace/file-indexer.test.ts +0 -156
  433. package/src/services/workspace/mention-enricher.test.ts +0 -106
  434. package/src/session/persistence-service.test.ts +0 -300
  435. package/src/session/rpc-session-service.ts +0 -114
  436. package/src/session/session-service.team-persistence.test.ts +0 -48
  437. package/src/session/sqlite-rpc-session-backend.ts +0 -301
  438. package/src/transports/local.e2e.test.ts +0 -380
  439. package/src/transports/local.test.ts +0 -2559
  440. package/src/transports/rpc.test.ts +0 -82
  441. package/src/transports/rpc.ts +0 -665
@@ -0,0 +1,772 @@
1
+ import {
2
+ createSessionId,
3
+ type HubScheduleCreateInput,
4
+ type HubScheduleUpdateInput,
5
+ type ScheduleExecutionRecord,
6
+ type ScheduleExecutionStatus,
7
+ type ScheduleRecord,
8
+ } from "@clinebot/shared";
9
+
10
+ function createHubId(prefix: string): string {
11
+ return createSessionId(`${prefix}_`);
12
+ }
13
+
14
+ function parseCronField(
15
+ token: string,
16
+ min: number,
17
+ max: number,
18
+ names?: readonly string[],
19
+ ): number[] {
20
+ const results = new Set<number>();
21
+
22
+ function resolveValue(raw: string): number {
23
+ const lower = raw.toLowerCase();
24
+ if (names) {
25
+ const index = names.indexOf(lower);
26
+ if (index !== -1) {
27
+ return index + min;
28
+ }
29
+ }
30
+ const value = Number(raw);
31
+ if (!Number.isInteger(value) || value < min || value > max) {
32
+ throw new Error(`Invalid cron value "${raw}" for range [${min}-${max}]`);
33
+ }
34
+ return value;
35
+ }
36
+
37
+ for (const part of token.split(",")) {
38
+ if (part === "*") {
39
+ for (let value = min; value <= max; value += 1) {
40
+ results.add(value);
41
+ }
42
+ continue;
43
+ }
44
+
45
+ const stepSeparator = part.indexOf("/");
46
+ if (stepSeparator !== -1) {
47
+ const rangePart = part.slice(0, stepSeparator);
48
+ const step = Number(part.slice(stepSeparator + 1));
49
+ if (!Number.isInteger(step) || step < 1) {
50
+ throw new Error(`Invalid step "${part.slice(stepSeparator + 1)}"`);
51
+ }
52
+ let from = min;
53
+ let to = max;
54
+ if (rangePart !== "*") {
55
+ const dashIndex = rangePart.indexOf("-");
56
+ if (dashIndex !== -1) {
57
+ from = resolveValue(rangePart.slice(0, dashIndex));
58
+ to = resolveValue(rangePart.slice(dashIndex + 1));
59
+ } else {
60
+ from = resolveValue(rangePart);
61
+ }
62
+ }
63
+ if (from > to) {
64
+ throw new Error(`Invalid cron range "${rangePart}"`);
65
+ }
66
+ for (let value = from; value <= to; value += step) {
67
+ results.add(value);
68
+ }
69
+ continue;
70
+ }
71
+
72
+ const dashIndex = part.indexOf("-");
73
+ if (dashIndex !== -1) {
74
+ const from = resolveValue(part.slice(0, dashIndex));
75
+ const to = resolveValue(part.slice(dashIndex + 1));
76
+ if (from > to) {
77
+ throw new Error(`Invalid cron range "${part}"`);
78
+ }
79
+ for (let value = from; value <= to; value += 1) {
80
+ results.add(value);
81
+ }
82
+ continue;
83
+ }
84
+
85
+ results.add(resolveValue(part));
86
+ }
87
+
88
+ return [...results].sort((left, right) => left - right);
89
+ }
90
+
91
+ const MONTH_NAMES = [
92
+ "jan",
93
+ "feb",
94
+ "mar",
95
+ "apr",
96
+ "may",
97
+ "jun",
98
+ "jul",
99
+ "aug",
100
+ "sep",
101
+ "oct",
102
+ "nov",
103
+ "dec",
104
+ ] as const;
105
+
106
+ const DOW_NAMES = ["sun", "mon", "tue", "wed", "thu", "fri", "sat"] as const;
107
+
108
+ export interface ParsedCron {
109
+ minutes: number[];
110
+ hours: number[];
111
+ daysOfMonth: number[];
112
+ months: number[];
113
+ daysOfWeek: number[];
114
+ }
115
+
116
+ interface CronDateParts {
117
+ month: number;
118
+ dayOfMonth: number;
119
+ dayOfWeek: number;
120
+ hour: number;
121
+ minute: number;
122
+ }
123
+
124
+ function getRequiredField(
125
+ fields: readonly string[],
126
+ index: number,
127
+ pattern: string,
128
+ ): string {
129
+ const value = fields[index];
130
+ if (typeof value !== "string") {
131
+ throw new Error(
132
+ `Invalid cron pattern "${pattern}": missing field ${index + 1}`,
133
+ );
134
+ }
135
+ return value;
136
+ }
137
+
138
+ function getFirstCronValue(values: readonly number[], label: string): number {
139
+ const value = values[0];
140
+ if (typeof value !== "number") {
141
+ throw new Error(`Invalid cron pattern: no values parsed for ${label}`);
142
+ }
143
+ return value;
144
+ }
145
+
146
+ export function parseCron(pattern: string): ParsedCron {
147
+ const fields = pattern.trim().split(/\s+/);
148
+ if (fields.length !== 5) {
149
+ throw new Error(
150
+ `Invalid cron pattern "${pattern}": expected 5 fields, got ${fields.length}`,
151
+ );
152
+ }
153
+ return {
154
+ minutes: parseCronField(getRequiredField(fields, 0, pattern), 0, 59),
155
+ hours: parseCronField(getRequiredField(fields, 1, pattern), 0, 23),
156
+ daysOfMonth: parseCronField(getRequiredField(fields, 2, pattern), 1, 31),
157
+ months: parseCronField(
158
+ getRequiredField(fields, 3, pattern),
159
+ 1,
160
+ 12,
161
+ MONTH_NAMES,
162
+ ),
163
+ daysOfWeek: parseCronField(
164
+ getRequiredField(fields, 4, pattern),
165
+ 0,
166
+ 6,
167
+ DOW_NAMES,
168
+ ),
169
+ };
170
+ }
171
+
172
+ export function validateCronPattern(pattern: string): void {
173
+ parseCron(pattern);
174
+ }
175
+
176
+ export function validateCronSchedule(
177
+ pattern: string,
178
+ timezone?: string,
179
+ after: number = Date.now(),
180
+ ): void {
181
+ getNextCronTime(pattern, after, timezone);
182
+ }
183
+
184
+ const TIMEZONE_FORMATTERS = new Map<string, Intl.DateTimeFormat>();
185
+ const DOW_BY_SHORT_NAME = new Map(
186
+ DOW_NAMES.map((name, index) => [name, index] as const),
187
+ );
188
+
189
+ function normalizeTimezone(timezone: string | undefined): string | undefined {
190
+ const trimmed = timezone?.trim();
191
+ return trimmed ? trimmed : undefined;
192
+ }
193
+
194
+ function getTimezoneFormatter(timezone: string): Intl.DateTimeFormat {
195
+ const existing = TIMEZONE_FORMATTERS.get(timezone);
196
+ if (existing) return existing;
197
+ const formatter = new Intl.DateTimeFormat("en-US", {
198
+ timeZone: timezone,
199
+ calendar: "gregory",
200
+ numberingSystem: "latn",
201
+ year: "numeric",
202
+ month: "2-digit",
203
+ day: "2-digit",
204
+ weekday: "short",
205
+ hour: "2-digit",
206
+ minute: "2-digit",
207
+ hourCycle: "h23",
208
+ });
209
+ TIMEZONE_FORMATTERS.set(timezone, formatter);
210
+ return formatter;
211
+ }
212
+
213
+ export function validateTimezone(timezone: string | undefined): void {
214
+ const normalized = normalizeTimezone(timezone);
215
+ if (!normalized) return;
216
+ getTimezoneFormatter(normalized).format(new Date());
217
+ }
218
+
219
+ function getZonedCronDateParts(
220
+ timestampMs: number,
221
+ timezone: string,
222
+ ): CronDateParts {
223
+ const parts = getTimezoneFormatter(timezone).formatToParts(
224
+ new Date(timestampMs),
225
+ );
226
+ const byType = new Map(parts.map((part) => [part.type, part.value]));
227
+ const weekdayRaw = byType.get("weekday")?.toLowerCase().slice(0, 3) ?? "";
228
+ const dayOfWeek = DOW_BY_SHORT_NAME.get(
229
+ weekdayRaw as (typeof DOW_NAMES)[number],
230
+ );
231
+ if (dayOfWeek === undefined) {
232
+ throw new Error(`Unable to resolve weekday for timezone "${timezone}"`);
233
+ }
234
+ return {
235
+ month: Number(byType.get("month")),
236
+ dayOfMonth: Number(byType.get("day")),
237
+ dayOfWeek,
238
+ hour: Number(byType.get("hour")),
239
+ minute: Number(byType.get("minute")),
240
+ };
241
+ }
242
+
243
+ function getLocalCronDateParts(timestampMs: number): CronDateParts {
244
+ const date = new Date(timestampMs);
245
+ return {
246
+ month: date.getMonth() + 1,
247
+ dayOfMonth: date.getDate(),
248
+ dayOfWeek: date.getDay(),
249
+ hour: date.getHours(),
250
+ minute: date.getMinutes(),
251
+ };
252
+ }
253
+
254
+ function cronMatchesParts(cron: ParsedCron, parts: CronDateParts): boolean {
255
+ return (
256
+ cron.months.includes(parts.month) &&
257
+ cron.daysOfMonth.includes(parts.dayOfMonth) &&
258
+ cron.daysOfWeek.includes(parts.dayOfWeek) &&
259
+ cron.hours.includes(parts.hour) &&
260
+ cron.minutes.includes(parts.minute)
261
+ );
262
+ }
263
+
264
+ function getNextCronTimeByMinuteScan(
265
+ pattern: string,
266
+ after: number,
267
+ timezone: string,
268
+ ): number {
269
+ const cron = parseCron(pattern);
270
+ const next = new Date(after);
271
+ next.setSeconds(0, 0);
272
+ let nextMs = next.getTime() + 60_000;
273
+
274
+ const limit = new Date(after);
275
+ limit.setFullYear(limit.getFullYear() + 4);
276
+ const limitMs = limit.getTime();
277
+
278
+ while (nextMs <= limitMs) {
279
+ if (cronMatchesParts(cron, getZonedCronDateParts(nextMs, timezone))) {
280
+ return nextMs;
281
+ }
282
+ nextMs += 60_000;
283
+ }
284
+
285
+ throw new Error(
286
+ `No cron occurrence found within 4 years for pattern "${pattern}" in timezone "${timezone}"`,
287
+ );
288
+ }
289
+
290
+ export function getNextCronTime(
291
+ pattern: string,
292
+ after: number,
293
+ timezone?: string,
294
+ ): number {
295
+ const normalizedTimezone = normalizeTimezone(timezone);
296
+ if (normalizedTimezone) {
297
+ validateTimezone(normalizedTimezone);
298
+ return getNextCronTimeByMinuteScan(pattern, after, normalizedTimezone);
299
+ }
300
+
301
+ const cron = parseCron(pattern);
302
+ let next = new Date(after);
303
+ next.setSeconds(0, 0);
304
+ next = new Date(next.getTime() + 60_000);
305
+
306
+ const limit = new Date(after);
307
+ limit.setFullYear(limit.getFullYear() + 4);
308
+
309
+ while (next <= limit) {
310
+ const { month, dayOfMonth, dayOfWeek, hour, minute } =
311
+ getLocalCronDateParts(next.getTime());
312
+
313
+ if (!cron.months.includes(month)) {
314
+ const targetMonth =
315
+ cron.months.find((value) => value > month) ??
316
+ getFirstCronValue(cron.months, "months");
317
+ const yearDelta = targetMonth <= month ? 1 : 0;
318
+ next = new Date(
319
+ next.getFullYear() + yearDelta,
320
+ targetMonth - 1,
321
+ 1,
322
+ 0,
323
+ 0,
324
+ 0,
325
+ 0,
326
+ );
327
+ continue;
328
+ }
329
+
330
+ if (
331
+ !cron.daysOfMonth.includes(dayOfMonth) ||
332
+ !cron.daysOfWeek.includes(dayOfWeek)
333
+ ) {
334
+ next = new Date(
335
+ next.getFullYear(),
336
+ next.getMonth(),
337
+ dayOfMonth + 1,
338
+ 0,
339
+ 0,
340
+ 0,
341
+ 0,
342
+ );
343
+ continue;
344
+ }
345
+
346
+ if (!cron.hours.includes(hour)) {
347
+ const targetHour =
348
+ cron.hours.find((value) => value > hour) ??
349
+ getFirstCronValue(cron.hours, "hours");
350
+ const dayDelta = targetHour <= hour ? 1 : 0;
351
+ next = new Date(
352
+ next.getFullYear(),
353
+ next.getMonth(),
354
+ next.getDate() + dayDelta,
355
+ targetHour,
356
+ 0,
357
+ 0,
358
+ 0,
359
+ );
360
+ continue;
361
+ }
362
+
363
+ if (!cron.minutes.includes(minute)) {
364
+ const targetMinute =
365
+ cron.minutes.find((value) => value > minute) ??
366
+ getFirstCronValue(cron.minutes, "minutes");
367
+ const hourDelta = targetMinute <= minute ? 1 : 0;
368
+ next = new Date(
369
+ next.getFullYear(),
370
+ next.getMonth(),
371
+ next.getDate(),
372
+ next.getHours() + hourDelta,
373
+ targetMinute,
374
+ 0,
375
+ 0,
376
+ );
377
+ continue;
378
+ }
379
+
380
+ return next.getTime();
381
+ }
382
+
383
+ throw new Error(
384
+ `No cron occurrence found within 4 years for pattern "${pattern}"`,
385
+ );
386
+ }
387
+
388
+ export class ScheduleStore {
389
+ private schedules = new Map<string, ScheduleRecord>();
390
+ private executions = new Map<string, ScheduleExecutionRecord>();
391
+
392
+ load(
393
+ schedules: ScheduleRecord[] | undefined,
394
+ executions: ScheduleExecutionRecord[] | undefined,
395
+ ): void {
396
+ this.schedules.clear();
397
+ this.executions.clear();
398
+ for (const schedule of schedules ?? []) {
399
+ this.schedules.set(schedule.scheduleId, { ...schedule });
400
+ }
401
+ for (const execution of executions ?? []) {
402
+ this.executions.set(execution.executionId, { ...execution });
403
+ }
404
+ }
405
+
406
+ snapshotSchedules(): ScheduleRecord[] {
407
+ return [...this.schedules.values()].map((record) => ({ ...record }));
408
+ }
409
+
410
+ snapshotExecutions(): ScheduleExecutionRecord[] {
411
+ return [...this.executions.values()].map((record) => ({ ...record }));
412
+ }
413
+
414
+ create(input: HubScheduleCreateInput, now: number): ScheduleRecord {
415
+ validateCronPattern(input.cronPattern);
416
+ const nextRunAt = getNextCronTime(input.cronPattern, now);
417
+ const record: ScheduleRecord = {
418
+ scheduleId: createHubId("sched"),
419
+ name: input.name,
420
+ cronPattern: input.cronPattern,
421
+ prompt: input.prompt,
422
+ workspaceRoot: input.workspaceRoot,
423
+ enabled: input.enabled ?? true,
424
+ createdAt: now,
425
+ updatedAt: now,
426
+ nextRunAt,
427
+ runtimeOptions: input.runtimeOptions
428
+ ? { ...input.runtimeOptions }
429
+ : undefined,
430
+ metadata: input.metadata ? { ...input.metadata } : undefined,
431
+ };
432
+ this.schedules.set(record.scheduleId, record);
433
+ return { ...record };
434
+ }
435
+
436
+ get(scheduleId: string): ScheduleRecord | undefined {
437
+ const record = this.schedules.get(scheduleId);
438
+ return record ? { ...record } : undefined;
439
+ }
440
+
441
+ list(): ScheduleRecord[] {
442
+ return [...this.schedules.values()].map((record) => ({ ...record }));
443
+ }
444
+
445
+ update(input: HubScheduleUpdateInput, now: number): ScheduleRecord {
446
+ const existing = this.schedules.get(input.scheduleId);
447
+ if (!existing) {
448
+ throw new Error(`Schedule "${input.scheduleId}" not found`);
449
+ }
450
+ const cronPattern = input.cronPattern ?? existing.cronPattern;
451
+ if (input.cronPattern) {
452
+ validateCronPattern(input.cronPattern);
453
+ }
454
+ const enabled = input.enabled ?? existing.enabled;
455
+ const nextRunAt =
456
+ (input.cronPattern || input.enabled !== undefined) && enabled
457
+ ? getNextCronTime(cronPattern, now)
458
+ : enabled
459
+ ? existing.nextRunAt
460
+ : undefined;
461
+ const updated: ScheduleRecord = {
462
+ ...existing,
463
+ name: input.name ?? existing.name,
464
+ cronPattern,
465
+ prompt: input.prompt ?? existing.prompt,
466
+ enabled,
467
+ updatedAt: now,
468
+ nextRunAt,
469
+ runtimeOptions:
470
+ input.runtimeOptions !== undefined
471
+ ? { ...input.runtimeOptions }
472
+ : existing.runtimeOptions,
473
+ metadata:
474
+ input.metadata !== undefined
475
+ ? { ...input.metadata }
476
+ : existing.metadata,
477
+ };
478
+ this.schedules.set(updated.scheduleId, updated);
479
+ return { ...updated };
480
+ }
481
+
482
+ setEnabled(
483
+ scheduleId: string,
484
+ enabled: boolean,
485
+ now: number,
486
+ ): ScheduleRecord {
487
+ const existing = this.schedules.get(scheduleId);
488
+ if (!existing) {
489
+ throw new Error(`Schedule "${scheduleId}" not found`);
490
+ }
491
+ const updated: ScheduleRecord = {
492
+ ...existing,
493
+ enabled,
494
+ updatedAt: now,
495
+ nextRunAt: enabled
496
+ ? getNextCronTime(existing.cronPattern, now)
497
+ : undefined,
498
+ };
499
+ this.schedules.set(scheduleId, updated);
500
+ return { ...updated };
501
+ }
502
+
503
+ delete(scheduleId: string): void {
504
+ if (!this.schedules.has(scheduleId)) {
505
+ throw new Error(`Schedule "${scheduleId}" not found`);
506
+ }
507
+ this.schedules.delete(scheduleId);
508
+ }
509
+
510
+ recordTriggered(scheduleId: string, now: number): ScheduleRecord {
511
+ const existing = this.schedules.get(scheduleId);
512
+ if (!existing) {
513
+ throw new Error(`Schedule "${scheduleId}" not found`);
514
+ }
515
+ const updated: ScheduleRecord = {
516
+ ...existing,
517
+ lastRunAt: now,
518
+ updatedAt: now,
519
+ nextRunAt: getNextCronTime(existing.cronPattern, now),
520
+ };
521
+ this.schedules.set(scheduleId, updated);
522
+ return { ...updated };
523
+ }
524
+
525
+ getDueSchedules(now: number): ScheduleRecord[] {
526
+ return [...this.schedules.values()]
527
+ .filter(
528
+ (record) =>
529
+ record.enabled &&
530
+ record.nextRunAt !== undefined &&
531
+ record.nextRunAt <= now,
532
+ )
533
+ .map((record) => ({ ...record }));
534
+ }
535
+
536
+ startExecution(
537
+ scheduleId: string,
538
+ sessionId: string,
539
+ now: number,
540
+ ): ScheduleExecutionRecord {
541
+ const record: ScheduleExecutionRecord = {
542
+ executionId: createHubId("sexec"),
543
+ scheduleId,
544
+ sessionId,
545
+ triggeredAt: now,
546
+ startedAt: now,
547
+ status: "running",
548
+ };
549
+ this.executions.set(record.executionId, record);
550
+ return { ...record };
551
+ }
552
+
553
+ completeExecution(
554
+ executionId: string,
555
+ status: Exclude<ScheduleExecutionStatus, "running">,
556
+ now: number,
557
+ errorMessage?: string,
558
+ ): ScheduleExecutionRecord {
559
+ const existing = this.executions.get(executionId);
560
+ if (!existing) {
561
+ throw new Error(`Execution "${executionId}" not found`);
562
+ }
563
+ const updated: ScheduleExecutionRecord = {
564
+ ...existing,
565
+ status,
566
+ endedAt: now,
567
+ ...(errorMessage ? { errorMessage } : {}),
568
+ };
569
+ this.executions.set(executionId, updated);
570
+ return { ...updated };
571
+ }
572
+
573
+ listExecutions(scheduleId?: string): ScheduleExecutionRecord[] {
574
+ const records = [...this.executions.values()];
575
+ return (
576
+ scheduleId
577
+ ? records.filter((record) => record.scheduleId === scheduleId)
578
+ : records
579
+ )
580
+ .map((record) => ({ ...record }))
581
+ .sort((left, right) => right.triggeredAt - left.triggeredAt);
582
+ }
583
+ }
584
+
585
+ export interface SchedulerTriggerContext {
586
+ scheduleId: string;
587
+ sessionId: string;
588
+ executionId: string;
589
+ }
590
+
591
+ export interface SchedulerCallbacks {
592
+ onTrigger(record: ScheduleRecord): Promise<{ sessionId: string }>;
593
+ onExecutionCompleted(
594
+ context: SchedulerTriggerContext,
595
+ execution: ScheduleExecutionRecord,
596
+ ): void;
597
+ onExecutionFailed(
598
+ context: SchedulerTriggerContext,
599
+ execution: ScheduleExecutionRecord,
600
+ error: unknown,
601
+ ): void;
602
+ onScheduleUpdated(record: ScheduleRecord): void;
603
+ onPersist(): Promise<void>;
604
+ }
605
+
606
+ export interface HubSchedulerOptions {
607
+ store: ScheduleStore;
608
+ callbacks: SchedulerCallbacks;
609
+ now?: () => number;
610
+ pollIntervalMs?: number;
611
+ }
612
+
613
+ const DEFAULT_POLL_INTERVAL_MS = 15_000;
614
+
615
+ export class HubScheduler {
616
+ private readonly store: ScheduleStore;
617
+ private readonly callbacks: SchedulerCallbacks;
618
+ private readonly now: () => number;
619
+ private readonly pollIntervalMs: number;
620
+ private timer: ReturnType<typeof setInterval> | undefined;
621
+ private readonly runningExecutions = new Map<
622
+ string,
623
+ SchedulerTriggerContext
624
+ >();
625
+
626
+ constructor(options: HubSchedulerOptions) {
627
+ this.store = options.store;
628
+ this.callbacks = options.callbacks;
629
+ this.now = options.now ?? (() => Date.now());
630
+ this.pollIntervalMs = Math.max(
631
+ 1_000,
632
+ Math.floor(options.pollIntervalMs ?? DEFAULT_POLL_INTERVAL_MS),
633
+ );
634
+ }
635
+
636
+ start(): void {
637
+ if (this.timer !== undefined) {
638
+ return;
639
+ }
640
+ this.timer = setInterval(() => {
641
+ void this.poll();
642
+ }, this.pollIntervalMs);
643
+ void this.poll();
644
+ }
645
+
646
+ stop(): void {
647
+ if (this.timer === undefined) {
648
+ return;
649
+ }
650
+ clearInterval(this.timer);
651
+ this.timer = undefined;
652
+ }
653
+
654
+ async triggerNow(scheduleId: string): Promise<ScheduleExecutionRecord> {
655
+ const record = this.store.get(scheduleId);
656
+ if (!record) {
657
+ throw new Error(`Schedule "${scheduleId}" not found`);
658
+ }
659
+ return await this.executeSchedule(record);
660
+ }
661
+
662
+ async notifySessionCompleted(
663
+ sessionId: string,
664
+ failed: boolean,
665
+ errorMessage?: string,
666
+ ): Promise<void> {
667
+ const entry = [...this.runningExecutions.entries()].find(
668
+ ([, context]) => context.sessionId === sessionId,
669
+ );
670
+ if (!entry) {
671
+ return;
672
+ }
673
+ const [executionId, context] = entry;
674
+ this.runningExecutions.delete(executionId);
675
+ const status: Exclude<ScheduleExecutionStatus, "running"> = failed
676
+ ? "failed"
677
+ : "completed";
678
+ const execution = this.store.completeExecution(
679
+ executionId,
680
+ status,
681
+ this.now(),
682
+ errorMessage,
683
+ );
684
+ if (failed) {
685
+ this.callbacks.onExecutionFailed(
686
+ context,
687
+ execution,
688
+ new Error(errorMessage),
689
+ );
690
+ } else {
691
+ this.callbacks.onExecutionCompleted(context, execution);
692
+ }
693
+ await this.callbacks.onPersist();
694
+ }
695
+
696
+ get scheduleStore(): ScheduleStore {
697
+ return this.store;
698
+ }
699
+
700
+ private async poll(): Promise<void> {
701
+ const due = this.store.getDueSchedules(this.now());
702
+ for (const record of due) {
703
+ const alreadyRunning = [...this.runningExecutions.values()].some(
704
+ (context) => context.scheduleId === record.scheduleId,
705
+ );
706
+ if (alreadyRunning) {
707
+ continue;
708
+ }
709
+ void this.executeSchedule(record);
710
+ }
711
+ }
712
+
713
+ private async executeSchedule(
714
+ record: ScheduleRecord,
715
+ ): Promise<ScheduleExecutionRecord> {
716
+ const startedAt = this.now();
717
+
718
+ let sessionId: string;
719
+ try {
720
+ const result = await this.callbacks.onTrigger(record);
721
+ sessionId = result.sessionId;
722
+ } catch (error) {
723
+ const execution = this.store.startExecution(
724
+ record.scheduleId,
725
+ "",
726
+ startedAt,
727
+ );
728
+ const failed = this.store.completeExecution(
729
+ execution.executionId,
730
+ "failed",
731
+ this.now(),
732
+ error instanceof Error ? error.message : String(error),
733
+ );
734
+ const updatedSchedule = this.store.recordTriggered(
735
+ record.scheduleId,
736
+ startedAt,
737
+ );
738
+ this.callbacks.onScheduleUpdated(updatedSchedule);
739
+ this.callbacks.onExecutionFailed(
740
+ {
741
+ scheduleId: record.scheduleId,
742
+ sessionId: "",
743
+ executionId: execution.executionId,
744
+ },
745
+ failed,
746
+ error,
747
+ );
748
+ await this.callbacks.onPersist();
749
+ return failed;
750
+ }
751
+
752
+ const execution = this.store.startExecution(
753
+ record.scheduleId,
754
+ sessionId,
755
+ startedAt,
756
+ );
757
+ this.runningExecutions.set(execution.executionId, {
758
+ scheduleId: record.scheduleId,
759
+ sessionId,
760
+ executionId: execution.executionId,
761
+ });
762
+
763
+ const updatedSchedule = this.store.recordTriggered(
764
+ record.scheduleId,
765
+ startedAt,
766
+ );
767
+ this.callbacks.onScheduleUpdated(updatedSchedule);
768
+ await this.callbacks.onPersist();
769
+
770
+ return execution;
771
+ }
772
+ }