@lumenflow/cli 3.1.2 → 3.2.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 (389) hide show
  1. package/README.md +36 -35
  2. package/dist/agent-issues-query.js +13 -8
  3. package/dist/agent-log-issue.js +15 -4
  4. package/dist/agent-session-end.js +15 -4
  5. package/dist/agent-session.js +18 -6
  6. package/dist/backlog-prune.js +1 -1
  7. package/dist/commands/integrate.js +32 -18
  8. package/dist/config-get.js +27 -15
  9. package/dist/config-set.js +104 -37
  10. package/dist/delegation-list.js +1 -1
  11. package/dist/doctor.js +19 -13
  12. package/dist/file-delete.js +1 -1
  13. package/dist/file-edit.js +1 -1
  14. package/dist/file-read.js +1 -1
  15. package/dist/file-write.js +1 -1
  16. package/dist/flow-bottlenecks.js +1 -1
  17. package/dist/flow-report.js +10 -9
  18. package/dist/gates.js +3 -2
  19. package/dist/git-branch.js +1 -1
  20. package/dist/git-diff.js +1 -1
  21. package/dist/git-log.js +1 -1
  22. package/dist/init.js +238 -42
  23. package/dist/initiative-add-wu.js +1 -1
  24. package/dist/initiative-bulk-assign-wus.js +1 -1
  25. package/dist/initiative-create.js +2 -2
  26. package/dist/initiative-edit.js +1 -1
  27. package/dist/initiative-list.js +1 -1
  28. package/dist/initiative-plan.js +1 -3
  29. package/dist/initiative-status.js +47 -6
  30. package/dist/lane-edit.js +19 -10
  31. package/dist/lane-health.js +13 -24
  32. package/dist/lane-lock.js +4 -5
  33. package/dist/lane-setup.js +5 -5
  34. package/dist/lane-status.js +4 -5
  35. package/dist/lane-suggest.js +9 -7
  36. package/dist/lane-validate.js +4 -5
  37. package/dist/lumenflow-upgrade.js +17 -11
  38. package/dist/mem-checkpoint.js +1 -1
  39. package/dist/mem-cleanup.js +6 -23
  40. package/dist/mem-context.js +1 -1
  41. package/dist/mem-create.js +1 -1
  42. package/dist/mem-delete.js +1 -1
  43. package/dist/mem-export.js +1 -1
  44. package/dist/mem-inbox.js +1 -1
  45. package/dist/mem-init.js +1 -1
  46. package/dist/mem-ready.js +1 -1
  47. package/dist/mem-recover.js +1 -1
  48. package/dist/mem-signal.js +1 -1
  49. package/dist/mem-start.js +1 -1
  50. package/dist/mem-summarize.js +8 -7
  51. package/dist/mem-triage.js +7 -5
  52. package/dist/metrics-cli.js +1 -1
  53. package/dist/metrics-snapshot.js +1 -1
  54. package/dist/onboard.js +295 -120
  55. package/dist/orchestrate-init-status.js +12 -7
  56. package/dist/orchestrate-initiative.js +23 -12
  57. package/dist/orchestrate-monitor.js +20 -8
  58. package/dist/pack-scaffold.js +1 -1
  59. package/dist/plan-create.js +1 -3
  60. package/dist/plan-edit.js +1 -3
  61. package/dist/plan-link.js +1 -3
  62. package/dist/plan-promote.js +1 -3
  63. package/dist/release.js +27 -10
  64. package/dist/signal-cleanup.js +4 -18
  65. package/dist/state-bootstrap.js +11 -8
  66. package/dist/state-cleanup.js +5 -19
  67. package/dist/state-doctor.js +213 -9
  68. package/dist/task-claim.js +1 -1
  69. package/dist/validate.js +1 -1
  70. package/dist/workspace-init.js +61 -61
  71. package/dist/wu-block.js +1 -1
  72. package/dist/wu-brief.js +1 -1
  73. package/dist/wu-claim.js +1 -1
  74. package/dist/wu-cleanup.js +1 -1
  75. package/dist/wu-create.js +3 -3
  76. package/dist/wu-delegate.js +1 -1
  77. package/dist/wu-deps.js +1 -1
  78. package/dist/wu-done.js +66 -34
  79. package/dist/wu-edit.js +1 -1
  80. package/dist/wu-infer-lane.js +1 -1
  81. package/dist/wu-preflight.js +1 -1
  82. package/dist/wu-prep.js +1 -1
  83. package/dist/wu-proto.js +1 -1
  84. package/dist/wu-prune.js +1 -1
  85. package/dist/wu-recover.js +1 -1
  86. package/dist/wu-release.js +1 -1
  87. package/dist/wu-repair.js +1 -1
  88. package/dist/wu-sandbox.js +40 -27
  89. package/dist/wu-status.js +1 -1
  90. package/dist/wu-unblock.js +1 -1
  91. package/dist/wu-unlock-lane.js +1 -1
  92. package/dist/wu-validate.js +1 -1
  93. package/package.json +13 -8
  94. package/packs/software-delivery/constants.ts +10 -0
  95. package/packs/software-delivery/extensions.ts +140 -0
  96. package/packs/software-delivery/gate-policies.ts +134 -0
  97. package/packs/software-delivery/index.ts +8 -0
  98. package/packs/software-delivery/manifest-schema.ts +236 -0
  99. package/packs/software-delivery/manifest.ts +417 -0
  100. package/packs/software-delivery/manifest.yaml +711 -0
  101. package/packs/software-delivery/pack-registration.ts +113 -0
  102. package/packs/software-delivery/tool-impl/agent-tools.ts +263 -0
  103. package/packs/software-delivery/tool-impl/delegation-tools.ts +66 -0
  104. package/packs/software-delivery/tool-impl/flow-metrics-tools.ts +219 -0
  105. package/packs/software-delivery/tool-impl/git-runner.ts +113 -0
  106. package/packs/software-delivery/tool-impl/git-tools.ts +316 -0
  107. package/packs/software-delivery/tool-impl/index.ts +15 -0
  108. package/packs/software-delivery/tool-impl/initiative-orchestration-tools.ts +720 -0
  109. package/packs/software-delivery/tool-impl/lane-lock.ts +246 -0
  110. package/packs/software-delivery/tool-impl/memory-tools.ts +415 -0
  111. package/packs/software-delivery/tool-impl/pending-runtime-tools.ts +21 -0
  112. package/packs/software-delivery/tool-impl/runtime-cli-adapter.ts +328 -0
  113. package/packs/software-delivery/tool-impl/runtime-native-tools.ts +687 -0
  114. package/packs/software-delivery/tool-impl/worker-loader.ts +52 -0
  115. package/packs/software-delivery/tool-impl/worktree-tools.ts +46 -0
  116. package/packs/software-delivery/tool-impl/wu-lifecycle-tools.ts +759 -0
  117. package/packs/software-delivery/tools/delegation-tools.ts +23 -0
  118. package/packs/software-delivery/tools/git-tools.ts +55 -0
  119. package/packs/software-delivery/tools/index.ts +8 -0
  120. package/packs/software-delivery/tools/lane-lock-tool.ts +37 -0
  121. package/packs/software-delivery/tools/types.ts +71 -0
  122. package/packs/software-delivery/tools/worktree-tools.ts +49 -0
  123. package/templates/core/LUMENFLOW.md.template +3 -3
  124. package/templates/core/ai/onboarding/agent-invocation-guide.md.template +1 -1
  125. package/templates/core/ai/onboarding/lumenflow-force-usage.md.template +1 -1
  126. package/templates/core/ai/onboarding/quick-ref-commands.md.template +5 -5
  127. package/templates/core/ai/onboarding/starting-prompt.md.template +3 -3
  128. package/templates/core/ai/onboarding/vendor-support.md.template +1 -1
  129. package/templates/core/ai/onboarding/wu-create-checklist.md.template +1 -1
  130. package/dist/agent-issues-query.js.map +0 -1
  131. package/dist/agent-log-issue.js.map +0 -1
  132. package/dist/agent-session-end.js.map +0 -1
  133. package/dist/agent-session.js.map +0 -1
  134. package/dist/backlog-prune.js.map +0 -1
  135. package/dist/cli-entry-point.js +0 -149
  136. package/dist/cli-entry-point.js.map +0 -1
  137. package/dist/commands/integrate.js.map +0 -1
  138. package/dist/commands.js.map +0 -1
  139. package/dist/config-get.js.map +0 -1
  140. package/dist/config-set.js.map +0 -1
  141. package/dist/delegation-list.js.map +0 -1
  142. package/dist/deps-add.js +0 -259
  143. package/dist/deps-add.js.map +0 -1
  144. package/dist/deps-remove.js +0 -105
  145. package/dist/deps-remove.js.map +0 -1
  146. package/dist/docs-sync.js.map +0 -1
  147. package/dist/doctor.js.map +0 -1
  148. package/dist/file-delete.js.map +0 -1
  149. package/dist/file-edit.js.map +0 -1
  150. package/dist/file-read.js.map +0 -1
  151. package/dist/file-write.js.map +0 -1
  152. package/dist/flow-bottlenecks.js.map +0 -1
  153. package/dist/flow-report.js.map +0 -1
  154. package/dist/formatters.js +0 -151
  155. package/dist/formatters.js.map +0 -1
  156. package/dist/gate-defaults.js +0 -131
  157. package/dist/gate-defaults.js.map +0 -1
  158. package/dist/gate-registry.js +0 -73
  159. package/dist/gate-registry.js.map +0 -1
  160. package/dist/gates-graceful-degradation.js +0 -153
  161. package/dist/gates-graceful-degradation.js.map +0 -1
  162. package/dist/gates-plan-resolvers.js +0 -152
  163. package/dist/gates-plan-resolvers.js.map +0 -1
  164. package/dist/gates-runners.js +0 -509
  165. package/dist/gates-runners.js.map +0 -1
  166. package/dist/gates-types.js +0 -4
  167. package/dist/gates-types.js.map +0 -1
  168. package/dist/gates-utils.js +0 -323
  169. package/dist/gates-utils.js.map +0 -1
  170. package/dist/gates.js.map +0 -1
  171. package/dist/git-branch.js.map +0 -1
  172. package/dist/git-diff.js.map +0 -1
  173. package/dist/git-log.js.map +0 -1
  174. package/dist/git-status.js.map +0 -1
  175. package/dist/guard-locked.js +0 -172
  176. package/dist/guard-locked.js.map +0 -1
  177. package/dist/guard-main-branch.js +0 -217
  178. package/dist/guard-main-branch.js.map +0 -1
  179. package/dist/guard-worktree-commit.js +0 -163
  180. package/dist/guard-worktree-commit.js.map +0 -1
  181. package/dist/hooks/auto-checkpoint-utils.js +0 -54
  182. package/dist/hooks/auto-checkpoint-utils.js.map +0 -1
  183. package/dist/hooks/enforcement-checks.js +0 -399
  184. package/dist/hooks/enforcement-checks.js.map +0 -1
  185. package/dist/hooks/enforcement-generator.js +0 -139
  186. package/dist/hooks/enforcement-generator.js.map +0 -1
  187. package/dist/hooks/enforcement-sync.js +0 -385
  188. package/dist/hooks/enforcement-sync.js.map +0 -1
  189. package/dist/hooks/generators/auto-checkpoint.js +0 -125
  190. package/dist/hooks/generators/auto-checkpoint.js.map +0 -1
  191. package/dist/hooks/generators/enforce-worktree.js +0 -190
  192. package/dist/hooks/generators/enforce-worktree.js.map +0 -1
  193. package/dist/hooks/generators/index.js +0 -18
  194. package/dist/hooks/generators/index.js.map +0 -1
  195. package/dist/hooks/generators/pre-compact-checkpoint.js +0 -136
  196. package/dist/hooks/generators/pre-compact-checkpoint.js.map +0 -1
  197. package/dist/hooks/generators/require-wu.js +0 -117
  198. package/dist/hooks/generators/require-wu.js.map +0 -1
  199. package/dist/hooks/generators/session-start-recovery.js +0 -103
  200. package/dist/hooks/generators/session-start-recovery.js.map +0 -1
  201. package/dist/hooks/generators/signal-utils.js +0 -54
  202. package/dist/hooks/generators/signal-utils.js.map +0 -1
  203. package/dist/hooks/generators/warn-incomplete.js +0 -67
  204. package/dist/hooks/generators/warn-incomplete.js.map +0 -1
  205. package/dist/hooks/index.js +0 -10
  206. package/dist/hooks/index.js.map +0 -1
  207. package/dist/index.js.map +0 -1
  208. package/dist/init-detection.js +0 -232
  209. package/dist/init-detection.js.map +0 -1
  210. package/dist/init-lane-validation.js +0 -143
  211. package/dist/init-lane-validation.js.map +0 -1
  212. package/dist/init-scaffolding.js +0 -158
  213. package/dist/init-scaffolding.js.map +0 -1
  214. package/dist/init-templates.js +0 -1982
  215. package/dist/init-templates.js.map +0 -1
  216. package/dist/init.js.map +0 -1
  217. package/dist/initiative-add-wu.js.map +0 -1
  218. package/dist/initiative-bulk-assign-wus.js.map +0 -1
  219. package/dist/initiative-create.js.map +0 -1
  220. package/dist/initiative-edit.js.map +0 -1
  221. package/dist/initiative-list.js.map +0 -1
  222. package/dist/initiative-plan.js.map +0 -1
  223. package/dist/initiative-remove-wu.js.map +0 -1
  224. package/dist/initiative-status.js.map +0 -1
  225. package/dist/lane-edit.js.map +0 -1
  226. package/dist/lane-health.js.map +0 -1
  227. package/dist/lane-lifecycle-process.js +0 -366
  228. package/dist/lane-lifecycle-process.js.map +0 -1
  229. package/dist/lane-lock.js.map +0 -1
  230. package/dist/lane-setup.js.map +0 -1
  231. package/dist/lane-status.js.map +0 -1
  232. package/dist/lane-suggest.js.map +0 -1
  233. package/dist/lane-validate.js.map +0 -1
  234. package/dist/lifecycle-regression-harness.js +0 -181
  235. package/dist/lifecycle-regression-harness.js.map +0 -1
  236. package/dist/lumenflow-upgrade.js.map +0 -1
  237. package/dist/mem-checkpoint.js.map +0 -1
  238. package/dist/mem-cleanup.js.map +0 -1
  239. package/dist/mem-context.js.map +0 -1
  240. package/dist/mem-create.js.map +0 -1
  241. package/dist/mem-delete.js.map +0 -1
  242. package/dist/mem-export.js.map +0 -1
  243. package/dist/mem-inbox.js.map +0 -1
  244. package/dist/mem-index.js +0 -214
  245. package/dist/mem-index.js.map +0 -1
  246. package/dist/mem-init.js.map +0 -1
  247. package/dist/mem-profile.js +0 -210
  248. package/dist/mem-profile.js.map +0 -1
  249. package/dist/mem-promote.js +0 -257
  250. package/dist/mem-promote.js.map +0 -1
  251. package/dist/mem-ready.js.map +0 -1
  252. package/dist/mem-recover.js.map +0 -1
  253. package/dist/mem-signal.js.map +0 -1
  254. package/dist/mem-start.js.map +0 -1
  255. package/dist/mem-summarize.js.map +0 -1
  256. package/dist/mem-triage.js.map +0 -1
  257. package/dist/merge-block.js +0 -225
  258. package/dist/merge-block.js.map +0 -1
  259. package/dist/metrics-cli.js.map +0 -1
  260. package/dist/metrics-snapshot.js.map +0 -1
  261. package/dist/onboard.js.map +0 -1
  262. package/dist/onboarding-smoke-test.js +0 -418
  263. package/dist/onboarding-smoke-test.js.map +0 -1
  264. package/dist/orchestrate-init-status.js.map +0 -1
  265. package/dist/orchestrate-initiative.js.map +0 -1
  266. package/dist/orchestrate-monitor.js.map +0 -1
  267. package/dist/pack-author.js.map +0 -1
  268. package/dist/pack-hash.js.map +0 -1
  269. package/dist/pack-install.js.map +0 -1
  270. package/dist/pack-publish.js.map +0 -1
  271. package/dist/pack-scaffold.js.map +0 -1
  272. package/dist/pack-search.js.map +0 -1
  273. package/dist/pack-validate.js.map +0 -1
  274. package/dist/plan-create.js.map +0 -1
  275. package/dist/plan-edit.js.map +0 -1
  276. package/dist/plan-link.js.map +0 -1
  277. package/dist/plan-promote.js.map +0 -1
  278. package/dist/public-manifest.js +0 -920
  279. package/dist/public-manifest.js.map +0 -1
  280. package/dist/release.js.map +0 -1
  281. package/dist/rotate-progress.js +0 -253
  282. package/dist/rotate-progress.js.map +0 -1
  283. package/dist/session-coordinator.js +0 -303
  284. package/dist/session-coordinator.js.map +0 -1
  285. package/dist/shared-validators.js +0 -81
  286. package/dist/shared-validators.js.map +0 -1
  287. package/dist/signal-cleanup.js.map +0 -1
  288. package/dist/state-bootstrap.js.map +0 -1
  289. package/dist/state-cleanup.js.map +0 -1
  290. package/dist/state-doctor-fix.js +0 -226
  291. package/dist/state-doctor-fix.js.map +0 -1
  292. package/dist/state-doctor-stamps.js +0 -23
  293. package/dist/state-doctor-stamps.js.map +0 -1
  294. package/dist/state-doctor.js.map +0 -1
  295. package/dist/strict-progress.js +0 -255
  296. package/dist/strict-progress.js.map +0 -1
  297. package/dist/sync-templates.js.map +0 -1
  298. package/dist/task-claim.js.map +0 -1
  299. package/dist/trace-gen.js +0 -401
  300. package/dist/trace-gen.js.map +0 -1
  301. package/dist/validate-agent-skills.js +0 -223
  302. package/dist/validate-agent-skills.js.map +0 -1
  303. package/dist/validate-agent-sync.js +0 -151
  304. package/dist/validate-agent-sync.js.map +0 -1
  305. package/dist/validate-backlog-sync.js +0 -77
  306. package/dist/validate-backlog-sync.js.map +0 -1
  307. package/dist/validate-skills-spec.js +0 -211
  308. package/dist/validate-skills-spec.js.map +0 -1
  309. package/dist/validate.js.map +0 -1
  310. package/dist/validator-defaults.js +0 -107
  311. package/dist/validator-defaults.js.map +0 -1
  312. package/dist/validator-registry.js +0 -71
  313. package/dist/validator-registry.js.map +0 -1
  314. package/dist/workspace-init.js.map +0 -1
  315. package/dist/wu-block.js.map +0 -1
  316. package/dist/wu-brief.js.map +0 -1
  317. package/dist/wu-claim-branch.js +0 -123
  318. package/dist/wu-claim-branch.js.map +0 -1
  319. package/dist/wu-claim-cloud.js +0 -79
  320. package/dist/wu-claim-cloud.js.map +0 -1
  321. package/dist/wu-claim-mode.js +0 -82
  322. package/dist/wu-claim-mode.js.map +0 -1
  323. package/dist/wu-claim-output.js +0 -85
  324. package/dist/wu-claim-output.js.map +0 -1
  325. package/dist/wu-claim-repair-guidance.js +0 -12
  326. package/dist/wu-claim-repair-guidance.js.map +0 -1
  327. package/dist/wu-claim-resume-handler.js +0 -87
  328. package/dist/wu-claim-resume-handler.js.map +0 -1
  329. package/dist/wu-claim-state.js +0 -581
  330. package/dist/wu-claim-state.js.map +0 -1
  331. package/dist/wu-claim-validation.js +0 -457
  332. package/dist/wu-claim-validation.js.map +0 -1
  333. package/dist/wu-claim-worktree.js +0 -223
  334. package/dist/wu-claim-worktree.js.map +0 -1
  335. package/dist/wu-claim.js.map +0 -1
  336. package/dist/wu-cleanup-cloud.js +0 -78
  337. package/dist/wu-cleanup-cloud.js.map +0 -1
  338. package/dist/wu-cleanup.js.map +0 -1
  339. package/dist/wu-code-path-coverage.js +0 -83
  340. package/dist/wu-code-path-coverage.js.map +0 -1
  341. package/dist/wu-create-cloud.js +0 -30
  342. package/dist/wu-create-cloud.js.map +0 -1
  343. package/dist/wu-create-content.js +0 -264
  344. package/dist/wu-create-content.js.map +0 -1
  345. package/dist/wu-create-readiness.js +0 -59
  346. package/dist/wu-create-readiness.js.map +0 -1
  347. package/dist/wu-create-validation.js +0 -128
  348. package/dist/wu-create-validation.js.map +0 -1
  349. package/dist/wu-create.js.map +0 -1
  350. package/dist/wu-delegate.js.map +0 -1
  351. package/dist/wu-delete.js.map +0 -1
  352. package/dist/wu-deps.js.map +0 -1
  353. package/dist/wu-done-auto-cleanup.js +0 -203
  354. package/dist/wu-done-auto-cleanup.js.map +0 -1
  355. package/dist/wu-done-check.js +0 -38
  356. package/dist/wu-done-check.js.map +0 -1
  357. package/dist/wu-done-cloud.js +0 -48
  358. package/dist/wu-done-cloud.js.map +0 -1
  359. package/dist/wu-done-decay.js +0 -86
  360. package/dist/wu-done-decay.js.map +0 -1
  361. package/dist/wu-done.js.map +0 -1
  362. package/dist/wu-edit-operations.js +0 -399
  363. package/dist/wu-edit-operations.js.map +0 -1
  364. package/dist/wu-edit-validators.js +0 -282
  365. package/dist/wu-edit-validators.js.map +0 -1
  366. package/dist/wu-edit.js.map +0 -1
  367. package/dist/wu-infer-lane.js.map +0 -1
  368. package/dist/wu-preflight.js.map +0 -1
  369. package/dist/wu-prep.js.map +0 -1
  370. package/dist/wu-proto.js.map +0 -1
  371. package/dist/wu-prune.js.map +0 -1
  372. package/dist/wu-recover.js.map +0 -1
  373. package/dist/wu-release.js.map +0 -1
  374. package/dist/wu-repair.js.map +0 -1
  375. package/dist/wu-sandbox.js.map +0 -1
  376. package/dist/wu-spawn-completion.js +0 -33
  377. package/dist/wu-spawn-completion.js.map +0 -1
  378. package/dist/wu-spawn-prompt-builders.js +0 -1197
  379. package/dist/wu-spawn-prompt-builders.js.map +0 -1
  380. package/dist/wu-spawn-strategy-resolver.js +0 -322
  381. package/dist/wu-spawn-strategy-resolver.js.map +0 -1
  382. package/dist/wu-spawn.js +0 -59
  383. package/dist/wu-spawn.js.map +0 -1
  384. package/dist/wu-state-cloud.js +0 -41
  385. package/dist/wu-state-cloud.js.map +0 -1
  386. package/dist/wu-status.js.map +0 -1
  387. package/dist/wu-unblock.js.map +0 -1
  388. package/dist/wu-unlock-lane.js.map +0 -1
  389. package/dist/wu-validate.js.map +0 -1
@@ -0,0 +1,246 @@
1
+ // Copyright (c) 2026 Hellmai Ltd
2
+ // SPDX-License-Identifier: AGPL-3.0-only
3
+
4
+ /* eslint-disable security/detect-non-literal-fs-filename */
5
+ import { mkdir, readFile, unlink, writeFile } from 'node:fs/promises';
6
+ import path from 'node:path';
7
+ import { UTF8_ENCODING } from '../constants.js';
8
+
9
+ export interface LaneLockMetadata {
10
+ lane: string;
11
+ wuId: string;
12
+ owner: string;
13
+ timestamp: string;
14
+ }
15
+
16
+ export interface AcquireLaneLockInput {
17
+ lane: string;
18
+ wuId: string;
19
+ owner: string;
20
+ locksDir: string;
21
+ staleAfterMs?: number;
22
+ }
23
+
24
+ export interface AcquireLaneLockResult {
25
+ acquired: boolean;
26
+ is_stale: boolean;
27
+ lock_path: string;
28
+ }
29
+
30
+ export interface ReleaseLaneLockInput {
31
+ lane: string;
32
+ owner: string;
33
+ locksDir: string;
34
+ }
35
+
36
+ export interface ReleaseLaneLockResult {
37
+ released: boolean;
38
+ lock_path: string;
39
+ }
40
+
41
+ function laneToLockFileName(lane: string): string {
42
+ const chars: string[] = [];
43
+ for (const char of lane.toLowerCase()) {
44
+ if ((char >= 'a' && char <= 'z') || (char >= '0' && char <= '9')) {
45
+ chars.push(char);
46
+ continue;
47
+ }
48
+ if (chars[chars.length - 1] !== '-') {
49
+ chars.push('-');
50
+ }
51
+ }
52
+ let normalized = chars.join('');
53
+ while (normalized.startsWith('-')) {
54
+ normalized = normalized.slice(1);
55
+ }
56
+ while (normalized.endsWith('-')) {
57
+ normalized = normalized.slice(0, -1);
58
+ }
59
+ return `${normalized || 'lane'}.lock`;
60
+ }
61
+
62
+ function lockPathFor(input: { lane: string; locksDir: string }): string {
63
+ return path.join(input.locksDir, laneToLockFileName(input.lane));
64
+ }
65
+
66
+ function isStale(metadata: LaneLockMetadata, staleAfterMs: number): boolean {
67
+ const lockTime = new Date(metadata.timestamp).getTime();
68
+ return Number.isFinite(lockTime) && Date.now() - lockTime > staleAfterMs;
69
+ }
70
+
71
+ function staleTakeoverMarkerPath(lockPath: string): string {
72
+ return `${lockPath}.takeover`;
73
+ }
74
+
75
+ async function acquireTakeoverMarker(
76
+ markerPath: string,
77
+ metadata: LaneLockMetadata,
78
+ ): Promise<boolean> {
79
+ try {
80
+ await writeFile(markerPath, JSON.stringify(metadata), { encoding: UTF8_ENCODING, flag: 'wx' });
81
+ return true;
82
+ } catch (error) {
83
+ if ((error as NodeJS.ErrnoException).code === 'EEXIST') {
84
+ return false;
85
+ }
86
+ throw error;
87
+ }
88
+ }
89
+
90
+ type LaneLockWarn = (message: string) => void;
91
+
92
+ function writeWarning(message: string): void {
93
+ process.stderr.write(`${message}\n`);
94
+ }
95
+
96
+ export async function cleanupTakeoverMarker(
97
+ markerPath: string,
98
+ options: {
99
+ unlinkFile?: (targetPath: string) => Promise<void>;
100
+ warn?: LaneLockWarn;
101
+ } = {},
102
+ ): Promise<void> {
103
+ const unlinkFile = options.unlinkFile ?? unlink;
104
+ const warn: LaneLockWarn = options.warn ?? writeWarning;
105
+
106
+ try {
107
+ await unlinkFile(markerPath);
108
+ } catch (error) {
109
+ const nodeError = error as NodeJS.ErrnoException;
110
+ if (nodeError.code === 'ENOENT') {
111
+ return;
112
+ }
113
+ const message = error instanceof Error ? error.message : String(error);
114
+ warn(`[lane-lock] failed to cleanup takeover marker at ${markerPath}: ${message}`);
115
+ }
116
+ }
117
+
118
+ export async function readLaneLockMetadata(lockPath: string): Promise<LaneLockMetadata | null> {
119
+ try {
120
+ const raw = await readFile(lockPath, UTF8_ENCODING);
121
+ const parsed = JSON.parse(raw) as LaneLockMetadata;
122
+ if (
123
+ typeof parsed.lane !== 'string' ||
124
+ typeof parsed.wuId !== 'string' ||
125
+ typeof parsed.owner !== 'string' ||
126
+ typeof parsed.timestamp !== 'string'
127
+ ) {
128
+ return null;
129
+ }
130
+ return parsed;
131
+ } catch {
132
+ return null;
133
+ }
134
+ }
135
+
136
+ export async function acquireLaneLockTool(
137
+ input: AcquireLaneLockInput,
138
+ ): Promise<AcquireLaneLockResult> {
139
+ await mkdir(input.locksDir, { recursive: true });
140
+ const lockPath = lockPathFor(input);
141
+ const staleAfterMs = input.staleAfterMs ?? 2 * 60 * 60 * 1000;
142
+ const nextMetadata: LaneLockMetadata = {
143
+ lane: input.lane,
144
+ wuId: input.wuId,
145
+ owner: input.owner,
146
+ timestamp: new Date().toISOString(),
147
+ };
148
+
149
+ try {
150
+ await writeFile(lockPath, JSON.stringify(nextMetadata), {
151
+ encoding: UTF8_ENCODING,
152
+ flag: 'wx',
153
+ });
154
+ return {
155
+ acquired: true,
156
+ is_stale: false,
157
+ lock_path: lockPath,
158
+ };
159
+ } catch (error) {
160
+ const isExists = (error as NodeJS.ErrnoException).code === 'EEXIST';
161
+ if (!isExists) {
162
+ throw error;
163
+ }
164
+
165
+ const existing = await readLaneLockMetadata(lockPath);
166
+ if (existing && isStale(existing, staleAfterMs)) {
167
+ const takeoverMarkerPath = staleTakeoverMarkerPath(lockPath);
168
+ const wonTakeoverRace = await acquireTakeoverMarker(takeoverMarkerPath, nextMetadata);
169
+ if (!wonTakeoverRace) {
170
+ return {
171
+ acquired: false,
172
+ is_stale: false,
173
+ lock_path: lockPath,
174
+ };
175
+ }
176
+
177
+ try {
178
+ const current = await readLaneLockMetadata(lockPath);
179
+ if (!current || !isStale(current, staleAfterMs)) {
180
+ return {
181
+ acquired: false,
182
+ is_stale: false,
183
+ lock_path: lockPath,
184
+ };
185
+ }
186
+
187
+ await unlink(lockPath).catch((unlinkError) => {
188
+ const nodeError = unlinkError as NodeJS.ErrnoException;
189
+ if (nodeError.code === 'ENOENT') {
190
+ return;
191
+ }
192
+ throw unlinkError;
193
+ });
194
+
195
+ try {
196
+ await writeFile(lockPath, JSON.stringify(nextMetadata), {
197
+ encoding: UTF8_ENCODING,
198
+ flag: 'wx',
199
+ });
200
+ } catch (writeError) {
201
+ if ((writeError as NodeJS.ErrnoException).code === 'EEXIST') {
202
+ return {
203
+ acquired: false,
204
+ is_stale: false,
205
+ lock_path: lockPath,
206
+ };
207
+ }
208
+ throw writeError;
209
+ }
210
+
211
+ return {
212
+ acquired: true,
213
+ is_stale: true,
214
+ lock_path: lockPath,
215
+ };
216
+ } finally {
217
+ await cleanupTakeoverMarker(takeoverMarkerPath);
218
+ }
219
+ }
220
+
221
+ return {
222
+ acquired: false,
223
+ is_stale: false,
224
+ lock_path: lockPath,
225
+ };
226
+ }
227
+ }
228
+
229
+ export async function releaseLaneLockTool(
230
+ input: ReleaseLaneLockInput,
231
+ ): Promise<ReleaseLaneLockResult> {
232
+ const lockPath = lockPathFor(input);
233
+ const metadata = await readLaneLockMetadata(lockPath);
234
+ if (!metadata || metadata.owner !== input.owner) {
235
+ return {
236
+ released: false,
237
+ lock_path: lockPath,
238
+ };
239
+ }
240
+
241
+ await unlink(lockPath);
242
+ return {
243
+ released: true,
244
+ lock_path: lockPath,
245
+ };
246
+ }
@@ -0,0 +1,415 @@
1
+ // Copyright (c) 2026 Hellmai Ltd
2
+ // SPDX-License-Identifier: AGPL-3.0-only
3
+
4
+ import type { ToolOutput } from '@lumenflow/kernel';
5
+ import { RUNTIME_CLI_COMMANDS, runtimeCliAdapter } from './runtime-cli-adapter.js';
6
+
7
+ const MEMORY_TOOLS = {
8
+ MEM_INIT: 'mem:init',
9
+ MEM_START: 'mem:start',
10
+ MEM_READY: 'mem:ready',
11
+ MEM_CHECKPOINT: 'mem:checkpoint',
12
+ MEM_CLEANUP: 'mem:cleanup',
13
+ MEM_CONTEXT: 'mem:context',
14
+ MEM_CREATE: 'mem:create',
15
+ MEM_DELETE: 'mem:delete',
16
+ MEM_EXPORT: 'mem:export',
17
+ MEM_INBOX: 'mem:inbox',
18
+ MEM_SIGNAL: 'mem:signal',
19
+ MEM_SUMMARIZE: 'mem:summarize',
20
+ MEM_TRIAGE: 'mem:triage',
21
+ MEM_RECOVER: 'mem:recover',
22
+ } as const;
23
+
24
+ type MemoryToolName = (typeof MEMORY_TOOLS)[keyof typeof MEMORY_TOOLS];
25
+
26
+ const MEMORY_TOOL_ERROR_CODES: Record<MemoryToolName, string> = {
27
+ 'mem:init': 'MEM_INIT_ERROR',
28
+ 'mem:start': 'MEM_START_ERROR',
29
+ 'mem:ready': 'MEM_READY_ERROR',
30
+ 'mem:checkpoint': 'MEM_CHECKPOINT_ERROR',
31
+ 'mem:cleanup': 'MEM_CLEANUP_ERROR',
32
+ 'mem:context': 'MEM_CONTEXT_ERROR',
33
+ 'mem:create': 'MEM_CREATE_ERROR',
34
+ 'mem:delete': 'MEM_DELETE_ERROR',
35
+ 'mem:export': 'MEM_EXPORT_ERROR',
36
+ 'mem:inbox': 'MEM_INBOX_ERROR',
37
+ 'mem:signal': 'MEM_SIGNAL_ERROR',
38
+ 'mem:summarize': 'MEM_SUMMARIZE_ERROR',
39
+ 'mem:triage': 'MEM_TRIAGE_ERROR',
40
+ 'mem:recover': 'MEM_RECOVER_ERROR',
41
+ };
42
+
43
+ const MEMORY_TOOL_COMMANDS: Record<
44
+ MemoryToolName,
45
+ (typeof RUNTIME_CLI_COMMANDS)[keyof typeof RUNTIME_CLI_COMMANDS]
46
+ > = {
47
+ 'mem:init': RUNTIME_CLI_COMMANDS.MEM_INIT,
48
+ 'mem:start': RUNTIME_CLI_COMMANDS.MEM_START,
49
+ 'mem:ready': RUNTIME_CLI_COMMANDS.MEM_READY,
50
+ 'mem:checkpoint': RUNTIME_CLI_COMMANDS.MEM_CHECKPOINT,
51
+ 'mem:cleanup': RUNTIME_CLI_COMMANDS.MEM_CLEANUP,
52
+ 'mem:context': RUNTIME_CLI_COMMANDS.MEM_CONTEXT,
53
+ 'mem:create': RUNTIME_CLI_COMMANDS.MEM_CREATE,
54
+ 'mem:delete': RUNTIME_CLI_COMMANDS.MEM_DELETE,
55
+ 'mem:export': RUNTIME_CLI_COMMANDS.MEM_EXPORT,
56
+ 'mem:inbox': RUNTIME_CLI_COMMANDS.MEM_INBOX,
57
+ 'mem:signal': RUNTIME_CLI_COMMANDS.MEM_SIGNAL,
58
+ 'mem:summarize': RUNTIME_CLI_COMMANDS.MEM_SUMMARIZE,
59
+ 'mem:triage': RUNTIME_CLI_COMMANDS.MEM_TRIAGE,
60
+ 'mem:recover': RUNTIME_CLI_COMMANDS.MEM_RECOVER,
61
+ };
62
+
63
+ const MISSING_PARAMETER_MESSAGES = {
64
+ ID_REQUIRED: 'id is required',
65
+ WU_REQUIRED: 'wu is required',
66
+ MESSAGE_REQUIRED: 'message is required',
67
+ } as const;
68
+
69
+ interface CommandExecutionResult {
70
+ ok: boolean;
71
+ status: number;
72
+ stdout: string;
73
+ stderr: string;
74
+ executionError?: string;
75
+ }
76
+
77
+ function toRecord(input: unknown): Record<string, unknown> {
78
+ if (input && typeof input === 'object') {
79
+ return input as Record<string, unknown>;
80
+ }
81
+ return {};
82
+ }
83
+
84
+ function toStringValue(value: unknown): string | null {
85
+ if (typeof value !== 'string') {
86
+ return null;
87
+ }
88
+ const trimmed = value.trim();
89
+ return trimmed.length > 0 ? trimmed : null;
90
+ }
91
+
92
+ function toStringArray(value: unknown): string[] {
93
+ if (!Array.isArray(value)) {
94
+ return [];
95
+ }
96
+ return value
97
+ .map((entry) => toStringValue(entry))
98
+ .filter((entry): entry is string => entry !== null);
99
+ }
100
+
101
+ function toIntegerString(value: unknown): string | null {
102
+ if (typeof value === 'number' && Number.isFinite(value)) {
103
+ return String(Math.trunc(value));
104
+ }
105
+ if (typeof value === 'string') {
106
+ const trimmed = value.trim();
107
+ return trimmed.length > 0 ? trimmed : null;
108
+ }
109
+ return null;
110
+ }
111
+
112
+ async function runMemoryCommand(
113
+ toolName: MemoryToolName,
114
+ args: string[],
115
+ ): Promise<CommandExecutionResult> {
116
+ return runtimeCliAdapter.run(MEMORY_TOOL_COMMANDS[toolName], args);
117
+ }
118
+
119
+ function createMissingParameterOutput(message: string): ToolOutput {
120
+ return {
121
+ success: false,
122
+ error: {
123
+ code: 'MISSING_PARAMETER',
124
+ message,
125
+ },
126
+ };
127
+ }
128
+
129
+ function createFailureOutput(
130
+ toolName: MemoryToolName,
131
+ execution: CommandExecutionResult,
132
+ ): ToolOutput {
133
+ const stderrMessage = execution.stderr.trim();
134
+ const stdoutMessage = execution.stdout.trim();
135
+ const message =
136
+ execution.executionError ??
137
+ (stderrMessage.length > 0
138
+ ? stderrMessage
139
+ : stdoutMessage.length > 0
140
+ ? stdoutMessage
141
+ : `${toolName} failed`);
142
+ return {
143
+ success: false,
144
+ error: {
145
+ code: MEMORY_TOOL_ERROR_CODES[toolName],
146
+ message,
147
+ details: {
148
+ exit_code: execution.status,
149
+ stdout: execution.stdout,
150
+ stderr: execution.stderr,
151
+ },
152
+ },
153
+ };
154
+ }
155
+
156
+ function parseJsonOutput(stdout: string): unknown | null {
157
+ const trimmed = stdout.trim();
158
+ if (trimmed.length === 0) {
159
+ return null;
160
+ }
161
+ try {
162
+ return JSON.parse(trimmed) as unknown;
163
+ } catch {
164
+ return null;
165
+ }
166
+ }
167
+
168
+ function createSuccessOutput(
169
+ toolName: MemoryToolName,
170
+ execution: CommandExecutionResult,
171
+ ): ToolOutput {
172
+ const parsedJson = parseJsonOutput(execution.stdout);
173
+ if (parsedJson !== null) {
174
+ return {
175
+ success: true,
176
+ data: parsedJson,
177
+ };
178
+ }
179
+
180
+ const message = execution.stdout.trim().length > 0 ? execution.stdout.trim() : `${toolName} ran`;
181
+ return {
182
+ success: true,
183
+ data: {
184
+ message,
185
+ },
186
+ };
187
+ }
188
+
189
+ async function executeMemoryTool(toolName: MemoryToolName, args: string[]): Promise<ToolOutput> {
190
+ const execution = await runMemoryCommand(toolName, args);
191
+ if (!execution.ok) {
192
+ return createFailureOutput(toolName, execution);
193
+ }
194
+ return createSuccessOutput(toolName, execution);
195
+ }
196
+
197
+ export async function memInitTool(input: unknown): Promise<ToolOutput> {
198
+ const parsed = toRecord(input);
199
+ const wu = toStringValue(parsed.wu);
200
+ if (!wu) {
201
+ return createMissingParameterOutput(MISSING_PARAMETER_MESSAGES.WU_REQUIRED);
202
+ }
203
+
204
+ return executeMemoryTool(MEMORY_TOOLS.MEM_INIT, ['--wu', wu]);
205
+ }
206
+
207
+ export async function memStartTool(input: unknown): Promise<ToolOutput> {
208
+ const parsed = toRecord(input);
209
+ const wu = toStringValue(parsed.wu);
210
+ if (!wu) {
211
+ return createMissingParameterOutput(MISSING_PARAMETER_MESSAGES.WU_REQUIRED);
212
+ }
213
+
214
+ const args = ['--wu', wu];
215
+ const lane = toStringValue(parsed.lane);
216
+ if (lane) {
217
+ args.push('--lane', lane);
218
+ }
219
+
220
+ return executeMemoryTool(MEMORY_TOOLS.MEM_START, args);
221
+ }
222
+
223
+ export async function memReadyTool(input: unknown): Promise<ToolOutput> {
224
+ const parsed = toRecord(input);
225
+ const wu = toStringValue(parsed.wu);
226
+ if (!wu) {
227
+ return createMissingParameterOutput(MISSING_PARAMETER_MESSAGES.WU_REQUIRED);
228
+ }
229
+
230
+ return executeMemoryTool(MEMORY_TOOLS.MEM_READY, ['--wu', wu]);
231
+ }
232
+
233
+ export async function memCheckpointTool(input: unknown): Promise<ToolOutput> {
234
+ const parsed = toRecord(input);
235
+ const wu = toStringValue(parsed.wu);
236
+ if (!wu) {
237
+ return createMissingParameterOutput(MISSING_PARAMETER_MESSAGES.WU_REQUIRED);
238
+ }
239
+
240
+ const args = ['--wu', wu];
241
+ const message = toStringValue(parsed.message);
242
+ if (message) {
243
+ args.push('--message', message);
244
+ }
245
+
246
+ return executeMemoryTool(MEMORY_TOOLS.MEM_CHECKPOINT, args);
247
+ }
248
+
249
+ export async function memCleanupTool(input: unknown): Promise<ToolOutput> {
250
+ const parsed = toRecord(input);
251
+ const args: string[] = [];
252
+ if (parsed.dry_run === true) {
253
+ args.push('--dry-run');
254
+ }
255
+
256
+ return executeMemoryTool(MEMORY_TOOLS.MEM_CLEANUP, args);
257
+ }
258
+
259
+ export async function memContextTool(input: unknown): Promise<ToolOutput> {
260
+ const parsed = toRecord(input);
261
+ const wu = toStringValue(parsed.wu);
262
+ if (!wu) {
263
+ return createMissingParameterOutput(MISSING_PARAMETER_MESSAGES.WU_REQUIRED);
264
+ }
265
+
266
+ const args = ['--wu', wu];
267
+ const lane = toStringValue(parsed.lane);
268
+ if (lane) {
269
+ args.push('--lane', lane);
270
+ }
271
+
272
+ return executeMemoryTool(MEMORY_TOOLS.MEM_CONTEXT, args);
273
+ }
274
+
275
+ export async function memCreateTool(input: unknown): Promise<ToolOutput> {
276
+ const parsed = toRecord(input);
277
+ const message = toStringValue(parsed.message);
278
+ if (!message) {
279
+ return createMissingParameterOutput(MISSING_PARAMETER_MESSAGES.MESSAGE_REQUIRED);
280
+ }
281
+
282
+ const wu = toStringValue(parsed.wu);
283
+ if (!wu) {
284
+ return createMissingParameterOutput(MISSING_PARAMETER_MESSAGES.WU_REQUIRED);
285
+ }
286
+
287
+ const args = [message, '--wu', wu];
288
+ const type = toStringValue(parsed.type);
289
+ if (type) {
290
+ args.push('--type', type);
291
+ }
292
+ const tags = toStringArray(parsed.tags);
293
+ if (tags.length > 0) {
294
+ args.push('--tags', tags.join(','));
295
+ }
296
+
297
+ return executeMemoryTool(MEMORY_TOOLS.MEM_CREATE, args);
298
+ }
299
+
300
+ export async function memDeleteTool(input: unknown): Promise<ToolOutput> {
301
+ const parsed = toRecord(input);
302
+ const id = toStringValue(parsed.id);
303
+ if (!id) {
304
+ return createMissingParameterOutput(MISSING_PARAMETER_MESSAGES.ID_REQUIRED);
305
+ }
306
+
307
+ return executeMemoryTool(MEMORY_TOOLS.MEM_DELETE, ['--id', id]);
308
+ }
309
+
310
+ export async function memExportTool(input: unknown): Promise<ToolOutput> {
311
+ const parsed = toRecord(input);
312
+ const wu = toStringValue(parsed.wu);
313
+ if (!wu) {
314
+ return createMissingParameterOutput(MISSING_PARAMETER_MESSAGES.WU_REQUIRED);
315
+ }
316
+
317
+ const args = ['--wu', wu];
318
+ const format = toStringValue(parsed.format);
319
+ if (format) {
320
+ args.push('--format', format);
321
+ }
322
+
323
+ return executeMemoryTool(MEMORY_TOOLS.MEM_EXPORT, args);
324
+ }
325
+
326
+ export async function memInboxTool(input: unknown): Promise<ToolOutput> {
327
+ const parsed = toRecord(input);
328
+ const args: string[] = [];
329
+ const since = toStringValue(parsed.since);
330
+ if (since) {
331
+ args.push('--since', since);
332
+ }
333
+ const wu = toStringValue(parsed.wu);
334
+ if (wu) {
335
+ args.push('--wu', wu);
336
+ }
337
+ const lane = toStringValue(parsed.lane);
338
+ if (lane) {
339
+ args.push('--lane', lane);
340
+ }
341
+
342
+ return executeMemoryTool(MEMORY_TOOLS.MEM_INBOX, args);
343
+ }
344
+
345
+ export async function memSignalTool(input: unknown): Promise<ToolOutput> {
346
+ const parsed = toRecord(input);
347
+ const message = toStringValue(parsed.message);
348
+ if (!message) {
349
+ return createMissingParameterOutput(MISSING_PARAMETER_MESSAGES.MESSAGE_REQUIRED);
350
+ }
351
+
352
+ const wu = toStringValue(parsed.wu);
353
+ if (!wu) {
354
+ return createMissingParameterOutput(MISSING_PARAMETER_MESSAGES.WU_REQUIRED);
355
+ }
356
+
357
+ return executeMemoryTool(MEMORY_TOOLS.MEM_SIGNAL, [message, '--wu', wu]);
358
+ }
359
+
360
+ export async function memSummarizeTool(input: unknown): Promise<ToolOutput> {
361
+ const parsed = toRecord(input);
362
+ const wu = toStringValue(parsed.wu);
363
+ if (!wu) {
364
+ return createMissingParameterOutput(MISSING_PARAMETER_MESSAGES.WU_REQUIRED);
365
+ }
366
+
367
+ return executeMemoryTool(MEMORY_TOOLS.MEM_SUMMARIZE, ['--wu', wu]);
368
+ }
369
+
370
+ export async function memTriageTool(input: unknown): Promise<ToolOutput> {
371
+ const parsed = toRecord(input);
372
+ const wu = toStringValue(parsed.wu);
373
+ if (!wu) {
374
+ return createMissingParameterOutput(MISSING_PARAMETER_MESSAGES.WU_REQUIRED);
375
+ }
376
+
377
+ const args = ['--wu', wu];
378
+ const promote = toStringValue(parsed.promote);
379
+ if (promote) {
380
+ args.push('--promote', promote);
381
+ }
382
+ const lane = toStringValue(parsed.lane);
383
+ if (lane) {
384
+ args.push('--lane', lane);
385
+ }
386
+
387
+ return executeMemoryTool(MEMORY_TOOLS.MEM_TRIAGE, args);
388
+ }
389
+
390
+ export async function memRecoverTool(input: unknown): Promise<ToolOutput> {
391
+ const parsed = toRecord(input);
392
+ const wu = toStringValue(parsed.wu);
393
+ if (!wu) {
394
+ return createMissingParameterOutput(MISSING_PARAMETER_MESSAGES.WU_REQUIRED);
395
+ }
396
+
397
+ const args = ['--wu', wu];
398
+ const maxSize = toIntegerString(parsed.max_size);
399
+ if (maxSize) {
400
+ args.push('--max-size', maxSize);
401
+ }
402
+ const format = toStringValue(parsed.format);
403
+ if (format) {
404
+ args.push('--format', format);
405
+ }
406
+ if (parsed.quiet === true) {
407
+ args.push('--quiet');
408
+ }
409
+ const baseDir = toStringValue(parsed.base_dir);
410
+ if (baseDir) {
411
+ args.push('--base-dir', baseDir);
412
+ }
413
+
414
+ return executeMemoryTool(MEMORY_TOOLS.MEM_RECOVER, args);
415
+ }
@@ -0,0 +1,21 @@
1
+ // Copyright (c) 2026 Hellmai Ltd
2
+ // SPDX-License-Identifier: AGPL-3.0-only
3
+
4
+ interface PendingRuntimeMigrationOutput {
5
+ success: false;
6
+ error: {
7
+ code: 'TOOL_NOT_IMPLEMENTED';
8
+ message: string;
9
+ };
10
+ }
11
+
12
+ export async function pendingRuntimeMigrationTool(): Promise<PendingRuntimeMigrationOutput> {
13
+ return {
14
+ success: false,
15
+ error: {
16
+ code: 'TOOL_NOT_IMPLEMENTED',
17
+ message:
18
+ 'Tool is declared in the software-delivery manifest but has not been migrated to a runtime handler yet.',
19
+ },
20
+ };
21
+ }