oh-my-codex 0.16.0 → 0.16.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (357) hide show
  1. package/Cargo.lock +5 -5
  2. package/Cargo.toml +1 -1
  3. package/README.md +2 -2
  4. package/crates/omx-explore/src/main.rs +434 -28
  5. package/dist/agents/__tests__/native-config.test.js +50 -0
  6. package/dist/agents/__tests__/native-config.test.js.map +1 -1
  7. package/dist/agents/native-config.d.ts.map +1 -1
  8. package/dist/agents/native-config.js +3 -2
  9. package/dist/agents/native-config.js.map +1 -1
  10. package/dist/cli/__tests__/codex-plugin-layout.test.js +1 -0
  11. package/dist/cli/__tests__/codex-plugin-layout.test.js.map +1 -1
  12. package/dist/cli/__tests__/doctor-warning-copy.test.js +1 -1
  13. package/dist/cli/__tests__/doctor-warning-copy.test.js.map +1 -1
  14. package/dist/cli/__tests__/explore.test.js +120 -3
  15. package/dist/cli/__tests__/explore.test.js.map +1 -1
  16. package/dist/cli/__tests__/imagegen-continuation.test.d.ts +2 -0
  17. package/dist/cli/__tests__/imagegen-continuation.test.d.ts.map +1 -0
  18. package/dist/cli/__tests__/imagegen-continuation.test.js +135 -0
  19. package/dist/cli/__tests__/imagegen-continuation.test.js.map +1 -0
  20. package/dist/cli/__tests__/index.test.js +182 -18
  21. package/dist/cli/__tests__/index.test.js.map +1 -1
  22. package/dist/cli/__tests__/launch-fallback.test.js +88 -2
  23. package/dist/cli/__tests__/launch-fallback.test.js.map +1 -1
  24. package/dist/cli/__tests__/ralph.test.js +62 -0
  25. package/dist/cli/__tests__/ralph.test.js.map +1 -1
  26. package/dist/cli/__tests__/setup-install-mode.test.js +48 -0
  27. package/dist/cli/__tests__/setup-install-mode.test.js.map +1 -1
  28. package/dist/cli/__tests__/setup-scope.test.js +12 -0
  29. package/dist/cli/__tests__/setup-scope.test.js.map +1 -1
  30. package/dist/cli/__tests__/team.test.js +465 -12
  31. package/dist/cli/__tests__/team.test.js.map +1 -1
  32. package/dist/cli/__tests__/ultragoal.test.js +50 -5
  33. package/dist/cli/__tests__/ultragoal.test.js.map +1 -1
  34. package/dist/cli/__tests__/uninstall.test.js +6 -2
  35. package/dist/cli/__tests__/uninstall.test.js.map +1 -1
  36. package/dist/cli/explore.d.ts.map +1 -1
  37. package/dist/cli/explore.js +211 -12
  38. package/dist/cli/explore.js.map +1 -1
  39. package/dist/cli/index.d.ts +11 -3
  40. package/dist/cli/index.d.ts.map +1 -1
  41. package/dist/cli/index.js +124 -18
  42. package/dist/cli/index.js.map +1 -1
  43. package/dist/cli/ralph.d.ts.map +1 -1
  44. package/dist/cli/ralph.js +37 -3
  45. package/dist/cli/ralph.js.map +1 -1
  46. package/dist/cli/setup.d.ts.map +1 -1
  47. package/dist/cli/setup.js +100 -9
  48. package/dist/cli/setup.js.map +1 -1
  49. package/dist/cli/team.d.ts +1 -0
  50. package/dist/cli/team.d.ts.map +1 -1
  51. package/dist/cli/team.js +42 -7
  52. package/dist/cli/team.js.map +1 -1
  53. package/dist/cli/ultragoal.d.ts +1 -1
  54. package/dist/cli/ultragoal.d.ts.map +1 -1
  55. package/dist/cli/ultragoal.js +29 -8
  56. package/dist/cli/ultragoal.js.map +1 -1
  57. package/dist/cli/uninstall.d.ts.map +1 -1
  58. package/dist/cli/uninstall.js +2 -1
  59. package/dist/cli/uninstall.js.map +1 -1
  60. package/dist/config/__tests__/codex-hooks.test.js +97 -2
  61. package/dist/config/__tests__/codex-hooks.test.js.map +1 -1
  62. package/dist/config/__tests__/generator-idempotent.test.js +1 -1
  63. package/dist/config/__tests__/generator-idempotent.test.js.map +1 -1
  64. package/dist/config/__tests__/generator-notify.test.js +22 -0
  65. package/dist/config/__tests__/generator-notify.test.js.map +1 -1
  66. package/dist/config/__tests__/models.test.js +18 -1
  67. package/dist/config/__tests__/models.test.js.map +1 -1
  68. package/dist/config/__tests__/wiki-config-contract.test.js +2 -1
  69. package/dist/config/__tests__/wiki-config-contract.test.js.map +1 -1
  70. package/dist/config/codex-hooks.d.ts +17 -3
  71. package/dist/config/codex-hooks.d.ts.map +1 -1
  72. package/dist/config/codex-hooks.js +102 -2
  73. package/dist/config/codex-hooks.js.map +1 -1
  74. package/dist/config/generator.d.ts +4 -1
  75. package/dist/config/generator.d.ts.map +1 -1
  76. package/dist/config/generator.js +69 -12
  77. package/dist/config/generator.js.map +1 -1
  78. package/dist/config/models.d.ts +6 -0
  79. package/dist/config/models.d.ts.map +1 -1
  80. package/dist/config/models.js +37 -0
  81. package/dist/config/models.js.map +1 -1
  82. package/dist/exec/followup.d.ts +1 -0
  83. package/dist/exec/followup.d.ts.map +1 -1
  84. package/dist/exec/followup.js +9 -3
  85. package/dist/exec/followup.js.map +1 -1
  86. package/dist/hooks/__tests__/anti-slop-workflow.test.js +19 -0
  87. package/dist/hooks/__tests__/anti-slop-workflow.test.js.map +1 -1
  88. package/dist/hooks/__tests__/consensus-execution-handoff.test.js +19 -2
  89. package/dist/hooks/__tests__/consensus-execution-handoff.test.js.map +1 -1
  90. package/dist/hooks/__tests__/deep-interview-contract.test.js +40 -0
  91. package/dist/hooks/__tests__/deep-interview-contract.test.js.map +1 -1
  92. package/dist/hooks/__tests__/foreground-isolation-contract.test.d.ts +2 -0
  93. package/dist/hooks/__tests__/foreground-isolation-contract.test.d.ts.map +1 -0
  94. package/dist/hooks/__tests__/foreground-isolation-contract.test.js +28 -0
  95. package/dist/hooks/__tests__/foreground-isolation-contract.test.js.map +1 -0
  96. package/dist/hooks/__tests__/keyword-detector.test.js +37 -25
  97. package/dist/hooks/__tests__/keyword-detector.test.js.map +1 -1
  98. package/dist/hooks/__tests__/notify-hook-auto-nudge.test.js +10 -4
  99. package/dist/hooks/__tests__/notify-hook-auto-nudge.test.js.map +1 -1
  100. package/dist/hooks/__tests__/session.test.js +32 -0
  101. package/dist/hooks/__tests__/session.test.js.map +1 -1
  102. package/dist/hooks/__tests__/wiki-docs-contract.test.js +6 -4
  103. package/dist/hooks/__tests__/wiki-docs-contract.test.js.map +1 -1
  104. package/dist/hooks/codebase-map.d.ts.map +1 -1
  105. package/dist/hooks/codebase-map.js +3 -2
  106. package/dist/hooks/codebase-map.js.map +1 -1
  107. package/dist/hooks/extensibility/dispatcher.d.ts.map +1 -1
  108. package/dist/hooks/extensibility/dispatcher.js +6 -4
  109. package/dist/hooks/extensibility/dispatcher.js.map +1 -1
  110. package/dist/hooks/extensibility/logging.d.ts.map +1 -1
  111. package/dist/hooks/extensibility/logging.js +3 -2
  112. package/dist/hooks/extensibility/logging.js.map +1 -1
  113. package/dist/hooks/extensibility/sdk/paths.d.ts.map +1 -1
  114. package/dist/hooks/extensibility/sdk/paths.js +4 -3
  115. package/dist/hooks/extensibility/sdk/paths.js.map +1 -1
  116. package/dist/hooks/keyword-detector.d.ts.map +1 -1
  117. package/dist/hooks/keyword-detector.js +2 -4
  118. package/dist/hooks/keyword-detector.js.map +1 -1
  119. package/dist/hooks/session.d.ts.map +1 -1
  120. package/dist/hooks/session.js +22 -12
  121. package/dist/hooks/session.js.map +1 -1
  122. package/dist/hud/__tests__/hud-tmux-injection.test.js +8 -7
  123. package/dist/hud/__tests__/hud-tmux-injection.test.js.map +1 -1
  124. package/dist/hud/__tests__/reconcile.test.js +1 -1
  125. package/dist/hud/__tests__/state.test.js +24 -0
  126. package/dist/hud/__tests__/state.test.js.map +1 -1
  127. package/dist/hud/index.js +1 -1
  128. package/dist/hud/index.js.map +1 -1
  129. package/dist/hud/state.d.ts.map +1 -1
  130. package/dist/hud/state.js +22 -8
  131. package/dist/hud/state.js.map +1 -1
  132. package/dist/hud/tmux.js +1 -1
  133. package/dist/hud/tmux.js.map +1 -1
  134. package/dist/imagegen/continuation.d.ts +44 -0
  135. package/dist/imagegen/continuation.d.ts.map +1 -0
  136. package/dist/imagegen/continuation.js +220 -0
  137. package/dist/imagegen/continuation.js.map +1 -0
  138. package/dist/mcp/__tests__/bootstrap.test.js +47 -2
  139. package/dist/mcp/__tests__/bootstrap.test.js.map +1 -1
  140. package/dist/mcp/__tests__/server-lifecycle.test.js +49 -1
  141. package/dist/mcp/__tests__/server-lifecycle.test.js.map +1 -1
  142. package/dist/mcp/__tests__/state-server.test.js +145 -6
  143. package/dist/mcp/__tests__/state-server.test.js.map +1 -1
  144. package/dist/mcp/__tests__/wiki-server.test.js +97 -1
  145. package/dist/mcp/__tests__/wiki-server.test.js.map +1 -1
  146. package/dist/mcp/bootstrap.d.ts +2 -0
  147. package/dist/mcp/bootstrap.d.ts.map +1 -1
  148. package/dist/mcp/bootstrap.js +95 -15
  149. package/dist/mcp/bootstrap.js.map +1 -1
  150. package/dist/mcp/lifecycle-telemetry.d.ts +16 -0
  151. package/dist/mcp/lifecycle-telemetry.d.ts.map +1 -0
  152. package/dist/mcp/lifecycle-telemetry.js +95 -0
  153. package/dist/mcp/lifecycle-telemetry.js.map +1 -0
  154. package/dist/mcp/wiki-server.d.ts.map +1 -1
  155. package/dist/mcp/wiki-server.js +11 -2
  156. package/dist/mcp/wiki-server.js.map +1 -1
  157. package/dist/pipeline/__tests__/stages.test.js +274 -5
  158. package/dist/pipeline/__tests__/stages.test.js.map +1 -1
  159. package/dist/pipeline/stages/team-exec.d.ts +2 -0
  160. package/dist/pipeline/stages/team-exec.d.ts.map +1 -1
  161. package/dist/pipeline/stages/team-exec.js +51 -26
  162. package/dist/pipeline/stages/team-exec.js.map +1 -1
  163. package/dist/planning/__tests__/artifacts.test.js +138 -3
  164. package/dist/planning/__tests__/artifacts.test.js.map +1 -1
  165. package/dist/planning/__tests__/context-pack-status.test.d.ts +2 -0
  166. package/dist/planning/__tests__/context-pack-status.test.d.ts.map +1 -0
  167. package/dist/planning/__tests__/context-pack-status.test.js +271 -0
  168. package/dist/planning/__tests__/context-pack-status.test.js.map +1 -0
  169. package/dist/planning/artifacts.d.ts +12 -1
  170. package/dist/planning/artifacts.d.ts.map +1 -1
  171. package/dist/planning/artifacts.js +32 -9
  172. package/dist/planning/artifacts.js.map +1 -1
  173. package/dist/planning/context-pack-status.d.ts +42 -0
  174. package/dist/planning/context-pack-status.d.ts.map +1 -0
  175. package/dist/planning/context-pack-status.js +479 -0
  176. package/dist/planning/context-pack-status.js.map +1 -0
  177. package/dist/runtime/__tests__/process-tree.test.d.ts +2 -0
  178. package/dist/runtime/__tests__/process-tree.test.d.ts.map +1 -0
  179. package/dist/runtime/__tests__/process-tree.test.js +107 -0
  180. package/dist/runtime/__tests__/process-tree.test.js.map +1 -0
  181. package/dist/runtime/process-tree.d.ts +28 -0
  182. package/dist/runtime/process-tree.d.ts.map +1 -0
  183. package/dist/runtime/process-tree.js +230 -0
  184. package/dist/runtime/process-tree.js.map +1 -0
  185. package/dist/scripts/__tests__/codex-native-hook.test.js +267 -2
  186. package/dist/scripts/__tests__/codex-native-hook.test.js.map +1 -1
  187. package/dist/scripts/__tests__/notify-state-io.test.d.ts +2 -0
  188. package/dist/scripts/__tests__/notify-state-io.test.d.ts.map +1 -0
  189. package/dist/scripts/__tests__/notify-state-io.test.js +40 -0
  190. package/dist/scripts/__tests__/notify-state-io.test.js.map +1 -0
  191. package/dist/scripts/codex-execution-surface.d.ts +1 -1
  192. package/dist/scripts/codex-execution-surface.d.ts.map +1 -1
  193. package/dist/scripts/codex-execution-surface.js.map +1 -1
  194. package/dist/scripts/codex-native-hook.d.ts +1 -1
  195. package/dist/scripts/codex-native-hook.d.ts.map +1 -1
  196. package/dist/scripts/codex-native-hook.js +141 -9
  197. package/dist/scripts/codex-native-hook.js.map +1 -1
  198. package/dist/scripts/notify-hook/managed-tmux.d.ts.map +1 -1
  199. package/dist/scripts/notify-hook/managed-tmux.js +6 -9
  200. package/dist/scripts/notify-hook/managed-tmux.js.map +1 -1
  201. package/dist/scripts/notify-hook/process-runner.d.ts.map +1 -1
  202. package/dist/scripts/notify-hook/process-runner.js +4 -1
  203. package/dist/scripts/notify-hook/process-runner.js.map +1 -1
  204. package/dist/scripts/notify-hook/state-io.d.ts.map +1 -1
  205. package/dist/scripts/notify-hook/state-io.js +4 -7
  206. package/dist/scripts/notify-hook/state-io.js.map +1 -1
  207. package/dist/scripts/notify-hook.js +25 -3
  208. package/dist/scripts/notify-hook.js.map +1 -1
  209. package/dist/scripts/verify-native-agents.d.ts.map +1 -1
  210. package/dist/scripts/verify-native-agents.js +3 -1
  211. package/dist/scripts/verify-native-agents.js.map +1 -1
  212. package/dist/sidecar/__tests__/tmux.test.js +1 -1
  213. package/dist/sidecar/__tests__/tmux.test.js.map +1 -1
  214. package/dist/sidecar/tmux.js +1 -1
  215. package/dist/sidecar/tmux.js.map +1 -1
  216. package/dist/state/__tests__/operations.test.js +79 -0
  217. package/dist/state/__tests__/operations.test.js.map +1 -1
  218. package/dist/state/__tests__/skill-active.test.js +10 -18
  219. package/dist/state/__tests__/skill-active.test.js.map +1 -1
  220. package/dist/state/__tests__/workflow-transition.test.js +45 -1
  221. package/dist/state/__tests__/workflow-transition.test.js.map +1 -1
  222. package/dist/state/operations.d.ts.map +1 -1
  223. package/dist/state/operations.js +1 -20
  224. package/dist/state/operations.js.map +1 -1
  225. package/dist/state/skill-active.d.ts +1 -0
  226. package/dist/state/skill-active.d.ts.map +1 -1
  227. package/dist/state/skill-active.js +28 -18
  228. package/dist/state/skill-active.js.map +1 -1
  229. package/dist/state/workflow-transition-reconcile.d.ts.map +1 -1
  230. package/dist/state/workflow-transition-reconcile.js +3 -2
  231. package/dist/state/workflow-transition-reconcile.js.map +1 -1
  232. package/dist/state/workflow-transition.js +2 -2
  233. package/dist/state/workflow-transition.js.map +1 -1
  234. package/dist/team/__tests__/approved-execution.test.js +96 -0
  235. package/dist/team/__tests__/approved-execution.test.js.map +1 -1
  236. package/dist/team/__tests__/followup-planner.test.js +16 -0
  237. package/dist/team/__tests__/followup-planner.test.js.map +1 -1
  238. package/dist/team/__tests__/model-contract.test.js +16 -0
  239. package/dist/team/__tests__/model-contract.test.js.map +1 -1
  240. package/dist/team/__tests__/repo-aware-decomposition.test.js +20 -0
  241. package/dist/team/__tests__/repo-aware-decomposition.test.js.map +1 -1
  242. package/dist/team/__tests__/runtime-cli.test.js +16 -0
  243. package/dist/team/__tests__/runtime-cli.test.js.map +1 -1
  244. package/dist/team/__tests__/runtime.test.js +209 -11
  245. package/dist/team/__tests__/runtime.test.js.map +1 -1
  246. package/dist/team/__tests__/scaling.test.js +110 -0
  247. package/dist/team/__tests__/scaling.test.js.map +1 -1
  248. package/dist/team/__tests__/tmux-session.test.js +9 -0
  249. package/dist/team/__tests__/tmux-session.test.js.map +1 -1
  250. package/dist/team/__tests__/worker-runtime-identity.test.js +6 -0
  251. package/dist/team/__tests__/worker-runtime-identity.test.js.map +1 -1
  252. package/dist/team/approved-execution.d.ts +13 -0
  253. package/dist/team/approved-execution.d.ts.map +1 -1
  254. package/dist/team/approved-execution.js +40 -22
  255. package/dist/team/approved-execution.js.map +1 -1
  256. package/dist/team/followup-planner.d.ts +1 -0
  257. package/dist/team/followup-planner.d.ts.map +1 -1
  258. package/dist/team/followup-planner.js +9 -9
  259. package/dist/team/followup-planner.js.map +1 -1
  260. package/dist/team/model-contract.d.ts +1 -1
  261. package/dist/team/model-contract.d.ts.map +1 -1
  262. package/dist/team/model-contract.js +4 -3
  263. package/dist/team/model-contract.js.map +1 -1
  264. package/dist/team/repo-aware-decomposition.d.ts +1 -0
  265. package/dist/team/repo-aware-decomposition.d.ts.map +1 -1
  266. package/dist/team/repo-aware-decomposition.js +5 -1
  267. package/dist/team/repo-aware-decomposition.js.map +1 -1
  268. package/dist/team/runtime-cli.d.ts +4 -0
  269. package/dist/team/runtime-cli.d.ts.map +1 -1
  270. package/dist/team/runtime-cli.js +14 -1
  271. package/dist/team/runtime-cli.js.map +1 -1
  272. package/dist/team/runtime.d.ts +1 -0
  273. package/dist/team/runtime.d.ts.map +1 -1
  274. package/dist/team/runtime.js +46 -16
  275. package/dist/team/runtime.js.map +1 -1
  276. package/dist/team/scaling.d.ts.map +1 -1
  277. package/dist/team/scaling.js +13 -6
  278. package/dist/team/scaling.js.map +1 -1
  279. package/dist/team/tmux-session.d.ts.map +1 -1
  280. package/dist/team/tmux-session.js +7 -0
  281. package/dist/team/tmux-session.js.map +1 -1
  282. package/dist/ultragoal/__tests__/artifacts.test.js +129 -4
  283. package/dist/ultragoal/__tests__/artifacts.test.js.map +1 -1
  284. package/dist/ultragoal/__tests__/docs-contract.test.d.ts +2 -0
  285. package/dist/ultragoal/__tests__/docs-contract.test.d.ts.map +1 -0
  286. package/dist/ultragoal/__tests__/docs-contract.test.js +31 -0
  287. package/dist/ultragoal/__tests__/docs-contract.test.js.map +1 -0
  288. package/dist/ultragoal/artifacts.d.ts +6 -2
  289. package/dist/ultragoal/artifacts.d.ts.map +1 -1
  290. package/dist/ultragoal/artifacts.js +108 -4
  291. package/dist/ultragoal/artifacts.js.map +1 -1
  292. package/dist/utils/paths.d.ts +3 -1
  293. package/dist/utils/paths.d.ts.map +1 -1
  294. package/dist/utils/paths.js +6 -2
  295. package/dist/utils/paths.js.map +1 -1
  296. package/dist/verification/__tests__/ci-rust-gates.test.js +44 -14
  297. package/dist/verification/__tests__/ci-rust-gates.test.js.map +1 -1
  298. package/dist/wiki/__tests__/ingest.test.js +35 -1
  299. package/dist/wiki/__tests__/ingest.test.js.map +1 -1
  300. package/dist/wiki/__tests__/lint.test.js +14 -1
  301. package/dist/wiki/__tests__/lint.test.js.map +1 -1
  302. package/dist/wiki/__tests__/query.test.js +28 -3
  303. package/dist/wiki/__tests__/query.test.js.map +1 -1
  304. package/dist/wiki/__tests__/session-hooks.test.js +30 -2
  305. package/dist/wiki/__tests__/session-hooks.test.js.map +1 -1
  306. package/dist/wiki/__tests__/storage.test.js +62 -22
  307. package/dist/wiki/__tests__/storage.test.js.map +1 -1
  308. package/dist/wiki/index.d.ts +2 -2
  309. package/dist/wiki/index.d.ts.map +1 -1
  310. package/dist/wiki/index.js +2 -2
  311. package/dist/wiki/index.js.map +1 -1
  312. package/dist/wiki/ingest.js +2 -2
  313. package/dist/wiki/ingest.js.map +1 -1
  314. package/dist/wiki/lifecycle.d.ts +5 -0
  315. package/dist/wiki/lifecycle.d.ts.map +1 -1
  316. package/dist/wiki/lifecycle.js +31 -4
  317. package/dist/wiki/lifecycle.js.map +1 -1
  318. package/dist/wiki/lint.d.ts.map +1 -1
  319. package/dist/wiki/lint.js +12 -8
  320. package/dist/wiki/lint.js.map +1 -1
  321. package/dist/wiki/query.d.ts.map +1 -1
  322. package/dist/wiki/query.js +3 -2
  323. package/dist/wiki/query.js.map +1 -1
  324. package/dist/wiki/storage.d.ts +4 -0
  325. package/dist/wiki/storage.d.ts.map +1 -1
  326. package/dist/wiki/storage.js +54 -18
  327. package/dist/wiki/storage.js.map +1 -1
  328. package/package.json +1 -1
  329. package/plugins/oh-my-codex/.codex-plugin/plugin.json +1 -1
  330. package/plugins/oh-my-codex/skills/ai-slop-cleaner/SKILL.md +9 -0
  331. package/plugins/oh-my-codex/skills/deep-interview/SKILL.md +25 -2
  332. package/plugins/oh-my-codex/skills/omx-setup/SKILL.md +1 -1
  333. package/plugins/oh-my-codex/skills/plan/SKILL.md +7 -4
  334. package/plugins/oh-my-codex/skills/ralplan/SKILL.md +13 -3
  335. package/plugins/oh-my-codex/skills/team/SKILL.md +2 -2
  336. package/plugins/oh-my-codex/skills/ultragoal/SKILL.md +11 -7
  337. package/plugins/oh-my-codex/skills/visual-ralph/SKILL.md +8 -0
  338. package/plugins/oh-my-codex/skills/wiki/SKILL.md +5 -5
  339. package/prompts/planner.md +1 -1
  340. package/skills/ai-slop-cleaner/SKILL.md +9 -0
  341. package/skills/deep-interview/SKILL.md +25 -2
  342. package/skills/omx-setup/SKILL.md +1 -1
  343. package/skills/plan/SKILL.md +7 -4
  344. package/skills/ralplan/SKILL.md +13 -3
  345. package/skills/team/SKILL.md +2 -2
  346. package/skills/ultragoal/SKILL.md +11 -7
  347. package/skills/visual-ralph/SKILL.md +8 -0
  348. package/skills/wiki/SKILL.md +5 -5
  349. package/src/scripts/__tests__/codex-native-hook.test.ts +302 -2
  350. package/src/scripts/__tests__/notify-state-io.test.ts +73 -0
  351. package/src/scripts/codex-execution-surface.ts +2 -0
  352. package/src/scripts/codex-native-hook.ts +163 -16
  353. package/src/scripts/notify-hook/managed-tmux.ts +6 -7
  354. package/src/scripts/notify-hook/process-runner.ts +4 -1
  355. package/src/scripts/notify-hook/state-io.ts +5 -7
  356. package/src/scripts/notify-hook.ts +26 -3
  357. package/src/scripts/verify-native-agents.ts +3 -1
@@ -0,0 +1,230 @@
1
+ import { spawn } from 'node:child_process';
2
+ import { readdirSync, readFileSync } from 'node:fs';
3
+ import { buildPlatformCommandSpec } from '../utils/platform-command.js';
4
+ const DEFAULT_SIGTERM_GRACE_MS = 1_000;
5
+ const DEFAULT_PROCESS_LIMIT_POLL_MS = 100;
6
+ function parsePositiveTimeout(value) {
7
+ if (value === undefined)
8
+ return undefined;
9
+ if (!Number.isFinite(value) || value <= 0)
10
+ return undefined;
11
+ return Math.floor(value);
12
+ }
13
+ function killProcessTree(child, platform, signal) {
14
+ if (child.pid === undefined)
15
+ return;
16
+ try {
17
+ if (platform === 'win32') {
18
+ child.kill(signal);
19
+ return;
20
+ }
21
+ // Children are launched as a detached process group on POSIX so a negative
22
+ // PID targets the whole tree instead of only the direct wrapper process.
23
+ process.kill(-child.pid, signal);
24
+ }
25
+ catch (err) {
26
+ if (err.code === 'ESRCH')
27
+ return;
28
+ try {
29
+ child.kill(signal);
30
+ }
31
+ catch (fallbackErr) {
32
+ if (fallbackErr.code !== 'ESRCH')
33
+ throw fallbackErr;
34
+ }
35
+ }
36
+ }
37
+ function parsePositiveInteger(value) {
38
+ if (value === undefined)
39
+ return undefined;
40
+ if (!Number.isFinite(value) || value <= 0)
41
+ return undefined;
42
+ return Math.floor(value);
43
+ }
44
+ function readLinuxProcessTable() {
45
+ try {
46
+ const entries = readdirSync('/proc', { withFileTypes: true });
47
+ const table = new Map();
48
+ for (const entry of entries) {
49
+ if (!entry.isDirectory() || !/^\d+$/.test(entry.name))
50
+ continue;
51
+ const pid = Number.parseInt(entry.name, 10);
52
+ let stat;
53
+ try {
54
+ stat = readFileSync(`/proc/${entry.name}/stat`, 'utf-8');
55
+ }
56
+ catch {
57
+ continue;
58
+ }
59
+ const closeParen = stat.lastIndexOf(')');
60
+ if (closeParen < 0)
61
+ continue;
62
+ const fields = stat.slice(closeParen + 2).split(' ');
63
+ const ppid = Number.parseInt(fields[1] ?? '', 10);
64
+ if (Number.isFinite(ppid))
65
+ table.set(pid, ppid);
66
+ }
67
+ return table;
68
+ }
69
+ catch {
70
+ return undefined;
71
+ }
72
+ }
73
+ function countDescendantsLinux(rootPid) {
74
+ const table = readLinuxProcessTable();
75
+ if (!table)
76
+ return undefined;
77
+ const children = new Map();
78
+ for (const [pid, ppid] of table) {
79
+ const list = children.get(ppid) ?? [];
80
+ list.push(pid);
81
+ children.set(ppid, list);
82
+ }
83
+ let count = 0;
84
+ const stack = [...(children.get(rootPid) ?? [])];
85
+ while (stack.length > 0) {
86
+ const pid = stack.pop();
87
+ if (pid === undefined)
88
+ continue;
89
+ count += 1;
90
+ stack.push(...(children.get(pid) ?? []));
91
+ }
92
+ return count;
93
+ }
94
+ export function runProcessTreeWithTimeout(command, args, options = {}) {
95
+ const platform = options.platform ?? process.platform;
96
+ const env = options.env ?? process.env;
97
+ const spec = buildPlatformCommandSpec(command, args, platform, env, options.existsImpl);
98
+ const spawnImpl = options.spawnImpl ?? spawn;
99
+ const timeoutMs = parsePositiveTimeout(options.timeoutMs);
100
+ const killSignal = options.killSignal ?? 'SIGTERM';
101
+ const sigkillGraceMs = options.sigkillGraceMs ?? DEFAULT_SIGTERM_GRACE_MS;
102
+ const encoding = options.encoding ?? 'utf-8';
103
+ const maxOutputBytes = parsePositiveInteger(options.maxOutputBytes);
104
+ const maxProcessCount = parsePositiveInteger(options.maxProcessCount);
105
+ const processLimitPollMs = parsePositiveInteger(options.processLimitPollMs) ?? DEFAULT_PROCESS_LIMIT_POLL_MS;
106
+ return new Promise((resolve) => {
107
+ let stdout = '';
108
+ let stderr = '';
109
+ let timedOut = false;
110
+ let processLimitExceeded = false;
111
+ let outputLimitExceeded = false;
112
+ let settled = false;
113
+ let timeoutTimer;
114
+ let sigkillTimer;
115
+ let processLimitTimer;
116
+ const cleanupSignals = ['SIGINT', 'SIGTERM', 'SIGHUP'];
117
+ const child = spawnImpl(spec.command, spec.args, {
118
+ cwd: options.cwd,
119
+ env,
120
+ detached: platform !== 'win32',
121
+ stdio: ['ignore', 'pipe', 'pipe'],
122
+ windowsHide: true,
123
+ });
124
+ const terminate = (signal = killSignal) => {
125
+ killProcessTree(child, platform, signal);
126
+ if (signal !== 'SIGKILL') {
127
+ sigkillTimer ??= setTimeout(() => {
128
+ if (!settled)
129
+ killProcessTree(child, platform, 'SIGKILL');
130
+ }, sigkillGraceMs);
131
+ }
132
+ };
133
+ const parentCleanupHandler = (signal) => {
134
+ terminate(typeof signal === 'string' ? signal : killSignal);
135
+ };
136
+ if (options.cleanupOnParentExit) {
137
+ for (const signal of cleanupSignals)
138
+ process.once(signal, parentCleanupHandler);
139
+ process.once('beforeExit', parentCleanupHandler);
140
+ process.once('exit', parentCleanupHandler);
141
+ }
142
+ const removeParentCleanupHandlers = () => {
143
+ if (!options.cleanupOnParentExit)
144
+ return;
145
+ for (const signal of cleanupSignals)
146
+ process.off(signal, parentCleanupHandler);
147
+ process.off('beforeExit', parentCleanupHandler);
148
+ process.off('exit', parentCleanupHandler);
149
+ };
150
+ const finish = (result) => {
151
+ if (settled)
152
+ return;
153
+ settled = true;
154
+ if (timeoutTimer)
155
+ clearTimeout(timeoutTimer);
156
+ if (sigkillTimer)
157
+ clearTimeout(sigkillTimer);
158
+ if (processLimitTimer)
159
+ clearInterval(processLimitTimer);
160
+ removeParentCleanupHandlers();
161
+ resolve({ ...result, stdout, stderr, timedOut, processLimitExceeded, outputLimitExceeded });
162
+ };
163
+ const appendBoundedOutput = (current, chunk) => {
164
+ if (outputLimitExceeded)
165
+ return current;
166
+ if (maxOutputBytes === undefined)
167
+ return current + chunk;
168
+ const currentBytes = Buffer.byteLength(current, encoding);
169
+ const chunkBytes = Buffer.byteLength(chunk, encoding);
170
+ if (currentBytes + chunkBytes <= maxOutputBytes)
171
+ return current + chunk;
172
+ outputLimitExceeded = true;
173
+ terminate();
174
+ const remaining = Math.max(0, maxOutputBytes - currentBytes);
175
+ return current + Buffer.from(chunk, encoding).subarray(0, remaining).toString(encoding);
176
+ };
177
+ child.stdout.setEncoding(encoding);
178
+ child.stderr.setEncoding(encoding);
179
+ child.stdout.on('data', (chunk) => {
180
+ stdout = appendBoundedOutput(stdout, chunk);
181
+ });
182
+ child.stderr.on('data', (chunk) => {
183
+ stderr = appendBoundedOutput(stderr, chunk);
184
+ });
185
+ const sweepProcessGroupAfterParentExit = () => {
186
+ if (platform === 'win32')
187
+ return;
188
+ killProcessTree(child, platform, killSignal);
189
+ const residualSigkillTimer = setTimeout(() => {
190
+ killProcessTree(child, platform, 'SIGKILL');
191
+ }, sigkillGraceMs);
192
+ residualSigkillTimer.unref?.();
193
+ };
194
+ child.on('error', (error) => {
195
+ finish({ status: null, signal: null, error });
196
+ });
197
+ child.on('exit', () => {
198
+ // `close` waits for stdio EOF, so a direct wrapper that exits while a
199
+ // grandchild keeps inherited stdout/stderr open can otherwise sit until
200
+ // timeout. Sweep as soon as the direct parent exits, then let `close`
201
+ // report the parent's status and captured output.
202
+ sweepProcessGroupAfterParentExit();
203
+ });
204
+ child.on('close', (status, signal) => {
205
+ finish({ status, signal });
206
+ });
207
+ if (timeoutMs !== undefined) {
208
+ timeoutTimer = setTimeout(() => {
209
+ if (settled)
210
+ return;
211
+ timedOut = true;
212
+ terminate();
213
+ }, timeoutMs);
214
+ }
215
+ if (platform === 'linux' && maxProcessCount !== undefined) {
216
+ processLimitTimer = setInterval(() => {
217
+ if (settled || child.pid === undefined)
218
+ return;
219
+ const descendants = countDescendantsLinux(child.pid);
220
+ if (descendants === undefined)
221
+ return;
222
+ if (descendants + 1 > maxProcessCount) {
223
+ processLimitExceeded = true;
224
+ terminate();
225
+ }
226
+ }, processLimitPollMs);
227
+ }
228
+ });
229
+ }
230
+ //# sourceMappingURL=process-tree.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"process-tree.js","sourceRoot":"","sources":["../../src/runtime/process-tree.ts"],"names":[],"mappings":"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"}
@@ -1,5 +1,5 @@
1
1
  import assert from "node:assert/strict";
2
- import { execFileSync } from "node:child_process";
2
+ import { execFileSync, spawnSync } from "node:child_process";
3
3
  import { existsSync } from "node:fs";
4
4
  import { chmod, mkdir, mkdtemp, readFile, readdir, rm, writeFile } from "node:fs/promises";
5
5
  import { tmpdir } from "node:os";
@@ -14,6 +14,8 @@ import { writeSessionStart } from "../../hooks/session.js";
14
14
  import { resetTriageConfigCache } from "../../hooks/triage-config.js";
15
15
  import { executeStateOperation } from "../../state/operations.js";
16
16
  import { OMX_TMUX_HUD_OWNER_ENV } from "../../hud/reconcile.js";
17
+ import { writePage } from "../../wiki/storage.js";
18
+ import { WIKI_SCHEMA_VERSION } from "../../wiki/types.js";
17
19
  function nativeHookScriptPath() {
18
20
  return join(process.cwd(), "dist", "scripts", "codex-native-hook.js");
19
21
  }
@@ -141,6 +143,8 @@ const TEAM_ENV_KEYS = [
141
143
  "OMX_TEAM_STATE_ROOT",
142
144
  "OMX_TEAM_LEADER_CWD",
143
145
  "OMX_SESSION_ID",
146
+ "OMX_ROOT",
147
+ "OMX_STATE_ROOT",
144
148
  "SESSION_ID",
145
149
  "OMX_QUESTION_RETURN_PANE",
146
150
  "OMX_LEADER_PANE_ID",
@@ -174,6 +178,8 @@ describe("codex native hook config", () => {
174
178
  "PreToolUse",
175
179
  "PostToolUse",
176
180
  "UserPromptSubmit",
181
+ "PreCompact",
182
+ "PostCompact",
177
183
  "Stop",
178
184
  ]);
179
185
  const sessionStart = config.hooks.SessionStart[0];
@@ -322,13 +328,132 @@ describe("codex native hook dispatch", () => {
322
328
  await rm(cwd, { recursive: true, force: true });
323
329
  }
324
330
  });
331
+ it("logs Stop dispatch failures without foreground stderr noise", async () => {
332
+ const cwd = await mkdtemp(join(tmpdir(), "omx-native-hook-cli-stop-dispatch-silent-"));
333
+ try {
334
+ const result = spawnSync(process.execPath, [nativeHookScriptPath()], {
335
+ cwd,
336
+ input: JSON.stringify({
337
+ hook_event_name: "Stop",
338
+ cwd,
339
+ session_id: "sess-cli-stop-dispatch-silent",
340
+ thread_id: "thread-cli-stop-dispatch-silent",
341
+ turn_id: "turn-cli-stop-dispatch-silent",
342
+ }),
343
+ encoding: "utf-8",
344
+ stdio: ["pipe", "pipe", "pipe"],
345
+ env: {
346
+ ...process.env,
347
+ NODE_ENV: "test",
348
+ OMX_NATIVE_HOOK_TEST_THROW_STOP_DISPATCH: "1",
349
+ },
350
+ });
351
+ assert.equal(result.status, 0, result.stderr || result.stdout);
352
+ assert.equal(result.stderr, "");
353
+ const output = parseSingleJsonStdout(result.stdout);
354
+ assert.equal(output.stopReason, "native_stop_dispatch_failure");
355
+ const logFiles = await readdir(join(cwd, ".omx", "logs"));
356
+ assert.equal(logFiles.some((name) => /^native-hook-\d{4}-\d{2}-\d{2}\.jsonl$/.test(name)), true);
357
+ }
358
+ finally {
359
+ await rm(cwd, { recursive: true, force: true });
360
+ }
361
+ });
362
+ it("keeps non-Stop dispatch failures fail-closed without foreground stderr noise", async () => {
363
+ const cwd = await mkdtemp(join(tmpdir(), "omx-native-hook-cli-pretool-dispatch-silent-"));
364
+ try {
365
+ const result = spawnSync(process.execPath, [nativeHookScriptPath()], {
366
+ cwd,
367
+ input: JSON.stringify({
368
+ hook_event_name: "PreToolUse",
369
+ cwd,
370
+ session_id: "sess-cli-pretool-dispatch-silent",
371
+ thread_id: "thread-cli-pretool-dispatch-silent",
372
+ turn_id: "turn-cli-pretool-dispatch-silent",
373
+ tool_name: "Bash",
374
+ tool_input: { command: "pwd" },
375
+ }),
376
+ encoding: "utf-8",
377
+ stdio: ["pipe", "pipe", "pipe"],
378
+ env: {
379
+ ...process.env,
380
+ NODE_ENV: "test",
381
+ OMX_NATIVE_HOOK_TEST_THROW_DISPATCH: "1",
382
+ },
383
+ });
384
+ assert.equal(result.status, 1);
385
+ assert.equal(result.stdout, "");
386
+ assert.equal(result.stderr, "");
387
+ const logFiles = await readdir(join(cwd, ".omx", "logs"));
388
+ assert.equal(logFiles.some((name) => /^native-hook-\d{4}-\d{2}-\d{2}\.jsonl$/.test(name)), true);
389
+ }
390
+ finally {
391
+ await rm(cwd, { recursive: true, force: true });
392
+ }
393
+ });
325
394
  it("maps Codex events onto OMX logical surfaces", () => {
326
395
  assert.equal(mapCodexHookEventToOmxEvent("SessionStart"), "session-start");
327
396
  assert.equal(mapCodexHookEventToOmxEvent("UserPromptSubmit"), "keyword-detector");
328
397
  assert.equal(mapCodexHookEventToOmxEvent("PreToolUse"), "pre-tool-use");
329
398
  assert.equal(mapCodexHookEventToOmxEvent("PostToolUse"), "post-tool-use");
399
+ assert.equal(mapCodexHookEventToOmxEvent("PreCompact"), "pre-compact");
400
+ assert.equal(mapCodexHookEventToOmxEvent("PostCompact"), "post-compact");
330
401
  assert.equal(mapCodexHookEventToOmxEvent("Stop"), "stop");
331
402
  });
403
+ it("returns bounded wiki context for PreCompact", async () => {
404
+ const cwd = await mkdtemp(join(tmpdir(), "omx-native-hook-precompact-"));
405
+ try {
406
+ writePage(cwd, {
407
+ filename: "architecture.md",
408
+ frontmatter: {
409
+ title: "Architecture",
410
+ tags: ["architecture"],
411
+ created: "2026-05-08T00:00:00.000Z",
412
+ updated: "2026-05-08T00:00:00.000Z",
413
+ sources: [],
414
+ links: [],
415
+ category: "architecture",
416
+ confidence: "high",
417
+ schemaVersion: WIKI_SCHEMA_VERSION,
418
+ },
419
+ content: "\n# Architecture\n\nCompaction-relevant architecture note.\n",
420
+ });
421
+ const result = await dispatchCodexNativeHook({
422
+ hook_event_name: "PreCompact",
423
+ cwd,
424
+ session_id: "sess-precompact",
425
+ });
426
+ assert.equal(result.hookEventName, "PreCompact");
427
+ assert.equal(result.omxEventName, "pre-compact");
428
+ const additionalContext = result.outputJson
429
+ ?.hookSpecificOutput?.additionalContext ?? "";
430
+ assert.match(additionalContext, /Wiki: 1 pages/);
431
+ assert.match(additionalContext, /architecture/);
432
+ }
433
+ finally {
434
+ await rm(cwd, { recursive: true, force: true });
435
+ }
436
+ });
437
+ it("returns a PostCompact nudge to write compaction artifacts to omx_wiki", async () => {
438
+ const cwd = await mkdtemp(join(tmpdir(), "omx-native-hook-postcompact-"));
439
+ try {
440
+ const result = await dispatchCodexNativeHook({
441
+ hook_event_name: "PostCompact",
442
+ cwd,
443
+ session_id: "sess-postcompact",
444
+ });
445
+ assert.equal(result.hookEventName, "PostCompact");
446
+ assert.equal(result.omxEventName, "post-compact");
447
+ const additionalContext = result.outputJson
448
+ ?.hookSpecificOutput?.additionalContext ?? "";
449
+ assert.match(additionalContext, /PostCompact Nudge/);
450
+ assert.match(additionalContext, /omx_wiki/);
451
+ assert.match(additionalContext, /compaction artifacts/);
452
+ }
453
+ finally {
454
+ await rm(cwd, { recursive: true, force: true });
455
+ }
456
+ });
332
457
  it("writes SessionStart state against the long-lived session owner pid and injects environment context", async () => {
333
458
  const cwd = await mkdtemp(join(tmpdir(), "omx-native-hook-session-start-"));
334
459
  try {
@@ -798,7 +923,8 @@ describe("codex native hook dispatch", () => {
798
923
  assert.equal(result.skillState?.skill, "ralplan");
799
924
  assert.ok(result.outputJson, "UserPromptSubmit should emit developer context");
800
925
  assert.match(JSON.stringify(result.outputJson), /skill: ralplan activated and initial state initialized at \.omx\/state\/sessions\/sess-1\/ralplan-state\.json; write subsequent updates via omx_state MCP\./);
801
- const statePath = join(cwd, ".omx", "state", "skill-active-state.json");
926
+ assert.equal(existsSync(join(cwd, ".omx", "state", "skill-active-state.json")), false, "session-scoped keyword activation should not write root skill-active-state.json");
927
+ const statePath = join(cwd, ".omx", "state", "sessions", "sess-1", "skill-active-state.json");
802
928
  assert.equal(existsSync(statePath), true);
803
929
  const state = JSON.parse(await readFile(statePath, "utf-8"));
804
930
  assert.equal(state.skill, "ralplan");
@@ -6885,6 +7011,145 @@ exit 0
6885
7011
  await rm(cwd, { recursive: true, force: true });
6886
7012
  }
6887
7013
  });
7014
+ it("clears stale root skill-active state when current session ralplan is terminal", async () => {
7015
+ const cwd = await mkdtemp(join(tmpdir(), "omx-native-hook-stop-stale-root-skill-terminal-"));
7016
+ try {
7017
+ const stateDir = join(cwd, ".omx", "state");
7018
+ const sessionId = "sess-stop-terminal-ralplan";
7019
+ await mkdir(join(stateDir, "sessions", sessionId), { recursive: true });
7020
+ await writeJson(join(stateDir, "session.json"), { session_id: sessionId });
7021
+ await writeJson(join(stateDir, "sessions", sessionId, "ralplan-state.json"), {
7022
+ active: false,
7023
+ mode: "ralplan",
7024
+ current_phase: "completed",
7025
+ lifecycle_outcome: "finished",
7026
+ run_outcome: "finish",
7027
+ final_artifact: "proposed_plan",
7028
+ });
7029
+ await writeJson(join(stateDir, "skill-active-state.json"), {
7030
+ active: true,
7031
+ skill: "ultrawork",
7032
+ phase: "planning",
7033
+ source: "keyword-detector",
7034
+ active_skills: [
7035
+ { skill: "ultrawork", phase: "planning", active: true },
7036
+ ],
7037
+ });
7038
+ const result = await dispatchCodexNativeHook({
7039
+ hook_event_name: "Stop",
7040
+ cwd,
7041
+ session_id: sessionId,
7042
+ thread_id: "thread-stop-terminal-ralplan",
7043
+ turn_id: "turn-stop-terminal-ralplan-1",
7044
+ last_assistant_message: "Done.",
7045
+ }, { cwd });
7046
+ assert.equal(result.omxEventName, "stop");
7047
+ assert.equal(result.outputJson, null);
7048
+ const rootSkillState = JSON.parse(await readFile(join(stateDir, "skill-active-state.json"), "utf-8"));
7049
+ assert.equal(rootSkillState.active, false);
7050
+ assert.deepEqual(rootSkillState.active_skills, []);
7051
+ assert.equal(rootSkillState.reconciliation_reason, "stop_hook_session_state_terminal");
7052
+ }
7053
+ finally {
7054
+ await rm(cwd, { recursive: true, force: true });
7055
+ }
7056
+ });
7057
+ it("preserves legitimate session-scoped ultrawork blocking while reconciling root skill-active state", async () => {
7058
+ const cwd = await mkdtemp(join(tmpdir(), "omx-native-hook-stop-active-root-skill-session-mode-"));
7059
+ try {
7060
+ const stateDir = join(cwd, ".omx", "state");
7061
+ const sessionId = "sess-stop-active-ultrawork";
7062
+ await mkdir(join(stateDir, "sessions", sessionId), { recursive: true });
7063
+ await writeJson(join(stateDir, "session.json"), { session_id: sessionId });
7064
+ await writeJson(join(stateDir, "sessions", sessionId, "ultrawork-state.json"), {
7065
+ active: true,
7066
+ mode: "ultrawork",
7067
+ current_phase: "executing",
7068
+ session_id: sessionId,
7069
+ });
7070
+ await writeJson(join(stateDir, "skill-active-state.json"), {
7071
+ active: true,
7072
+ skill: "ultrawork",
7073
+ phase: "planning",
7074
+ source: "keyword-detector",
7075
+ active_skills: [
7076
+ { skill: "ultrawork", phase: "planning", active: true, session_id: sessionId },
7077
+ ],
7078
+ });
7079
+ const result = await dispatchCodexNativeHook({
7080
+ hook_event_name: "Stop",
7081
+ cwd,
7082
+ session_id: sessionId,
7083
+ thread_id: "thread-stop-active-ultrawork",
7084
+ turn_id: "turn-stop-active-ultrawork-1",
7085
+ }, { cwd });
7086
+ assert.equal(result.omxEventName, "stop");
7087
+ assert.deepEqual(result.outputJson, {
7088
+ decision: "block",
7089
+ reason: "OMX ultrawork is still active (phase: executing); continue the task and gather fresh verification evidence before stopping.",
7090
+ stopReason: "ultrawork_executing",
7091
+ systemMessage: "OMX ultrawork is still active (phase: executing).",
7092
+ });
7093
+ const rootSkillState = JSON.parse(await readFile(join(stateDir, "skill-active-state.json"), "utf-8"));
7094
+ assert.equal(rootSkillState.active, true);
7095
+ assert.deepEqual(rootSkillState.active_skills?.map((entry) => entry.skill), ["ultrawork"]);
7096
+ }
7097
+ finally {
7098
+ await rm(cwd, { recursive: true, force: true });
7099
+ }
7100
+ });
7101
+ it("reconciles stale root skill-active state under OMX_ROOT boxed state", async () => {
7102
+ const cwd = await mkdtemp(join(tmpdir(), "omx-native-hook-stop-boxed-source-"));
7103
+ const omxRoot = await mkdtemp(join(tmpdir(), "omx-native-hook-stop-boxed-root-"));
7104
+ const previousOmxRoot = process.env.OMX_ROOT;
7105
+ try {
7106
+ process.env.OMX_ROOT = omxRoot;
7107
+ const stateDir = join(omxRoot, ".omx", "state");
7108
+ const sourceStateDir = join(cwd, ".omx", "state");
7109
+ const sessionId = "sess-stop-boxed-ralplan";
7110
+ await mkdir(join(stateDir, "sessions", sessionId), { recursive: true });
7111
+ await writeJson(join(stateDir, "session.json"), { session_id: sessionId });
7112
+ await writeJson(join(stateDir, "sessions", sessionId, "ralplan-state.json"), {
7113
+ active: false,
7114
+ mode: "ralplan",
7115
+ current_phase: "completed",
7116
+ lifecycle_outcome: "finished",
7117
+ run_outcome: "finish",
7118
+ });
7119
+ await writeJson(join(stateDir, "skill-active-state.json"), {
7120
+ active: true,
7121
+ skill: "ultrawork",
7122
+ phase: "planning",
7123
+ source: "keyword-detector",
7124
+ active_skills: [
7125
+ { skill: "ultrawork", phase: "planning", active: true },
7126
+ ],
7127
+ });
7128
+ const result = await dispatchCodexNativeHook({
7129
+ hook_event_name: "Stop",
7130
+ cwd,
7131
+ session_id: sessionId,
7132
+ thread_id: "thread-stop-boxed-ralplan",
7133
+ turn_id: "turn-stop-boxed-ralplan-1",
7134
+ last_assistant_message: "Done.",
7135
+ }, { cwd });
7136
+ assert.equal(result.omxEventName, "stop");
7137
+ assert.equal(result.outputJson, null);
7138
+ const boxedRootSkillState = JSON.parse(await readFile(join(stateDir, "skill-active-state.json"), "utf-8"));
7139
+ assert.equal(boxedRootSkillState.active, false);
7140
+ assert.deepEqual(boxedRootSkillState.active_skills, []);
7141
+ assert.equal(boxedRootSkillState.reconciliation_reason, "stop_hook_session_state_terminal");
7142
+ assert.equal(existsSync(join(sourceStateDir, "skill-active-state.json")), false);
7143
+ }
7144
+ finally {
7145
+ if (previousOmxRoot === undefined)
7146
+ delete process.env.OMX_ROOT;
7147
+ else
7148
+ process.env.OMX_ROOT = previousOmxRoot;
7149
+ await rm(cwd, { recursive: true, force: true });
7150
+ await rm(omxRoot, { recursive: true, force: true });
7151
+ }
7152
+ });
6888
7153
  it("auto-continues native Stop for permission-seeking prompts even outside OMX runtime", async () => {
6889
7154
  const cwd = await mkdtemp(join(tmpdir(), "omx-native-hook-auto-nudge-plain-session-"));
6890
7155
  try {