oh-my-opencode 4.5.1 → 4.6.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 (668) hide show
  1. package/.agents/skills/opencode-qa/SKILL.md +194 -0
  2. package/.agents/skills/opencode-qa/references/cli-commands.md +188 -0
  3. package/.agents/skills/opencode-qa/references/db-investigation.md +197 -0
  4. package/.agents/skills/opencode-qa/references/events-hooks.md +110 -0
  5. package/.agents/skills/opencode-qa/references/sdk.md +96 -0
  6. package/.agents/skills/opencode-qa/references/server-api.md +200 -0
  7. package/.agents/skills/opencode-qa/references/testing-harness.md +218 -0
  8. package/.agents/skills/opencode-qa/references/tui-tmux.md +52 -0
  9. package/.agents/skills/opencode-qa/scripts/db-session-by-id.sh +53 -0
  10. package/.agents/skills/opencode-qa/scripts/db-session-by-name.sh +57 -0
  11. package/.agents/skills/opencode-qa/scripts/db-session-by-text.sh +158 -0
  12. package/.agents/skills/opencode-qa/scripts/export-roundtrip.sh +57 -0
  13. package/.agents/skills/opencode-qa/scripts/lib/common.sh +216 -0
  14. package/.agents/skills/opencode-qa/scripts/server-smoke.sh +64 -0
  15. package/.agents/skills/opencode-qa/scripts/sse-hook-probe.sh +106 -0
  16. package/.agents/skills/opencode-qa/scripts/tui-smoke.sh +89 -0
  17. package/README.ja.md +43 -18
  18. package/README.ko.md +43 -18
  19. package/README.md +92 -24
  20. package/README.ru.md +43 -18
  21. package/README.zh-cn.md +55 -24
  22. package/bin/oh-my-opencode.js +49 -3
  23. package/bin/oh-my-opencode.test.ts +202 -0
  24. package/bin/platform.d.ts +7 -1
  25. package/bin/platform.js +23 -4
  26. package/bin/platform.test.ts +97 -10
  27. package/dist/cli/cleanup-command.d.ts +4 -0
  28. package/dist/cli/cleanup.d.ts +11 -0
  29. package/dist/cli/cli-program.d.ts +19 -0
  30. package/dist/cli/index.js +30085 -26385
  31. package/dist/cli/install-codex/codex-cache-bundled-mcps.d.ts +5 -0
  32. package/dist/cli/install-codex/codex-cache-command-shim.d.ts +1 -0
  33. package/dist/cli/install-codex/codex-cache-legacy-bins.d.ts +3 -0
  34. package/dist/cli/install-codex/codex-cache-local-dependencies.d.ts +1 -0
  35. package/dist/cli/install-codex/codex-cache-paths.d.ts +2 -0
  36. package/dist/cli/install-codex/codex-cache.d.ts +32 -0
  37. package/dist/cli/install-codex/codex-cleanup-config.d.ts +6 -0
  38. package/dist/cli/install-codex/codex-cleanup.d.ts +21 -0
  39. package/dist/cli/install-codex/codex-config-mcp.d.ts +1 -0
  40. package/dist/cli/install-codex/codex-config-permissions.d.ts +1 -0
  41. package/dist/cli/install-codex/codex-config-reasoning.d.ts +1 -0
  42. package/dist/cli/install-codex/codex-config-toml.d.ts +12 -0
  43. package/dist/cli/install-codex/codex-hook-trust.d.ts +6 -0
  44. package/dist/cli/install-codex/codex-installation-detection.d.ts +36 -0
  45. package/dist/cli/install-codex/codex-marketplace-snapshot.d.ts +15 -0
  46. package/dist/cli/install-codex/codex-marketplace.d.ts +9 -0
  47. package/dist/cli/install-codex/codex-multi-agent-v2-config.d.ts +1 -0
  48. package/dist/cli/install-codex/codex-package-layout.d.ts +1 -0
  49. package/dist/cli/install-codex/codex-process.d.ts +2 -0
  50. package/dist/cli/install-codex/codex-project-local-cleanup-best-effort.d.ts +7 -0
  51. package/dist/cli/install-codex/codex-project-local-cleanup.d.ts +35 -0
  52. package/dist/cli/install-codex/git-bash.d.ts +35 -0
  53. package/dist/cli/install-codex/index.d.ts +11 -0
  54. package/dist/cli/install-codex/install-codex.d.ts +17 -0
  55. package/dist/cli/install-codex/link-cached-plugin-agents.d.ts +12 -0
  56. package/dist/cli/install-codex/toml-section-editor.d.ts +11 -0
  57. package/dist/cli/install-codex/types.d.ts +76 -0
  58. package/dist/cli/run/event-state.d.ts +1 -0
  59. package/dist/cli/run/poll-for-completion.d.ts +1 -0
  60. package/dist/cli/run/prompt-start.d.ts +7 -0
  61. package/dist/cli/star-request.d.ts +11 -0
  62. package/dist/cli/tui-install-prompts.d.ts +3 -2
  63. package/dist/cli/types.d.ts +8 -0
  64. package/dist/config/schema/agent-names.d.ts +4 -1
  65. package/dist/config/schema/commands.d.ts +1 -2
  66. package/dist/config/schema/hooks.d.ts +0 -1
  67. package/dist/config/schema/oh-my-opencode-config.d.ts +5 -3
  68. package/dist/create-hooks.d.ts +0 -1
  69. package/dist/create-managers.d.ts +1 -0
  70. package/dist/features/background-agent/parent-wake-message-activity.d.ts +23 -0
  71. package/dist/features/background-agent/parent-wake-notifier.d.ts +0 -1
  72. package/dist/features/boulder-state/storage.d.ts +1 -1
  73. package/dist/features/builtin-commands/templates/refactor.d.ts +1 -1
  74. package/dist/features/builtin-commands/templates/remove-ai-slops.d.ts +2 -2
  75. package/dist/features/builtin-commands/types.d.ts +1 -1
  76. package/dist/features/builtin-skills/skill-file-loader.d.ts +4 -0
  77. package/dist/features/builtin-skills/skills/debugging.d.ts +2 -0
  78. package/dist/features/builtin-skills/skills/index.d.ts +5 -1
  79. package/dist/features/builtin-skills/skills/init-deep.d.ts +2 -0
  80. package/dist/features/builtin-skills/skills/remove-ai-slops.d.ts +2 -0
  81. package/dist/features/builtin-skills/skills/security-research.d.ts +2 -0
  82. package/dist/features/builtin-skills/skills/security-review.d.ts +2 -0
  83. package/dist/features/opencode-runtime-skills/index.d.ts +2 -0
  84. package/dist/features/opencode-runtime-skills/runtime-skill-config.d.ts +17 -0
  85. package/dist/features/opencode-runtime-skills/skill-markdown.d.ts +7 -0
  86. package/dist/features/opencode-runtime-skills/source-server.d.ts +8 -0
  87. package/dist/hooks/claude-code-hooks/hook-text.d.ts +2 -0
  88. package/dist/hooks/index.d.ts +0 -1
  89. package/dist/hooks/session-recovery/storage/latest-assistant-message.d.ts +5 -0
  90. package/dist/hooks/session-recovery/storage/thinking-prepend.d.ts +3 -0
  91. package/dist/hooks/thinking-block-validator/hook.d.ts +0 -18
  92. package/dist/hooks/todo-continuation-enforcer/pending-question-detection.d.ts +4 -0
  93. package/dist/index.js +1949 -1913
  94. package/dist/oh-my-opencode.schema.json +4 -2
  95. package/dist/plugin/hooks/create-core-hooks.d.ts +0 -1
  96. package/dist/plugin/hooks/create-session-hooks.d.ts +1 -2
  97. package/dist/plugin/messages-transform.d.ts +8 -1
  98. package/dist/plugin/user-abort-interrupted-recovery-guard.d.ts +6 -0
  99. package/dist/plugin-handlers/config-handler.d.ts +2 -1
  100. package/dist/shared/external-plugin-detector.d.ts +8 -0
  101. package/dist/shared/prompt-async-gate/message-inspection-error.d.ts +1 -0
  102. package/dist/shared/prompt-async-gate/pending-tool-turn.d.ts +1 -0
  103. package/dist/shared/prompt-async-gate/prompt-message-state.d.ts +8 -0
  104. package/dist/shared/prompt-async-gate/recent-dispatches.d.ts +14 -0
  105. package/dist/shared/prompt-async-gate/semantic-dedupe.d.ts +7 -0
  106. package/dist/shared/prompt-async-gate/session-idle-dispatch.d.ts +1 -0
  107. package/dist/shared/prompt-async-gate/timing.d.ts +1 -0
  108. package/dist/shared/prompt-async-gate/types.d.ts +2 -0
  109. package/dist/shared/prompt-async-gate.d.ts +1 -1
  110. package/dist/shared/prompt-timeout-context.d.ts +2 -0
  111. package/dist/testing/create-plugin-module.d.ts +5 -1
  112. package/dist/tools/delegate-task/sync-prompt-sender.d.ts +2 -2
  113. package/package.json +38 -18
  114. package/packages/ast-grep-mcp/dist/cli.js +245 -40
  115. package/packages/git-bash-mcp/dist/cli.js +367 -0
  116. package/packages/lsp-tools-mcp/dist/cli.js +1 -1
  117. package/packages/omo-codex/marketplace.json +17 -0
  118. package/packages/omo-codex/plugin/.codex-plugin/plugin.json +35 -0
  119. package/packages/omo-codex/plugin/.mcp.json +25 -0
  120. package/packages/omo-codex/plugin/README.md +13 -0
  121. package/packages/omo-codex/plugin/components/comment-checker/.gitattributes +13 -0
  122. package/packages/omo-codex/plugin/components/comment-checker/.github/CODEOWNERS +12 -0
  123. package/packages/omo-codex/plugin/components/comment-checker/.github/ISSUE_TEMPLATE/bug.yml +40 -0
  124. package/packages/omo-codex/plugin/components/comment-checker/.github/ISSUE_TEMPLATE/feature.yml +27 -0
  125. package/packages/omo-codex/plugin/components/comment-checker/.github/branch-ruleset.json +45 -0
  126. package/packages/omo-codex/plugin/components/comment-checker/.github/dependabot.yml +16 -0
  127. package/packages/omo-codex/plugin/components/comment-checker/.github/pull_request_template.md +19 -0
  128. package/packages/omo-codex/plugin/components/comment-checker/.github/workflows/ci.yml +47 -0
  129. package/packages/omo-codex/plugin/components/comment-checker/.github/workflows/publish.yml +51 -0
  130. package/packages/omo-codex/plugin/components/comment-checker/AGENTS.md +35 -0
  131. package/packages/omo-codex/plugin/components/comment-checker/CHANGELOG.md +33 -0
  132. package/packages/omo-codex/plugin/components/comment-checker/LICENSE +21 -0
  133. package/packages/omo-codex/plugin/components/comment-checker/NOTICE +6 -0
  134. package/packages/omo-codex/plugin/components/comment-checker/README.md +87 -0
  135. package/packages/omo-codex/plugin/components/comment-checker/biome.json +48 -0
  136. package/packages/omo-codex/plugin/components/comment-checker/hooks/hooks.json +17 -0
  137. package/packages/omo-codex/plugin/components/comment-checker/package.json +57 -0
  138. package/packages/omo-codex/plugin/components/comment-checker/skills/comment-checker/SKILL.md +16 -0
  139. package/packages/omo-codex/plugin/components/comment-checker/src/cli.ts +12 -0
  140. package/packages/omo-codex/plugin/components/comment-checker/src/codex-hook.ts +205 -0
  141. package/packages/omo-codex/plugin/components/comment-checker/src/core.ts +361 -0
  142. package/packages/omo-codex/plugin/components/comment-checker/src/runner.ts +195 -0
  143. package/packages/omo-codex/plugin/components/comment-checker/test/codex-hook-newline.test.ts +52 -0
  144. package/packages/omo-codex/plugin/components/comment-checker/test/codex-hook.test.ts +368 -0
  145. package/packages/omo-codex/plugin/components/comment-checker/test/fixtures/post-tool-use.json +15 -0
  146. package/packages/omo-codex/plugin/components/comment-checker/test/package-smoke.test.ts +93 -0
  147. package/packages/omo-codex/plugin/components/comment-checker/test/runner.test.ts +66 -0
  148. package/packages/omo-codex/plugin/components/comment-checker/tsconfig.build.json +12 -0
  149. package/packages/omo-codex/plugin/components/comment-checker/tsconfig.json +27 -0
  150. package/packages/omo-codex/plugin/components/comment-checker/vitest.config.ts +9 -0
  151. package/packages/omo-codex/plugin/components/git-bash/hooks/hooks.json +29 -0
  152. package/packages/omo-codex/plugin/components/git-bash/package.json +23 -0
  153. package/packages/omo-codex/plugin/components/git-bash/src/cli.ts +33 -0
  154. package/packages/omo-codex/plugin/components/git-bash/src/codex-hook.ts +180 -0
  155. package/packages/omo-codex/plugin/components/git-bash/src/index.ts +10 -0
  156. package/packages/omo-codex/plugin/components/git-bash/test/codex-hook.test.ts +195 -0
  157. package/packages/omo-codex/plugin/components/git-bash/tsconfig.build.json +13 -0
  158. package/packages/omo-codex/plugin/components/git-bash/tsconfig.json +25 -0
  159. package/packages/omo-codex/plugin/components/lsp/.gitattributes +13 -0
  160. package/packages/omo-codex/plugin/components/lsp/.github/CODEOWNERS +1 -0
  161. package/packages/omo-codex/plugin/components/lsp/.github/ISSUE_TEMPLATE/bug.yml +26 -0
  162. package/packages/omo-codex/plugin/components/lsp/.github/ISSUE_TEMPLATE/feature.yml +19 -0
  163. package/packages/omo-codex/plugin/components/lsp/.github/branch-ruleset.json +45 -0
  164. package/packages/omo-codex/plugin/components/lsp/.github/dependabot.yml +11 -0
  165. package/packages/omo-codex/plugin/components/lsp/.github/pull_request_template.md +11 -0
  166. package/packages/omo-codex/plugin/components/lsp/.github/workflows/ci.yml +56 -0
  167. package/packages/omo-codex/plugin/components/lsp/.github/workflows/publish.yml +60 -0
  168. package/packages/omo-codex/plugin/components/lsp/.mcp.json +9 -0
  169. package/packages/omo-codex/plugin/components/lsp/AGENTS.md +25 -0
  170. package/packages/omo-codex/plugin/components/lsp/CHANGELOG.md +25 -0
  171. package/packages/omo-codex/plugin/components/lsp/LICENSE +21 -0
  172. package/packages/omo-codex/plugin/components/lsp/NOTICE +3 -0
  173. package/packages/omo-codex/plugin/components/lsp/README.md +148 -0
  174. package/packages/omo-codex/plugin/components/lsp/biome.json +48 -0
  175. package/packages/omo-codex/plugin/components/lsp/hooks/hooks.json +17 -0
  176. package/packages/omo-codex/plugin/components/lsp/package.json +64 -0
  177. package/packages/omo-codex/plugin/components/lsp/scripts/build-lsp-tools.mjs +46 -0
  178. package/packages/omo-codex/plugin/components/lsp/scripts/build-lsp-tools.test.mjs +104 -0
  179. package/packages/omo-codex/plugin/components/lsp/scripts/clean-dist.mjs +5 -0
  180. package/packages/omo-codex/plugin/components/lsp/scripts/test.mjs +8 -0
  181. package/packages/omo-codex/plugin/components/lsp/skills/lsp/SKILL.md +35 -0
  182. package/packages/omo-codex/plugin/components/lsp/src/cli.ts +44 -0
  183. package/packages/omo-codex/plugin/components/lsp/src/codex-hook-cli.ts +33 -0
  184. package/packages/omo-codex/plugin/components/lsp/src/codex-hook.ts +277 -0
  185. package/packages/omo-codex/plugin/components/lsp/test/codex-hook-cli.test.ts +28 -0
  186. package/packages/omo-codex/plugin/components/lsp/test/codex-hook-errors.test.ts +55 -0
  187. package/packages/omo-codex/plugin/components/lsp/test/codex-hook.test.ts +358 -0
  188. package/packages/omo-codex/plugin/components/lsp/test/fixtures/broken.py +1 -0
  189. package/packages/omo-codex/plugin/components/lsp/test/fixtures/post-tool-use.json +15 -0
  190. package/packages/omo-codex/plugin/components/lsp/test/package-smoke.test.ts +155 -0
  191. package/packages/omo-codex/plugin/components/lsp/tsconfig.build.json +12 -0
  192. package/packages/omo-codex/plugin/components/lsp/tsconfig.json +27 -0
  193. package/packages/omo-codex/plugin/components/lsp/vitest.config.ts +9 -0
  194. package/packages/omo-codex/plugin/components/rules/.codex-plugin/plugin.json +3 -0
  195. package/packages/omo-codex/plugin/components/rules/.gitattributes +13 -0
  196. package/packages/omo-codex/plugin/components/rules/.github/CODEOWNERS +12 -0
  197. package/packages/omo-codex/plugin/components/rules/.github/ISSUE_TEMPLATE/bug.yml +49 -0
  198. package/packages/omo-codex/plugin/components/rules/.github/ISSUE_TEMPLATE/feature.yml +27 -0
  199. package/packages/omo-codex/plugin/components/rules/.github/branch-ruleset.json +45 -0
  200. package/packages/omo-codex/plugin/components/rules/.github/dependabot.yml +16 -0
  201. package/packages/omo-codex/plugin/components/rules/.github/pull_request_template.md +20 -0
  202. package/packages/omo-codex/plugin/components/rules/.github/workflows/ci.yml +47 -0
  203. package/packages/omo-codex/plugin/components/rules/.github/workflows/publish.yml +51 -0
  204. package/packages/omo-codex/plugin/components/rules/AGENTS.md +34 -0
  205. package/packages/omo-codex/plugin/components/rules/CHANGELOG.md +19 -0
  206. package/packages/omo-codex/plugin/components/rules/LICENSE +21 -0
  207. package/packages/omo-codex/plugin/components/rules/NOTICE +15 -0
  208. package/packages/omo-codex/plugin/components/rules/README.md +124 -0
  209. package/packages/omo-codex/plugin/components/rules/biome.json +48 -0
  210. package/packages/omo-codex/plugin/components/rules/bundled-rules/hephaestus.md +209 -0
  211. package/packages/omo-codex/plugin/components/rules/bundled-rules/windows-git-bash.md +10 -0
  212. package/packages/omo-codex/plugin/components/rules/hooks/hooks.json +54 -0
  213. package/packages/omo-codex/plugin/components/rules/package.json +62 -0
  214. package/packages/omo-codex/plugin/components/rules/scripts/bench-codex-rules.mjs +268 -0
  215. package/packages/omo-codex/plugin/components/rules/skills/rules/SKILL.md +34 -0
  216. package/packages/omo-codex/plugin/components/rules/src/cli.ts +143 -0
  217. package/packages/omo-codex/plugin/components/rules/src/codex-hook-options.ts +4 -0
  218. package/packages/omo-codex/plugin/components/rules/src/codex-hook.ts +238 -0
  219. package/packages/omo-codex/plugin/components/rules/src/config.ts +107 -0
  220. package/packages/omo-codex/plugin/components/rules/src/context-pressure.ts +26 -0
  221. package/packages/omo-codex/plugin/components/rules/src/debug-log.ts +65 -0
  222. package/packages/omo-codex/plugin/components/rules/src/dynamic-target-fingerprints.ts +98 -0
  223. package/packages/omo-codex/plugin/components/rules/src/hook-output.ts +19 -0
  224. package/packages/omo-codex/plugin/components/rules/src/path-utils.ts +29 -0
  225. package/packages/omo-codex/plugin/components/rules/src/persistent-cache.ts +234 -0
  226. package/packages/omo-codex/plugin/components/rules/src/post-compact-budget.ts +104 -0
  227. package/packages/omo-codex/plugin/components/rules/src/post-compact-claim.ts +13 -0
  228. package/packages/omo-codex/plugin/components/rules/src/post-compact-state.ts +45 -0
  229. package/packages/omo-codex/plugin/components/rules/src/rules/cache.ts +64 -0
  230. package/packages/omo-codex/plugin/components/rules/src/rules/constants.ts +115 -0
  231. package/packages/omo-codex/plugin/components/rules/src/rules/engine.ts +535 -0
  232. package/packages/omo-codex/plugin/components/rules/src/rules/errors.ts +13 -0
  233. package/packages/omo-codex/plugin/components/rules/src/rules/finder-cache.ts +73 -0
  234. package/packages/omo-codex/plugin/components/rules/src/rules/finder-paths.ts +47 -0
  235. package/packages/omo-codex/plugin/components/rules/src/rules/finder-sources.ts +50 -0
  236. package/packages/omo-codex/plugin/components/rules/src/rules/finder.ts +207 -0
  237. package/packages/omo-codex/plugin/components/rules/src/rules/formatter.ts +123 -0
  238. package/packages/omo-codex/plugin/components/rules/src/rules/matcher.ts +142 -0
  239. package/packages/omo-codex/plugin/components/rules/src/rules/ordering.ts +33 -0
  240. package/packages/omo-codex/plugin/components/rules/src/rules/parser.ts +326 -0
  241. package/packages/omo-codex/plugin/components/rules/src/rules/plugin-root.ts +55 -0
  242. package/packages/omo-codex/plugin/components/rules/src/rules/project-root.ts +30 -0
  243. package/packages/omo-codex/plugin/components/rules/src/rules/scanner.ts +162 -0
  244. package/packages/omo-codex/plugin/components/rules/src/rules/truncator.ts +67 -0
  245. package/packages/omo-codex/plugin/components/rules/src/rules/types.ts +141 -0
  246. package/packages/omo-codex/plugin/components/rules/src/rules-engine-factory.ts +24 -0
  247. package/packages/omo-codex/plugin/components/rules/src/session-state-lock.ts +47 -0
  248. package/packages/omo-codex/plugin/components/rules/src/static-injection.ts +56 -0
  249. package/packages/omo-codex/plugin/components/rules/src/tool-paths.ts +192 -0
  250. package/packages/omo-codex/plugin/components/rules/src/transcript-rule-filter.ts +44 -0
  251. package/packages/omo-codex/plugin/components/rules/src/transcript-search.ts +108 -0
  252. package/packages/omo-codex/plugin/components/rules/test/bundled-rules-priority.test.ts +107 -0
  253. package/packages/omo-codex/plugin/components/rules/test/bundled-rules.test.ts +268 -0
  254. package/packages/omo-codex/plugin/components/rules/test/codex-hook-context-pressure.test.ts +243 -0
  255. package/packages/omo-codex/plugin/components/rules/test/codex-hook-performance.test.ts +99 -0
  256. package/packages/omo-codex/plugin/components/rules/test/codex-hook-post-compact-budget.test.ts +132 -0
  257. package/packages/omo-codex/plugin/components/rules/test/codex-hook-post-compact-context.test.ts +156 -0
  258. package/packages/omo-codex/plugin/components/rules/test/codex-hook-post-compact-dedup.test.ts +299 -0
  259. package/packages/omo-codex/plugin/components/rules/test/codex-hook-post-compact-lock.test.ts +46 -0
  260. package/packages/omo-codex/plugin/components/rules/test/codex-hook-post-compact-process.test.ts +83 -0
  261. package/packages/omo-codex/plugin/components/rules/test/codex-hook.test.ts +667 -0
  262. package/packages/omo-codex/plugin/components/rules/test/engine.test.ts +192 -0
  263. package/packages/omo-codex/plugin/components/rules/test/finder.test.ts +102 -0
  264. package/packages/omo-codex/plugin/components/rules/test/formatter.test.ts +168 -0
  265. package/packages/omo-codex/plugin/components/rules/test/hook-output.test.ts +42 -0
  266. package/packages/omo-codex/plugin/components/rules/test/matcher.test.ts +206 -0
  267. package/packages/omo-codex/plugin/components/rules/test/package-smoke.test.ts +153 -0
  268. package/packages/omo-codex/plugin/components/rules/test/persistent-cache.test.ts +63 -0
  269. package/packages/omo-codex/plugin/components/rules/test/post-compact-budget.test.ts +172 -0
  270. package/packages/omo-codex/plugin/components/rules/test/post-compact-test-fixture.ts +196 -0
  271. package/packages/omo-codex/plugin/components/rules/test/scanner.test.ts +63 -0
  272. package/packages/omo-codex/plugin/components/rules/test/tool-paths.test.ts +198 -0
  273. package/packages/omo-codex/plugin/components/rules/test/windows-git-bash-bundled-rule.test.ts +97 -0
  274. package/packages/omo-codex/plugin/components/rules/tsconfig.build.json +12 -0
  275. package/packages/omo-codex/plugin/components/rules/tsconfig.json +27 -0
  276. package/packages/omo-codex/plugin/components/rules/vitest.config.ts +8 -0
  277. package/packages/omo-codex/plugin/components/start-work-continuation/.gitattributes +13 -0
  278. package/packages/omo-codex/plugin/components/start-work-continuation/AGENTS.md +43 -0
  279. package/packages/omo-codex/plugin/components/start-work-continuation/CHANGELOG.md +5 -0
  280. package/packages/omo-codex/plugin/components/start-work-continuation/LICENSE +21 -0
  281. package/packages/omo-codex/plugin/components/start-work-continuation/NOTICE +5 -0
  282. package/packages/omo-codex/plugin/components/start-work-continuation/README.md +55 -0
  283. package/packages/omo-codex/plugin/components/start-work-continuation/biome.json +48 -0
  284. package/packages/omo-codex/plugin/components/start-work-continuation/directive.md +52 -0
  285. package/packages/omo-codex/plugin/components/start-work-continuation/hooks/hooks.json +28 -0
  286. package/packages/omo-codex/plugin/components/start-work-continuation/package.json +53 -0
  287. package/packages/omo-codex/plugin/components/start-work-continuation/src/boulder-reader.ts +167 -0
  288. package/packages/omo-codex/plugin/components/start-work-continuation/src/cli.ts +52 -0
  289. package/packages/omo-codex/plugin/components/start-work-continuation/src/codex-hook.ts +66 -0
  290. package/packages/omo-codex/plugin/components/start-work-continuation/src/directive.ts +6 -0
  291. package/packages/omo-codex/plugin/components/start-work-continuation/src/index.ts +5 -0
  292. package/packages/omo-codex/plugin/components/start-work-continuation/src/types.ts +23 -0
  293. package/packages/omo-codex/plugin/components/start-work-continuation/test/boulder-reader.test.ts +63 -0
  294. package/packages/omo-codex/plugin/components/start-work-continuation/test/cli.test.ts +124 -0
  295. package/packages/omo-codex/plugin/components/start-work-continuation/test/codex-hook.test.ts +182 -0
  296. package/packages/omo-codex/plugin/components/start-work-continuation/test/fixtures/boulder-completed.json +19 -0
  297. package/packages/omo-codex/plugin/components/start-work-continuation/test/fixtures/boulder-mixed-platforms.json +27 -0
  298. package/packages/omo-codex/plugin/components/start-work-continuation/test/fixtures/boulder-single-codex-work.json +19 -0
  299. package/packages/omo-codex/plugin/components/start-work-continuation/test/fixtures/plan-all-done.md +5 -0
  300. package/packages/omo-codex/plugin/components/start-work-continuation/test/fixtures/plan-with-nested-checkboxes.md +11 -0
  301. package/packages/omo-codex/plugin/components/start-work-continuation/test/fixtures/plan-with-unchecked.md +6 -0
  302. package/packages/omo-codex/plugin/components/start-work-continuation/tsconfig.build.json +12 -0
  303. package/packages/omo-codex/plugin/components/start-work-continuation/tsconfig.json +27 -0
  304. package/packages/omo-codex/plugin/components/start-work-continuation/vitest.config.ts +10 -0
  305. package/packages/omo-codex/plugin/components/telemetry/AGENTS.md +37 -0
  306. package/packages/omo-codex/plugin/components/telemetry/README.md +102 -0
  307. package/packages/omo-codex/plugin/components/telemetry/biome.json +48 -0
  308. package/packages/omo-codex/plugin/components/telemetry/hooks/hooks.json +16 -0
  309. package/packages/omo-codex/plugin/components/telemetry/package.json +56 -0
  310. package/packages/omo-codex/plugin/components/telemetry/src/atomic-write.ts +22 -0
  311. package/packages/omo-codex/plugin/components/telemetry/src/cli.ts +69 -0
  312. package/packages/omo-codex/plugin/components/telemetry/src/codex-hook.ts +49 -0
  313. package/packages/omo-codex/plugin/components/telemetry/src/data-path.ts +45 -0
  314. package/packages/omo-codex/plugin/components/telemetry/src/env-flags.ts +43 -0
  315. package/packages/omo-codex/plugin/components/telemetry/src/posthog-activity-state.ts +81 -0
  316. package/packages/omo-codex/plugin/components/telemetry/src/posthog.ts +165 -0
  317. package/packages/omo-codex/plugin/components/telemetry/src/product-identity.ts +35 -0
  318. package/packages/omo-codex/plugin/components/telemetry/test/codex-hook.test.ts +270 -0
  319. package/packages/omo-codex/plugin/components/telemetry/tsconfig.build.json +12 -0
  320. package/packages/omo-codex/plugin/components/telemetry/tsconfig.json +27 -0
  321. package/packages/omo-codex/plugin/components/telemetry/vitest.config.ts +8 -0
  322. package/packages/omo-codex/plugin/components/ultrawork/AGENTS.md +41 -0
  323. package/packages/omo-codex/plugin/components/ultrawork/CHANGELOG.md +25 -0
  324. package/packages/omo-codex/plugin/components/ultrawork/LICENSE +21 -0
  325. package/packages/omo-codex/plugin/components/ultrawork/NOTICE +5 -0
  326. package/packages/omo-codex/plugin/components/ultrawork/README.md +60 -0
  327. package/packages/omo-codex/plugin/components/ultrawork/agents/codex-ultrawork-reviewer.toml +18 -0
  328. package/packages/omo-codex/plugin/components/ultrawork/agents/explorer.toml +82 -0
  329. package/packages/omo-codex/plugin/components/ultrawork/agents/librarian.toml +222 -0
  330. package/packages/omo-codex/plugin/components/ultrawork/agents/metis.toml +65 -0
  331. package/packages/omo-codex/plugin/components/ultrawork/agents/momus.toml +69 -0
  332. package/packages/omo-codex/plugin/components/ultrawork/agents/plan.toml +164 -0
  333. package/packages/omo-codex/plugin/components/ultrawork/biome.json +48 -0
  334. package/packages/omo-codex/plugin/components/ultrawork/directive.md +290 -0
  335. package/packages/omo-codex/plugin/components/ultrawork/hooks/hooks.json +16 -0
  336. package/packages/omo-codex/plugin/components/ultrawork/package.json +54 -0
  337. package/packages/omo-codex/plugin/components/ultrawork/src/cli.ts +50 -0
  338. package/packages/omo-codex/plugin/components/ultrawork/src/codex-hook.ts +84 -0
  339. package/packages/omo-codex/plugin/components/ultrawork/src/directive.ts +3 -0
  340. package/packages/omo-codex/plugin/components/ultrawork/test/codex-hook.test.ts +275 -0
  341. package/packages/omo-codex/plugin/components/ultrawork/test/package-smoke.test.ts +103 -0
  342. package/packages/omo-codex/plugin/components/ultrawork/tsconfig.build.json +12 -0
  343. package/packages/omo-codex/plugin/components/ultrawork/tsconfig.json +27 -0
  344. package/packages/omo-codex/plugin/components/ulw-loop/.gitattributes +13 -0
  345. package/packages/omo-codex/plugin/components/ulw-loop/AGENTS.md +48 -0
  346. package/packages/omo-codex/plugin/components/ulw-loop/CHANGELOG.md +7 -0
  347. package/packages/omo-codex/plugin/components/ulw-loop/LICENSE +21 -0
  348. package/packages/omo-codex/plugin/components/ulw-loop/NOTICE +6 -0
  349. package/packages/omo-codex/plugin/components/ulw-loop/README.md +74 -0
  350. package/packages/omo-codex/plugin/components/ulw-loop/biome.json +48 -0
  351. package/packages/omo-codex/plugin/components/ulw-loop/hooks/hooks.json +29 -0
  352. package/packages/omo-codex/plugin/components/ulw-loop/package.json +55 -0
  353. package/packages/omo-codex/plugin/components/ulw-loop/skills/ulw-loop/.gitkeep +0 -0
  354. package/packages/omo-codex/plugin/components/ulw-loop/skills/ulw-loop/SKILL.md +44 -0
  355. package/packages/omo-codex/plugin/components/ulw-loop/skills/ulw-loop/agents/openai.yaml +6 -0
  356. package/packages/omo-codex/plugin/components/ulw-loop/skills/ulw-loop/references/full-workflow.md +230 -0
  357. package/packages/omo-codex/plugin/components/ulw-loop/src/.gitkeep +0 -0
  358. package/packages/omo-codex/plugin/components/ulw-loop/src/checkpoint.ts +155 -0
  359. package/packages/omo-codex/plugin/components/ulw-loop/src/cli-arg-parser.ts +95 -0
  360. package/packages/omo-codex/plugin/components/ulw-loop/src/cli-commands.ts +156 -0
  361. package/packages/omo-codex/plugin/components/ulw-loop/src/cli-output.ts +63 -0
  362. package/packages/omo-codex/plugin/components/ulw-loop/src/cli-steering.ts +94 -0
  363. package/packages/omo-codex/plugin/components/ulw-loop/src/cli.ts +40 -0
  364. package/packages/omo-codex/plugin/components/ulw-loop/src/codex-goal-instruction.ts +129 -0
  365. package/packages/omo-codex/plugin/components/ulw-loop/src/codex-goal-snapshot.ts +139 -0
  366. package/packages/omo-codex/plugin/components/ulw-loop/src/codex-hook.ts +177 -0
  367. package/packages/omo-codex/plugin/components/ulw-loop/src/evidence.ts +122 -0
  368. package/packages/omo-codex/plugin/components/ulw-loop/src/goal-status.ts +88 -0
  369. package/packages/omo-codex/plugin/components/ulw-loop/src/paths.ts +73 -0
  370. package/packages/omo-codex/plugin/components/ulw-loop/src/plan-crud.ts +113 -0
  371. package/packages/omo-codex/plugin/components/ulw-loop/src/plan-io.ts +124 -0
  372. package/packages/omo-codex/plugin/components/ulw-loop/src/quality-gate.ts +102 -0
  373. package/packages/omo-codex/plugin/components/ulw-loop/src/review-blockers.ts +81 -0
  374. package/packages/omo-codex/plugin/components/ulw-loop/src/steering.ts +270 -0
  375. package/packages/omo-codex/plugin/components/ulw-loop/src/types.ts +277 -0
  376. package/packages/omo-codex/plugin/components/ulw-loop/test/checkpoint.test.ts +213 -0
  377. package/packages/omo-codex/plugin/components/ulw-loop/test/cli-commands.test.ts +375 -0
  378. package/packages/omo-codex/plugin/components/ulw-loop/test/cli-helpers.test.ts +250 -0
  379. package/packages/omo-codex/plugin/components/ulw-loop/test/cli-steering.test.ts +407 -0
  380. package/packages/omo-codex/plugin/components/ulw-loop/test/codex-goal-instruction.test.ts +169 -0
  381. package/packages/omo-codex/plugin/components/ulw-loop/test/codex-goal-snapshot.test.ts +156 -0
  382. package/packages/omo-codex/plugin/components/ulw-loop/test/codex-hook.test.ts +275 -0
  383. package/packages/omo-codex/plugin/components/ulw-loop/test/evidence-criteria-gate.test.ts +100 -0
  384. package/packages/omo-codex/plugin/components/ulw-loop/test/evidence.test.ts +263 -0
  385. package/packages/omo-codex/plugin/components/ulw-loop/test/fixtures/.gitkeep +0 -0
  386. package/packages/omo-codex/plugin/components/ulw-loop/test/fixtures/codex-goal-snapshot.json +1 -0
  387. package/packages/omo-codex/plugin/components/ulw-loop/test/fixtures/sample-brief.md +5 -0
  388. package/packages/omo-codex/plugin/components/ulw-loop/test/fixtures/sample-plan.json +108 -0
  389. package/packages/omo-codex/plugin/components/ulw-loop/test/fixtures/sample-quality-gate.json +18 -0
  390. package/packages/omo-codex/plugin/components/ulw-loop/test/fixtures/steering-proposal.json +8 -0
  391. package/packages/omo-codex/plugin/components/ulw-loop/test/fixtures/user-prompt-submit.json +10 -0
  392. package/packages/omo-codex/plugin/components/ulw-loop/test/goal-status.test.ts +327 -0
  393. package/packages/omo-codex/plugin/components/ulw-loop/test/package-smoke.test.ts +261 -0
  394. package/packages/omo-codex/plugin/components/ulw-loop/test/paths.test.ts +62 -0
  395. package/packages/omo-codex/plugin/components/ulw-loop/test/plan-crud.test.ts +256 -0
  396. package/packages/omo-codex/plugin/components/ulw-loop/test/plan-io.test.ts +239 -0
  397. package/packages/omo-codex/plugin/components/ulw-loop/test/quality-gate.test.ts +203 -0
  398. package/packages/omo-codex/plugin/components/ulw-loop/test/review-blockers.test.ts +180 -0
  399. package/packages/omo-codex/plugin/components/ulw-loop/test/steering.test.ts +353 -0
  400. package/packages/omo-codex/plugin/components/ulw-loop/test/types.test.ts +79 -0
  401. package/packages/omo-codex/plugin/components/ulw-loop/tsconfig.build.json +12 -0
  402. package/packages/omo-codex/plugin/components/ulw-loop/tsconfig.json +27 -0
  403. package/packages/omo-codex/plugin/components/ulw-loop/vitest.config.ts +10 -0
  404. package/packages/omo-codex/plugin/hooks/hooks.json +160 -0
  405. package/packages/omo-codex/plugin/package-lock.json +1769 -0
  406. package/packages/omo-codex/plugin/package.json +28 -0
  407. package/packages/omo-codex/plugin/scripts/build-bundled-mcp-runtimes.mjs +65 -0
  408. package/packages/omo-codex/plugin/scripts/build-components.mjs +24 -0
  409. package/packages/omo-codex/plugin/scripts/hook-status-message.mjs +46 -0
  410. package/packages/omo-codex/plugin/scripts/sync-hook-status-messages.mjs +87 -0
  411. package/packages/omo-codex/plugin/scripts/sync-skills.mjs +75 -0
  412. package/packages/omo-codex/plugin/skills/comment-checker/SKILL.md +16 -0
  413. package/packages/omo-codex/plugin/skills/debugging/SKILL.md +116 -0
  414. package/packages/omo-codex/plugin/skills/debugging/references/methodology/00-setup.md +108 -0
  415. package/packages/omo-codex/plugin/skills/debugging/references/methodology/02-investigate.md +130 -0
  416. package/packages/omo-codex/plugin/skills/debugging/references/methodology/04-oracle-triple.md +136 -0
  417. package/packages/omo-codex/plugin/skills/debugging/references/methodology/05-escalate.md +69 -0
  418. package/packages/omo-codex/plugin/skills/debugging/references/methodology/06-fix.md +116 -0
  419. package/packages/omo-codex/plugin/skills/debugging/references/methodology/08-qa.md +94 -0
  420. package/packages/omo-codex/plugin/skills/debugging/references/methodology/09-cleanup.md +164 -0
  421. package/packages/omo-codex/plugin/skills/debugging/references/methodology/partial-runtime-evidence.md +229 -0
  422. package/packages/omo-codex/plugin/skills/debugging/references/runtimes/bundled-js-binary.md +415 -0
  423. package/packages/omo-codex/plugin/skills/debugging/references/runtimes/go.md +252 -0
  424. package/packages/omo-codex/plugin/skills/debugging/references/runtimes/native-binary.md +484 -0
  425. package/packages/omo-codex/plugin/skills/debugging/references/runtimes/node.md +260 -0
  426. package/packages/omo-codex/plugin/skills/debugging/references/runtimes/python.md +248 -0
  427. package/packages/omo-codex/plugin/skills/debugging/references/runtimes/rust.md +234 -0
  428. package/packages/omo-codex/plugin/skills/debugging/references/tools/ghidra.md +212 -0
  429. package/packages/omo-codex/plugin/skills/debugging/references/tools/playwright-cli.md +194 -0
  430. package/packages/omo-codex/plugin/skills/debugging/references/tools/pwndbg.md +263 -0
  431. package/packages/omo-codex/plugin/skills/debugging/references/tools/pwntools.md +265 -0
  432. package/packages/omo-codex/plugin/skills/frontend-ui-ux/SKILL.md +77 -0
  433. package/packages/omo-codex/plugin/skills/init-deep/SKILL.md +325 -0
  434. package/packages/omo-codex/plugin/skills/lsp/SKILL.md +35 -0
  435. package/packages/omo-codex/plugin/skills/programming/SKILL.md +463 -0
  436. package/packages/omo-codex/plugin/skills/programming/references/go/README.md +90 -0
  437. package/packages/omo-codex/plugin/skills/programming/references/go/backend-stack.md +641 -0
  438. package/packages/omo-codex/plugin/skills/programming/references/go/bootstrap.md +328 -0
  439. package/packages/omo-codex/plugin/skills/programming/references/go/bubbletea-v2.md +360 -0
  440. package/packages/omo-codex/plugin/skills/programming/references/go/cobra-stack.md +468 -0
  441. package/packages/omo-codex/plugin/skills/programming/references/go/concurrency.md +362 -0
  442. package/packages/omo-codex/plugin/skills/programming/references/go/data-modeling.md +329 -0
  443. package/packages/omo-codex/plugin/skills/programming/references/go/error-handling.md +359 -0
  444. package/packages/omo-codex/plugin/skills/programming/references/go/golangci-strict.md +236 -0
  445. package/packages/omo-codex/plugin/skills/programming/references/go/grpc-connect.md +375 -0
  446. package/packages/omo-codex/plugin/skills/programming/references/go/libraries.md +337 -0
  447. package/packages/omo-codex/plugin/skills/programming/references/go/one-liners.md +202 -0
  448. package/packages/omo-codex/plugin/skills/programming/references/go/sqlc-pgx.md +471 -0
  449. package/packages/omo-codex/plugin/skills/programming/references/go/testing.md +467 -0
  450. package/packages/omo-codex/plugin/skills/programming/references/go/type-patterns.md +298 -0
  451. package/packages/omo-codex/plugin/skills/programming/references/python/README.md +314 -0
  452. package/packages/omo-codex/plugin/skills/programming/references/python/async-anyio.md +442 -0
  453. package/packages/omo-codex/plugin/skills/programming/references/python/data-modeling.md +233 -0
  454. package/packages/omo-codex/plugin/skills/programming/references/python/data-processing.md +133 -0
  455. package/packages/omo-codex/plugin/skills/programming/references/python/error-handling.md +218 -0
  456. package/packages/omo-codex/plugin/skills/programming/references/python/fastapi-stack.md +316 -0
  457. package/packages/omo-codex/plugin/skills/programming/references/python/httpx2-optimization.md +360 -0
  458. package/packages/omo-codex/plugin/skills/programming/references/python/libraries.md +307 -0
  459. package/packages/omo-codex/plugin/skills/programming/references/python/one-liners.md +268 -0
  460. package/packages/omo-codex/plugin/skills/programming/references/python/orjson-stack.md +378 -0
  461. package/packages/omo-codex/plugin/skills/programming/references/python/pydantic-ai.md +285 -0
  462. package/packages/omo-codex/plugin/skills/programming/references/python/pyproject-strict.md +232 -0
  463. package/packages/omo-codex/plugin/skills/programming/references/python/textual-tui.md +201 -0
  464. package/packages/omo-codex/plugin/skills/programming/references/python/type-patterns.md +176 -0
  465. package/packages/omo-codex/plugin/skills/programming/references/rust/README.md +317 -0
  466. package/packages/omo-codex/plugin/skills/programming/references/rust/async-tokio.md +299 -0
  467. package/packages/omo-codex/plugin/skills/programming/references/rust/axum-stack.md +467 -0
  468. package/packages/omo-codex/plugin/skills/programming/references/rust/cargo-strict.md +317 -0
  469. package/packages/omo-codex/plugin/skills/programming/references/rust/clap-stack.md +409 -0
  470. package/packages/omo-codex/plugin/skills/programming/references/rust/concurrency.md +375 -0
  471. package/packages/omo-codex/plugin/skills/programming/references/rust/libraries.md +439 -0
  472. package/packages/omo-codex/plugin/skills/programming/references/rust/one-liners.md +291 -0
  473. package/packages/omo-codex/plugin/skills/programming/references/rust/proptest-insta.md +429 -0
  474. package/packages/omo-codex/plugin/skills/programming/references/rust/type-state.md +354 -0
  475. package/packages/omo-codex/plugin/skills/programming/references/rust/unsafe-discipline.md +250 -0
  476. package/packages/omo-codex/plugin/skills/programming/references/rust/zero-cost-safety.md +527 -0
  477. package/packages/omo-codex/plugin/skills/programming/references/rust-ub/README.md +289 -0
  478. package/packages/omo-codex/plugin/skills/programming/references/rust-ub/miri-sanitizers-loom.md +411 -0
  479. package/packages/omo-codex/plugin/skills/programming/references/rust-ub/ub-taxonomy.md +269 -0
  480. package/packages/omo-codex/plugin/skills/programming/references/typescript/README.md +195 -0
  481. package/packages/omo-codex/plugin/skills/programming/references/typescript/backend-hono.md +672 -0
  482. package/packages/omo-codex/plugin/skills/programming/references/typescript/bootstrap.md +199 -0
  483. package/packages/omo-codex/plugin/skills/programming/references/typescript/data-modeling.md +202 -0
  484. package/packages/omo-codex/plugin/skills/programming/references/typescript/error-handling.md +169 -0
  485. package/packages/omo-codex/plugin/skills/programming/references/typescript/tsconfig-strict.md +152 -0
  486. package/packages/omo-codex/plugin/skills/programming/references/typescript/type-patterns.md +196 -0
  487. package/packages/omo-codex/plugin/skills/programming/scripts/go/check-no-excuse-rules.sh +173 -0
  488. package/packages/omo-codex/plugin/skills/programming/scripts/go/new-project.py +138 -0
  489. package/packages/omo-codex/plugin/skills/programming/scripts/go/templates/.editorconfig +13 -0
  490. package/packages/omo-codex/plugin/skills/programming/scripts/go/templates/.golangci.yml +95 -0
  491. package/packages/omo-codex/plugin/skills/programming/scripts/go/templates/AGENTS.md.tmpl +24 -0
  492. package/packages/omo-codex/plugin/skills/programming/scripts/go/templates/README.md.tmpl +12 -0
  493. package/packages/omo-codex/plugin/skills/programming/scripts/go/templates/Taskfile.yml +40 -0
  494. package/packages/omo-codex/plugin/skills/programming/scripts/go/templates/ci.yml +37 -0
  495. package/packages/omo-codex/plugin/skills/programming/scripts/go/templates/config.go +24 -0
  496. package/packages/omo-codex/plugin/skills/programming/scripts/go/templates/gitignore +15 -0
  497. package/packages/omo-codex/plugin/skills/programming/scripts/go/templates/main.go.tmpl +22 -0
  498. package/packages/omo-codex/plugin/skills/programming/scripts/go/templates/run.go +15 -0
  499. package/packages/omo-codex/plugin/skills/programming/scripts/python/check-no-excuse-rules.py +687 -0
  500. package/packages/omo-codex/plugin/skills/programming/scripts/python/new-project.py +172 -0
  501. package/packages/omo-codex/plugin/skills/programming/scripts/python/new-script.py +116 -0
  502. package/packages/omo-codex/plugin/skills/programming/scripts/rust/check-no-excuse-rules.py +296 -0
  503. package/packages/omo-codex/plugin/skills/programming/scripts/rust/check-no-excuse-rules.sh +158 -0
  504. package/packages/omo-codex/plugin/skills/programming/scripts/rust/new-project.py +175 -0
  505. package/packages/omo-codex/plugin/skills/programming/scripts/typescript/check-no-excuse-rules.ts +282 -0
  506. package/packages/omo-codex/plugin/skills/programming/scripts/typescript/new-project.ts +177 -0
  507. package/packages/omo-codex/plugin/skills/refactor/SKILL.md +779 -0
  508. package/packages/omo-codex/plugin/skills/remove-ai-slops/SKILL.md +333 -0
  509. package/packages/omo-codex/plugin/skills/review-work/SKILL.md +574 -0
  510. package/packages/omo-codex/plugin/skills/rules/SKILL.md +34 -0
  511. package/packages/omo-codex/plugin/skills/start-work/SKILL.md +149 -0
  512. package/packages/omo-codex/plugin/skills/ulw-loop/.gitkeep +0 -0
  513. package/packages/omo-codex/plugin/skills/ulw-loop/SKILL.md +44 -0
  514. package/packages/omo-codex/plugin/skills/ulw-loop/agents/openai.yaml +6 -0
  515. package/packages/omo-codex/plugin/skills/ulw-loop/references/full-workflow.md +230 -0
  516. package/packages/omo-codex/plugin/skills/ulw-plan/SKILL.md +399 -0
  517. package/packages/omo-codex/plugin/test/aggregate.test.mjs +345 -0
  518. package/packages/omo-codex/plugin/test/component-bin-names.test.mjs +66 -0
  519. package/packages/omo-codex/plugin/test/hook-status-message.test.mjs +195 -0
  520. package/packages/omo-codex/plugin/test/install-time-build-runtime.test.mjs +34 -0
  521. package/packages/omo-codex/plugin/test/mcp-research-servers.test.mjs +21 -0
  522. package/packages/omo-codex/plugin/test/node-install-surface.test.mjs +48 -0
  523. package/packages/omo-codex/plugin/test/subagent-guidance.test.mjs +76 -0
  524. package/packages/omo-codex/plugin/test/sync-hook-status-messages.test.mjs +66 -0
  525. package/packages/omo-codex/plugin/test/sync-skills.test.mjs +229 -0
  526. package/packages/omo-codex/scripts/install/agents.mjs +84 -0
  527. package/packages/omo-codex/scripts/install/cache.mjs +247 -0
  528. package/packages/omo-codex/scripts/install/cli-args.mjs +112 -0
  529. package/packages/omo-codex/scripts/install/command-shim.mjs +1 -0
  530. package/packages/omo-codex/scripts/install/config.mjs +264 -0
  531. package/packages/omo-codex/scripts/install/delegated-command.mjs +25 -0
  532. package/packages/omo-codex/scripts/install/git-bash.mjs +99 -0
  533. package/packages/omo-codex/scripts/install/git-bash.test.mjs +174 -0
  534. package/packages/omo-codex/scripts/install/hook-trust.mjs +84 -0
  535. package/packages/omo-codex/scripts/install/legacy-bins.mjs +57 -0
  536. package/packages/omo-codex/scripts/install/marketplace.mjs +104 -0
  537. package/packages/omo-codex/scripts/install/mcp-runtime-cache.mjs +81 -0
  538. package/packages/omo-codex/scripts/install/multi-agent-v2-config.mjs +38 -0
  539. package/packages/omo-codex/scripts/install/permissions.d.mts +1 -0
  540. package/packages/omo-codex/scripts/install/permissions.mjs +26 -0
  541. package/packages/omo-codex/scripts/install/process.mjs +19 -0
  542. package/packages/omo-codex/scripts/install/project-local-cleanup.mjs +229 -0
  543. package/packages/omo-codex/scripts/install/reasoning-config.mjs +14 -0
  544. package/packages/omo-codex/scripts/install/snapshot.mjs +54 -0
  545. package/packages/omo-codex/scripts/install/source-package-build.mjs +20 -0
  546. package/packages/omo-codex/scripts/install/toml-editor.mjs +64 -0
  547. package/packages/omo-codex/scripts/install/utils.mjs +15 -0
  548. package/packages/omo-codex/scripts/install-agent-links.test.mjs +104 -0
  549. package/packages/omo-codex/scripts/install-bin-links.test.mjs +123 -0
  550. package/packages/omo-codex/scripts/install-cache-copy.test.mjs +30 -0
  551. package/packages/omo-codex/scripts/install-cli-args.test.mjs +146 -0
  552. package/packages/omo-codex/scripts/install-config-autonomous.test.mjs +48 -0
  553. package/packages/omo-codex/scripts/install-config-reasoning.test.mjs +62 -0
  554. package/packages/omo-codex/scripts/install-config.test.mjs +324 -0
  555. package/packages/omo-codex/scripts/install-local-entrypoint.test.mjs +129 -0
  556. package/packages/omo-codex/scripts/install-local-git-bash-preflight.test.mjs +145 -0
  557. package/packages/omo-codex/scripts/install-local.mjs +275 -0
  558. package/packages/omo-codex/scripts/install-local.test.mjs +394 -0
  559. package/packages/omo-codex/scripts/install-mcp-runtime.test.mjs +233 -0
  560. package/packages/omo-codex/scripts/install-packaged-local.test.mjs +67 -0
  561. package/packages/omo-codex/scripts/install-project-local-cleanup.test.mjs +277 -0
  562. package/packages/omo-codex/scripts/install-test-fixtures.mjs +58 -0
  563. package/packages/omo-codex/scripts/sync-telemetry-component.mjs +115 -0
  564. package/packages/omo-codex/scripts/sync-telemetry-component.test.mjs +94 -0
  565. package/packages/shared-skills/index.mjs +5 -0
  566. package/packages/shared-skills/package.json +14 -0
  567. package/packages/shared-skills/skills/debugging/SKILL.md +116 -0
  568. package/packages/shared-skills/skills/debugging/references/methodology/00-setup.md +108 -0
  569. package/packages/shared-skills/skills/debugging/references/methodology/02-investigate.md +130 -0
  570. package/packages/shared-skills/skills/debugging/references/methodology/04-oracle-triple.md +136 -0
  571. package/packages/shared-skills/skills/debugging/references/methodology/05-escalate.md +69 -0
  572. package/packages/shared-skills/skills/debugging/references/methodology/06-fix.md +116 -0
  573. package/packages/shared-skills/skills/debugging/references/methodology/08-qa.md +94 -0
  574. package/packages/shared-skills/skills/debugging/references/methodology/09-cleanup.md +164 -0
  575. package/packages/shared-skills/skills/debugging/references/methodology/partial-runtime-evidence.md +229 -0
  576. package/packages/shared-skills/skills/debugging/references/runtimes/bundled-js-binary.md +415 -0
  577. package/packages/shared-skills/skills/debugging/references/runtimes/go.md +252 -0
  578. package/packages/shared-skills/skills/debugging/references/runtimes/native-binary.md +484 -0
  579. package/packages/shared-skills/skills/debugging/references/runtimes/node.md +260 -0
  580. package/packages/shared-skills/skills/debugging/references/runtimes/python.md +248 -0
  581. package/packages/shared-skills/skills/debugging/references/runtimes/rust.md +234 -0
  582. package/packages/shared-skills/skills/debugging/references/tools/ghidra.md +212 -0
  583. package/packages/shared-skills/skills/debugging/references/tools/playwright-cli.md +194 -0
  584. package/packages/shared-skills/skills/debugging/references/tools/pwndbg.md +263 -0
  585. package/packages/shared-skills/skills/debugging/references/tools/pwntools.md +265 -0
  586. package/packages/shared-skills/skills/frontend-ui-ux/SKILL.md +77 -0
  587. package/packages/shared-skills/skills/init-deep/SKILL.md +309 -0
  588. package/packages/shared-skills/skills/programming/SKILL.md +463 -0
  589. package/packages/shared-skills/skills/programming/references/go/README.md +90 -0
  590. package/packages/shared-skills/skills/programming/references/go/backend-stack.md +641 -0
  591. package/packages/shared-skills/skills/programming/references/go/bootstrap.md +328 -0
  592. package/packages/shared-skills/skills/programming/references/go/bubbletea-v2.md +360 -0
  593. package/packages/shared-skills/skills/programming/references/go/cobra-stack.md +468 -0
  594. package/packages/shared-skills/skills/programming/references/go/concurrency.md +362 -0
  595. package/packages/shared-skills/skills/programming/references/go/data-modeling.md +329 -0
  596. package/packages/shared-skills/skills/programming/references/go/error-handling.md +359 -0
  597. package/packages/shared-skills/skills/programming/references/go/golangci-strict.md +236 -0
  598. package/packages/shared-skills/skills/programming/references/go/grpc-connect.md +375 -0
  599. package/packages/shared-skills/skills/programming/references/go/libraries.md +337 -0
  600. package/packages/shared-skills/skills/programming/references/go/one-liners.md +202 -0
  601. package/packages/shared-skills/skills/programming/references/go/sqlc-pgx.md +471 -0
  602. package/packages/shared-skills/skills/programming/references/go/testing.md +467 -0
  603. package/packages/shared-skills/skills/programming/references/go/type-patterns.md +298 -0
  604. package/packages/shared-skills/skills/programming/references/python/README.md +314 -0
  605. package/packages/shared-skills/skills/programming/references/python/async-anyio.md +442 -0
  606. package/packages/shared-skills/skills/programming/references/python/data-modeling.md +233 -0
  607. package/packages/shared-skills/skills/programming/references/python/data-processing.md +133 -0
  608. package/packages/shared-skills/skills/programming/references/python/error-handling.md +218 -0
  609. package/packages/shared-skills/skills/programming/references/python/fastapi-stack.md +316 -0
  610. package/packages/shared-skills/skills/programming/references/python/httpx2-optimization.md +360 -0
  611. package/packages/shared-skills/skills/programming/references/python/libraries.md +307 -0
  612. package/packages/shared-skills/skills/programming/references/python/one-liners.md +268 -0
  613. package/packages/shared-skills/skills/programming/references/python/orjson-stack.md +378 -0
  614. package/packages/shared-skills/skills/programming/references/python/pydantic-ai.md +285 -0
  615. package/packages/shared-skills/skills/programming/references/python/pyproject-strict.md +232 -0
  616. package/packages/shared-skills/skills/programming/references/python/textual-tui.md +201 -0
  617. package/packages/shared-skills/skills/programming/references/python/type-patterns.md +176 -0
  618. package/packages/shared-skills/skills/programming/references/rust/README.md +317 -0
  619. package/packages/shared-skills/skills/programming/references/rust/async-tokio.md +299 -0
  620. package/packages/shared-skills/skills/programming/references/rust/axum-stack.md +467 -0
  621. package/packages/shared-skills/skills/programming/references/rust/cargo-strict.md +317 -0
  622. package/packages/shared-skills/skills/programming/references/rust/clap-stack.md +409 -0
  623. package/packages/shared-skills/skills/programming/references/rust/concurrency.md +375 -0
  624. package/packages/shared-skills/skills/programming/references/rust/libraries.md +439 -0
  625. package/packages/shared-skills/skills/programming/references/rust/one-liners.md +291 -0
  626. package/packages/shared-skills/skills/programming/references/rust/proptest-insta.md +429 -0
  627. package/packages/shared-skills/skills/programming/references/rust/type-state.md +354 -0
  628. package/packages/shared-skills/skills/programming/references/rust/unsafe-discipline.md +250 -0
  629. package/packages/shared-skills/skills/programming/references/rust/zero-cost-safety.md +527 -0
  630. package/packages/shared-skills/skills/programming/references/rust-ub/README.md +289 -0
  631. package/packages/shared-skills/skills/programming/references/rust-ub/miri-sanitizers-loom.md +411 -0
  632. package/packages/shared-skills/skills/programming/references/rust-ub/ub-taxonomy.md +269 -0
  633. package/packages/shared-skills/skills/programming/references/typescript/README.md +195 -0
  634. package/packages/shared-skills/skills/programming/references/typescript/backend-hono.md +672 -0
  635. package/packages/shared-skills/skills/programming/references/typescript/bootstrap.md +199 -0
  636. package/packages/shared-skills/skills/programming/references/typescript/data-modeling.md +202 -0
  637. package/packages/shared-skills/skills/programming/references/typescript/error-handling.md +169 -0
  638. package/packages/shared-skills/skills/programming/references/typescript/tsconfig-strict.md +152 -0
  639. package/packages/shared-skills/skills/programming/references/typescript/type-patterns.md +196 -0
  640. package/packages/shared-skills/skills/programming/scripts/go/check-no-excuse-rules.sh +173 -0
  641. package/packages/shared-skills/skills/programming/scripts/go/new-project.py +138 -0
  642. package/packages/shared-skills/skills/programming/scripts/go/templates/.editorconfig +13 -0
  643. package/packages/shared-skills/skills/programming/scripts/go/templates/.golangci.yml +95 -0
  644. package/packages/shared-skills/skills/programming/scripts/go/templates/AGENTS.md.tmpl +24 -0
  645. package/packages/shared-skills/skills/programming/scripts/go/templates/README.md.tmpl +12 -0
  646. package/packages/shared-skills/skills/programming/scripts/go/templates/Taskfile.yml +40 -0
  647. package/packages/shared-skills/skills/programming/scripts/go/templates/ci.yml +37 -0
  648. package/packages/shared-skills/skills/programming/scripts/go/templates/config.go +24 -0
  649. package/packages/shared-skills/skills/programming/scripts/go/templates/gitignore +15 -0
  650. package/packages/shared-skills/skills/programming/scripts/go/templates/main.go.tmpl +22 -0
  651. package/packages/shared-skills/skills/programming/scripts/go/templates/run.go +15 -0
  652. package/packages/shared-skills/skills/programming/scripts/python/check-no-excuse-rules.py +687 -0
  653. package/packages/shared-skills/skills/programming/scripts/python/new-project.py +172 -0
  654. package/packages/shared-skills/skills/programming/scripts/python/new-script.py +116 -0
  655. package/packages/shared-skills/skills/programming/scripts/rust/check-no-excuse-rules.py +296 -0
  656. package/packages/shared-skills/skills/programming/scripts/rust/check-no-excuse-rules.sh +158 -0
  657. package/packages/shared-skills/skills/programming/scripts/rust/new-project.py +175 -0
  658. package/packages/shared-skills/skills/programming/scripts/typescript/check-no-excuse-rules.ts +282 -0
  659. package/packages/shared-skills/skills/programming/scripts/typescript/new-project.ts +177 -0
  660. package/packages/shared-skills/skills/refactor/SKILL.md +763 -0
  661. package/packages/shared-skills/skills/remove-ai-slops/SKILL.md +317 -0
  662. package/packages/shared-skills/skills/review-work/SKILL.md +574 -0
  663. package/packages/shared-skills/skills/start-work/SKILL.md +149 -0
  664. package/packages/shared-skills/skills/ulw-plan/SKILL.md +383 -0
  665. package/postinstall.mjs +6 -2
  666. package/dist/features/builtin-commands/templates/init-deep.d.ts +0 -1
  667. package/dist/features/builtin-skills/skills/ai-slop-remover.d.ts +0 -2
  668. package/dist/hooks/context-window-monitor.d.ts +0 -19
@@ -0,0 +1,177 @@
1
+ import type { UlwLoopScope } from "./paths.js";
2
+ import { parseUlwLoopSteeringDirective, steerUlwLoop } from "./steering.js";
3
+
4
+ export interface UserPromptSubmitPayload {
5
+ readonly cwd: string;
6
+ readonly hook_event_name: "UserPromptSubmit";
7
+ readonly model?: string;
8
+ readonly permission_mode?: string;
9
+ readonly prompt: string;
10
+ readonly session_id: string;
11
+ readonly transcript_path?: string;
12
+ readonly turn_id?: string;
13
+ }
14
+
15
+ export interface PreToolUsePayload {
16
+ readonly cwd: string;
17
+ readonly hook_event_name: "PreToolUse";
18
+ readonly model: string;
19
+ readonly permission_mode: string;
20
+ readonly session_id: string;
21
+ readonly tool_input: unknown;
22
+ readonly tool_name: string;
23
+ readonly tool_use_id: string;
24
+ readonly transcript_path: string | null;
25
+ readonly turn_id: string;
26
+ }
27
+
28
+ interface PreToolUseHookOutput {
29
+ readonly hookSpecificOutput: {
30
+ readonly hookEventName: "PreToolUse";
31
+ readonly permissionDecision: "deny";
32
+ readonly permissionDecisionReason: string;
33
+ readonly additionalContext: string;
34
+ };
35
+ }
36
+
37
+ const CREATE_GOAL_TOOL_NAME = "create_goal";
38
+ const GOAL_BUDGET_WARNING =
39
+ "Do not set token_budget on create_goal. Omit the budget field so the goal stays unlimited; ultrawork and ulw-loop runs must always use unlimited goals.";
40
+
41
+ export function parseUserPromptSubmitPayload(raw: string): UserPromptSubmitPayload | null {
42
+ if (raw.trim().length === 0) return null;
43
+ try {
44
+ const parsed: unknown = JSON.parse(raw);
45
+ return isUserPromptSubmitPayload(parsed) ? parsed : null;
46
+ } catch (error) {
47
+ if (error instanceof SyntaxError) return null;
48
+ return null;
49
+ }
50
+ }
51
+
52
+ export function parsePreToolUsePayload(raw: string): PreToolUsePayload | null {
53
+ if (raw.trim().length === 0) return null;
54
+ try {
55
+ const parsed: unknown = JSON.parse(raw);
56
+ return isPreToolUsePayload(parsed) ? parsed : null;
57
+ } catch (error) {
58
+ if (error instanceof SyntaxError) return null;
59
+ return null;
60
+ }
61
+ }
62
+
63
+ export async function applyUserPromptUlwLoopSteering(payload: UserPromptSubmitPayload): Promise<string> {
64
+ try {
65
+ if (payload.hook_event_name !== "UserPromptSubmit") return "";
66
+ const proposal = parseUlwLoopSteeringDirective(payload.prompt);
67
+ if (proposal === null) return "";
68
+ const result = await steerUlwLoop(payload.cwd, proposal, payloadScope(payload));
69
+ if (!result.accepted) return "";
70
+ return JSON.stringify({
71
+ status: "accepted",
72
+ kind: result.audit.kind,
73
+ source: result.audit.source,
74
+ deduped: result.deduped,
75
+ });
76
+ } catch (error) {
77
+ if (error instanceof Error) return "";
78
+ return "";
79
+ }
80
+ }
81
+
82
+ function payloadScope(payload: UserPromptSubmitPayload): UlwLoopScope {
83
+ return { sessionId: payload.session_id };
84
+ }
85
+
86
+ export function applyPreToolUseGoalBudgetGuard(payload: PreToolUsePayload): string {
87
+ if (payload.hook_event_name !== "PreToolUse") return "";
88
+ if (payload.tool_name !== CREATE_GOAL_TOOL_NAME) return "";
89
+ if (!hasGoalBudgetInput(payload.tool_input)) return "";
90
+ const output: PreToolUseHookOutput = {
91
+ hookSpecificOutput: {
92
+ hookEventName: "PreToolUse",
93
+ permissionDecision: "deny",
94
+ permissionDecisionReason: GOAL_BUDGET_WARNING,
95
+ additionalContext: GOAL_BUDGET_WARNING,
96
+ },
97
+ };
98
+ return `${JSON.stringify(output)}\n`;
99
+ }
100
+
101
+ export async function runUlwLoopHookCli(stdin: NodeJS.ReadableStream, stdout: NodeJS.WritableStream): Promise<void> {
102
+ try {
103
+ const payload = parseUserPromptSubmitPayload(await readAll(stdin));
104
+ if (payload === null) return;
105
+ const output = await applyUserPromptUlwLoopSteering(payload);
106
+ if (output.length > 0) stdout.write(output);
107
+ } catch (error) {
108
+ if (error instanceof Error) return;
109
+ return;
110
+ }
111
+ }
112
+
113
+ export async function runPreToolUseGoalBudgetGuardCli(
114
+ stdin: NodeJS.ReadableStream,
115
+ stdout: NodeJS.WritableStream,
116
+ ): Promise<void> {
117
+ try {
118
+ const payload = parsePreToolUsePayload(await readAll(stdin));
119
+ if (payload === null) return;
120
+ const output = applyPreToolUseGoalBudgetGuard(payload);
121
+ if (output.length > 0) stdout.write(output);
122
+ } catch (error) {
123
+ if (error instanceof Error) return;
124
+ return;
125
+ }
126
+ }
127
+
128
+ function isUserPromptSubmitPayload(value: unknown): value is UserPromptSubmitPayload {
129
+ if (!isRecord(value)) return false;
130
+ return (
131
+ value["hook_event_name"] === "UserPromptSubmit" &&
132
+ typeof value["cwd"] === "string" &&
133
+ typeof value["prompt"] === "string" &&
134
+ typeof value["session_id"] === "string" &&
135
+ ["model", "permission_mode", "transcript_path", "turn_id"].every((key) => optionalString(value[key]))
136
+ );
137
+ }
138
+
139
+ function isPreToolUsePayload(value: unknown): value is PreToolUsePayload {
140
+ if (!isRecord(value)) return false;
141
+ return (
142
+ value["hook_event_name"] === "PreToolUse" &&
143
+ typeof value["cwd"] === "string" &&
144
+ typeof value["model"] === "string" &&
145
+ typeof value["permission_mode"] === "string" &&
146
+ typeof value["session_id"] === "string" &&
147
+ typeof value["tool_name"] === "string" &&
148
+ typeof value["tool_use_id"] === "string" &&
149
+ (value["transcript_path"] === null || typeof value["transcript_path"] === "string") &&
150
+ typeof value["turn_id"] === "string" &&
151
+ Object.hasOwn(value, "tool_input")
152
+ );
153
+ }
154
+
155
+ function hasGoalBudgetInput(value: unknown): boolean {
156
+ return isRecord(value) && (Object.hasOwn(value, "token_budget") || Object.hasOwn(value, "tokenBudget"));
157
+ }
158
+
159
+ function isRecord(value: unknown): value is Record<string, unknown> {
160
+ return typeof value === "object" && value !== null && !Array.isArray(value);
161
+ }
162
+
163
+ function optionalString(value: unknown): boolean {
164
+ return value === undefined || typeof value === "string";
165
+ }
166
+
167
+ function readAll(stdin: NodeJS.ReadableStream): Promise<string> {
168
+ return new Promise((resolve, reject) => {
169
+ let data = "";
170
+ stdin.setEncoding("utf8");
171
+ stdin.on("data", (chunk: unknown) => {
172
+ data += chunk instanceof Buffer ? chunk.toString() : String(chunk);
173
+ });
174
+ stdin.once("error", reject);
175
+ stdin.once("end", () => resolve(data));
176
+ });
177
+ }
@@ -0,0 +1,122 @@
1
+ // biome-ignore-all format: keep this module under the mandated pure LOC budget.
2
+ import { hasAllCriteriaPass } from "./goal-status.js";
3
+ import type { UlwLoopScope } from "./paths.js";
4
+ import { appendLedger, readUlwLoopPlan, withUlwLoopMutationLock, writePlan } from "./plan-io.js";
5
+ import type { UlwLoopItem, UlwLoopLedgerEntry, UlwLoopPlan, UlwLoopSuccessCriterion } from "./types.js";
6
+ import { iso, UlwLoopError } from "./types.js";
7
+
8
+ type EvidenceStatus = "pass" | "fail" | "blocked";
9
+ type RecordEvidenceArgs = { readonly goalId: string; readonly criterionId: string; readonly status: EvidenceStatus; readonly evidence: string; readonly notes?: string };
10
+
11
+ function ulwLoopFail(message: string, code: string, details: Record<string, unknown>): never { throw new UlwLoopError(message, code, { details }); }
12
+
13
+ function ledgerKind(status: EvidenceStatus): UlwLoopLedgerEntry["kind"] {
14
+ switch (status) {
15
+ case "pass":
16
+ return "evidence_captured";
17
+ case "fail":
18
+ return "criterion_failed";
19
+ case "blocked":
20
+ return "criterion_blocked";
21
+ default:
22
+ return ulwLoopFail("Invalid criterion status.", "ULW_LOOP_CRITERION_STATUS_INVALID", { status });
23
+ }
24
+ }
25
+
26
+ function findGoal(plan: UlwLoopPlan, goalId: string): UlwLoopItem {
27
+ const goal = plan.goals.find((candidate) => candidate.id === goalId);
28
+ return goal ?? ulwLoopFail(`UlwLoop goal not found: ${goalId}.`, "ULW_LOOP_GOAL_NOT_FOUND", { goalId });
29
+ }
30
+
31
+ function findCriterion(goal: UlwLoopItem, criterionId: string): UlwLoopSuccessCriterion {
32
+ const criterion = goal.successCriteria.find((candidate) => candidate.id === criterionId);
33
+ return criterion ?? ulwLoopFail(`Success criterion not found: ${criterionId}.`, "ULW_LOOP_CRITERION_NOT_FOUND", { goalId: goal.id, criterionId });
34
+ }
35
+
36
+ function nonEmptyEvidence(evidence: string): string { const trimmed = evidence.trim(); return trimmed || ulwLoopFail("Evidence must be a non-empty string.", "ULW_LOOP_EVIDENCE_REQUIRED", {}); }
37
+
38
+ export async function recordEvidence(repoRoot: string, args: RecordEvidenceArgs, scope?: UlwLoopScope): Promise<{ plan: UlwLoopPlan; goal: UlwLoopItem; criterion: UlwLoopSuccessCriterion; ledgerEntry: UlwLoopLedgerEntry }> {
39
+ return withUlwLoopMutationLock(repoRoot, scope, async () => {
40
+ const plan = await readUlwLoopPlan(repoRoot, scope);
41
+ const goal = findGoal(plan, args.goalId);
42
+ const criterion = findCriterion(goal, args.criterionId);
43
+ const evidence = nonEmptyEvidence(args.evidence);
44
+ const kind = ledgerKind(args.status);
45
+ const prevStatus = criterion.status;
46
+ const capturedAt = iso();
47
+ criterion.status = args.status;
48
+ criterion.capturedEvidence = evidence;
49
+ criterion.capturedAt = capturedAt;
50
+ if (args.notes !== undefined) criterion.notes = args.notes;
51
+ goal.updatedAt = capturedAt;
52
+ plan.updatedAt = capturedAt;
53
+ await writePlan(repoRoot, plan, scope);
54
+ const ledgerEntry: UlwLoopLedgerEntry = {
55
+ at: capturedAt,
56
+ kind,
57
+ goalId: goal.id,
58
+ criterionId: criterion.id,
59
+ criterionStatus: args.status,
60
+ evidence,
61
+ capturedEvidence: evidence,
62
+ before: { status: prevStatus },
63
+ after: { goalId: goal.id, criterionId: criterion.id, status: args.status, evidence, capturedAt, prevStatus },
64
+ };
65
+ await appendLedger(repoRoot, ledgerEntry, scope);
66
+ return { plan, goal, criterion, ledgerEntry };
67
+ });
68
+ }
69
+
70
+ export async function markCriteriaPendingResetForGoal(repoRoot: string, goalId: string, scope?: UlwLoopScope): Promise<{ plan: UlwLoopPlan; resetCount: number }> {
71
+ return withUlwLoopMutationLock(repoRoot, scope, async () => {
72
+ const plan = await readUlwLoopPlan(repoRoot, scope);
73
+ const goal = findGoal(plan, goalId);
74
+ const now = iso();
75
+ const before = goal.successCriteria.map((criterion) => ({ id: criterion.id, status: criterion.status, capturedEvidence: criterion.capturedEvidence, capturedAt: criterion.capturedAt ?? null }));
76
+ for (const criterion of goal.successCriteria) {
77
+ criterion.status = "pending";
78
+ criterion.capturedEvidence = null;
79
+ delete criterion.capturedAt;
80
+ delete criterion.notes;
81
+ }
82
+ goal.updatedAt = now;
83
+ plan.updatedAt = now;
84
+ await writePlan(repoRoot, plan, scope);
85
+ await appendLedger(repoRoot, { at: now, kind: "criteria_revised", goalId, message: `Reset ${goal.successCriteria.length} criteria to pending.`, before, after: { resetCount: goal.successCriteria.length } }, scope);
86
+ return { plan, resetCount: goal.successCriteria.length };
87
+ });
88
+ }
89
+
90
+ export function criteriaSummary(plan: UlwLoopPlan): { totalCriteria: number; passCount: number; pendingCount: number; failCount: number; blockedCount: number; goalsWithUnresolvedCriteria: string[] } {
91
+ let totalCriteria = 0;
92
+ let passCount = 0;
93
+ let pendingCount = 0;
94
+ let failCount = 0;
95
+ let blockedCount = 0;
96
+ const goalsWithUnresolvedCriteria: string[] = [];
97
+ for (const goal of plan.goals) {
98
+ let unresolved = false;
99
+ for (const criterion of goal.successCriteria) {
100
+ totalCriteria += 1;
101
+ if (criterion.status !== "pass") unresolved = true;
102
+ switch (criterion.status) {
103
+ case "pass": passCount += 1; break;
104
+ case "pending": pendingCount += 1; break;
105
+ case "fail": failCount += 1; break;
106
+ case "blocked": blockedCount += 1; break;
107
+ default: ulwLoopFail("Invalid criterion status.", "ULW_LOOP_CRITERION_STATUS_INVALID", { status: criterion.status });
108
+ }
109
+ }
110
+ if (unresolved) goalsWithUnresolvedCriteria.push(goal.id);
111
+ }
112
+ return { totalCriteria, passCount, pendingCount, failCount, blockedCount, goalsWithUnresolvedCriteria };
113
+ }
114
+
115
+ export function unresolvedCriteriaOf(goal: UlwLoopItem): UlwLoopSuccessCriterion[] { return goal.successCriteria.filter((criterion) => criterion.status !== "pass"); }
116
+
117
+ export function requireAllCriteriaPass(goal: UlwLoopItem): void {
118
+ if (hasAllCriteriaPass(goal)) return;
119
+ throw new UlwLoopError(`Goal ${goal.id} has unresolved success criteria.`, "ulw_loop_criteria_not_all_pass", {
120
+ details: { goalId: goal.id, unresolved: unresolvedCriteriaOf(goal).map((criterion) => ({ id: criterion.id, status: criterion.status })) },
121
+ });
122
+ }
@@ -0,0 +1,88 @@
1
+ import { type UlwLoopScope, ulwLoopGoalsRelativePath, ulwLoopLedgerRelativePath } from "./paths.js";
2
+ import type {
3
+ UlwLoopCodexGoalMode,
4
+ UlwLoopItem,
5
+ UlwLoopPlan,
6
+ UlwLoopStatus,
7
+ UlwLoopSuccessCriterion,
8
+ } from "./types.js";
9
+
10
+ export const ULW_LOOP_AGGREGATE_CODEX_OBJECTIVE: string = aggregateCodexObjectiveForScope();
11
+
12
+ export function aggregateCodexObjectiveForScope(scope?: UlwLoopScope): string {
13
+ return `Complete the durable ulw-loop plan in ${ulwLoopGoalsRelativePath(scope)}, including later accepted/appended stories, under the original brief constraints; use ${ulwLoopLedgerRelativePath(scope)} as the audit trail.`;
14
+ }
15
+
16
+ export function codexGoalMode(plan: UlwLoopPlan): UlwLoopCodexGoalMode {
17
+ return plan.codexGoalMode ?? "per_story";
18
+ }
19
+
20
+ function isResolvedStatus(status: UlwLoopStatus): boolean {
21
+ return status === "complete";
22
+ }
23
+
24
+ function isSupersededResolved(goal: UlwLoopItem, plan: UlwLoopPlan): boolean {
25
+ if (goal.steeringStatus !== "superseded") return false;
26
+ const replacements = goal.supersededBy ?? [];
27
+ if (replacements.length === 0) return false;
28
+ return replacements.every((id) => {
29
+ const replacement = plan.goals.find((candidate) => candidate.id === id);
30
+ return replacement !== undefined && isResolvedStatus(replacement.status);
31
+ });
32
+ }
33
+
34
+ function isCompletionBlocking(goal: UlwLoopItem, plan: UlwLoopPlan): boolean {
35
+ if (goal.steeringStatus === "superseded") return !isSupersededResolved(goal, plan);
36
+ if (goal.steeringStatus === "blocked") return true;
37
+ return !isResolvedStatus(goal.status);
38
+ }
39
+
40
+ function isCompletionBlockingForFinalCandidate(
41
+ candidate: UlwLoopItem,
42
+ finalCandidate: UlwLoopItem,
43
+ plan: UlwLoopPlan,
44
+ ): boolean {
45
+ if (candidate.id === finalCandidate.id) return false;
46
+ if (candidate.steeringStatus === "superseded") {
47
+ const replacements = candidate.supersededBy ?? [];
48
+ if (replacements.length === 0) return true;
49
+ return !replacements.every((id) => {
50
+ if (id === finalCandidate.id) return true;
51
+ const replacement = plan.goals.find((goal) => goal.id === id);
52
+ return replacement !== undefined && isResolvedStatus(replacement.status);
53
+ });
54
+ }
55
+ return isCompletionBlocking(candidate, plan);
56
+ }
57
+
58
+ export function isUlwLoopDone(plan: UlwLoopPlan): boolean {
59
+ if (plan.aggregateCompletion?.status === "complete") return true;
60
+ return plan.goals.every((goal) => !isCompletionBlocking(goal, plan));
61
+ }
62
+
63
+ export function isFinalRunCompletionCandidate(plan: UlwLoopPlan, goal: UlwLoopItem): boolean {
64
+ return (
65
+ isCompletionBlocking(goal, plan) &&
66
+ plan.goals.every((candidate) => !isCompletionBlockingForFinalCandidate(candidate, goal, plan))
67
+ );
68
+ }
69
+
70
+ export function aggregateCodexObjective(plan: UlwLoopPlan): string {
71
+ return plan.codexObjective ?? ULW_LOOP_AGGREGATE_CODEX_OBJECTIVE;
72
+ }
73
+
74
+ export function expectedCodexObjective(plan: UlwLoopPlan, goal: UlwLoopItem): string {
75
+ return codexGoalMode(plan) === "aggregate" ? aggregateCodexObjective(plan) : goal.objective;
76
+ }
77
+
78
+ export function compatibleCodexObjectives(plan: UlwLoopPlan): readonly string[] {
79
+ return [aggregateCodexObjective(plan), ...(plan.codexObjectiveAliases ?? [])];
80
+ }
81
+
82
+ export function hasAllCriteriaPass(goal: UlwLoopItem): boolean {
83
+ return goal.successCriteria.length > 0 && goal.successCriteria.every((criterion) => criterion.status === "pass");
84
+ }
85
+
86
+ export function firstUnresolvedCriterion(goal: UlwLoopItem): UlwLoopSuccessCriterion | undefined {
87
+ return goal.successCriteria.find((criterion) => criterion.status !== "pass");
88
+ }
@@ -0,0 +1,73 @@
1
+ import { join } from "node:path";
2
+ import { ULW_LOOP_BRIEF, ULW_LOOP_DIR, ULW_LOOP_GOALS, ULW_LOOP_LEDGER } from "./types.js";
3
+
4
+ export interface UlwLoopScope {
5
+ readonly sessionId?: string | null;
6
+ }
7
+
8
+ const SESSION_ENV_KEYS = ["OMO_ULW_LOOP_SESSION_ID", "CODEX_SESSION_ID", "CODEX_THREAD_ID"] as const;
9
+ type EnvMap = Readonly<Record<string, string | undefined>>;
10
+
11
+ export function normalizeUlwLoopSessionId(sessionId: string | null | undefined): string | null {
12
+ const trimmed = sessionId?.trim();
13
+ if (!trimmed) return null;
14
+ const pathSegments = trimmed
15
+ .split(/[\\/]+/)
16
+ .filter((segment) => segment.length > 0 && segment !== "." && segment !== "..");
17
+ const candidate = (pathSegments.length > 0 ? pathSegments.join("-") : trimmed)
18
+ .replace(/[^A-Za-z0-9._-]+/g, "-")
19
+ .replace(/-+/g, "-")
20
+ .replace(/^\.+/, "")
21
+ .replace(/^[.-]+|[.-]+$/g, "");
22
+ return candidate.length > 0 ? candidate : null;
23
+ }
24
+
25
+ export function resolveUlwLoopSessionIdFromEnv(env: EnvMap = process.env): string | null {
26
+ for (const key of SESSION_ENV_KEYS) {
27
+ const normalized = normalizeUlwLoopSessionId(env[key]);
28
+ if (normalized !== null) return normalized;
29
+ }
30
+ return null;
31
+ }
32
+
33
+ export function ulwLoopRelativeDir(scope?: UlwLoopScope): string {
34
+ const sessionId = normalizeUlwLoopSessionId(scope?.sessionId);
35
+ return sessionId === null ? ULW_LOOP_DIR : `${ULW_LOOP_DIR}/${sessionId}`;
36
+ }
37
+
38
+ export function ulwLoopDir(repoRoot: string, scope?: UlwLoopScope): string {
39
+ return join(repoRoot, ulwLoopRelativeDir(scope));
40
+ }
41
+
42
+ export function ulwLoopBriefRelativePath(scope?: UlwLoopScope): string {
43
+ return `${ulwLoopRelativeDir(scope)}/${ULW_LOOP_BRIEF}`;
44
+ }
45
+
46
+ export function ulwLoopGoalsRelativePath(scope?: UlwLoopScope): string {
47
+ return `${ulwLoopRelativeDir(scope)}/${ULW_LOOP_GOALS}`;
48
+ }
49
+
50
+ export function ulwLoopLedgerRelativePath(scope?: UlwLoopScope): string {
51
+ return `${ulwLoopRelativeDir(scope)}/${ULW_LOOP_LEDGER}`;
52
+ }
53
+
54
+ export function ulwLoopBriefPath(repoRoot: string, scope?: UlwLoopScope): string {
55
+ return join(ulwLoopDir(repoRoot, scope), ULW_LOOP_BRIEF);
56
+ }
57
+
58
+ export function ulwLoopGoalsPath(repoRoot: string, scope?: UlwLoopScope): string {
59
+ return join(ulwLoopDir(repoRoot, scope), ULW_LOOP_GOALS);
60
+ }
61
+
62
+ export function ulwLoopLedgerPath(repoRoot: string, scope?: UlwLoopScope): string {
63
+ return join(ulwLoopDir(repoRoot, scope), ULW_LOOP_LEDGER);
64
+ }
65
+
66
+ export function repoRelative(absolutePath: string, repoRoot: string): string {
67
+ const slashPrefix = `${repoRoot}/`;
68
+ const backslashPrefix = `${repoRoot}\\`;
69
+ if (absolutePath.startsWith(slashPrefix)) return absolutePath.slice(slashPrefix.length).split("\\").join("/");
70
+ if (absolutePath.startsWith(backslashPrefix))
71
+ return absolutePath.slice(backslashPrefix.length).split("\\").join("/");
72
+ return absolutePath.split("\\").join("/");
73
+ }
@@ -0,0 +1,113 @@
1
+ // biome-ignore-all format: keep this port under the mandated pure LOC budget.
2
+ import { existsSync } from "node:fs";
3
+ import { mkdir, writeFile } from "node:fs/promises";
4
+
5
+ import { aggregateCodexObjectiveForScope } from "./goal-status.js";
6
+ import { type UlwLoopScope, ulwLoopBriefPath, ulwLoopBriefRelativePath, ulwLoopDir, ulwLoopGoalsPath, ulwLoopGoalsRelativePath, ulwLoopLedgerPath, ulwLoopLedgerRelativePath } from "./paths.js";
7
+ import { appendLedger, readUlwLoopPlan, withUlwLoopMutationLock, writePlan } from "./plan-io.js";
8
+ import type { UlwLoopCodexGoalMode, UlwLoopItem, UlwLoopPlan, UlwLoopSuccessCriterion } from "./types.js";
9
+ import { iso, UlwLoopError } from "./types.js";
10
+
11
+ export type UlwLoopPlanSummary = { readonly total: number; readonly pending: number; readonly in_progress: number; readonly complete: number; readonly failed: number; readonly blocked: number; readonly review_blocked: number; readonly needs_user_decision: number; readonly superseded: number; readonly criteria: { readonly total: number; readonly pass: number; readonly pending: number; readonly fail: number; readonly blocked: number } };
12
+
13
+ function cleanLine(line: string): string { return line.replace(/^\s*(?:[-*+]\s+|\d+[.)]\s+)/, "").trim(); }
14
+ function normalizeObjective(value: string): string { return value.replace(/\s+/g, " ").trim(); }
15
+ function titleFromObjective(objective: string, fallback: string): string { const firstLine = objective.split(/\r?\n/).map((line) => line.trim()).find(Boolean) ?? fallback; return firstLine.length > 72 ? `${firstLine.slice(0, 69).trimEnd()}...` : firstLine; }
16
+ function normalizeGoalId(title: string, index: number): string { const slug = title.toLowerCase().replace(/[^a-z0-9]+/g, "-").replace(/^-+|-+$/g, "").slice(0, 36).replace(/-+$/g, ""); return `G${String(index + 1).padStart(3, "0")}${slug ? `-${slug}` : ""}`; }
17
+ function assertNonEmpty(value: string | undefined, label: string): string { const trimmed = value?.trim(); if (!trimmed) throw new UlwLoopError(`Missing ${label}.`, "ULW_LOOP_ARGUMENT_MISSING"); return trimmed; }
18
+ function truncateObjective(objective: string): string { return objective.length > 80 ? `${objective.slice(0, 77).trimEnd()}...` : objective; }
19
+
20
+ export function seedDefaultSuccessCriteria(goalIndex: number, objective: string): UlwLoopSuccessCriterion[] {
21
+ const subject = truncateObjective(normalizeObjective(objective) || `Goal ${goalIndex + 1}`);
22
+ const rows = [
23
+ ["C001", "happy", `happy path for: ${subject}`, `Replace via revise_criterion with observable happy-path proof for goal ${goalIndex + 1}.`],
24
+ ["C002", "edge", "edge case (boundary/empty/malformed)", `Replace via revise_criterion with boundary or malformed-input proof for: ${subject}.`],
25
+ ["C003", "regression", "regression: adjacent surface still works", `Replace via revise_criterion with regression proof for neighboring behavior after: ${subject}.`],
26
+ ] as const;
27
+ return rows.map(([id, userModel, scenario, expectedEvidence]) => ({ id, scenario, userModel, expectedEvidence, capturedEvidence: null, status: "pending" }));
28
+ }
29
+
30
+ export function deriveGoalCandidates(brief: string): Array<{ title: string; objective: string }> {
31
+ const bulletGoals = brief.split(/\r?\n/).map((line) => ({ original: line, cleaned: normalizeObjective(cleanLine(line)) })).filter(({ cleaned }) => cleaned.length > 0 && cleaned.length <= 1200).filter(({ original, cleaned }, index, all) => /^\s*(?:[-*+]\s+|\d+[.)]\s+)/.test(original) && all.findIndex((candidate) => candidate.cleaned === cleaned) === index).map(({ cleaned }) => cleaned);
32
+ const paragraphs = brief.split(/\n\s*\n/).map(normalizeObjective).filter((paragraph) => paragraph.length > 0 && !paragraph.startsWith("#"));
33
+ const selected = (bulletGoals.length > 0 ? bulletGoals : paragraphs).length > 0 ? (bulletGoals.length > 0 ? bulletGoals : paragraphs) : ["Complete the requested project objective."];
34
+ return selected.map((objective, index) => ({ title: titleFromObjective(objective, `Goal ${index + 1}`), objective }));
35
+ }
36
+
37
+ function makeGoal(title: string, objective: string, index: number, now: string): UlwLoopItem {
38
+ const cleanTitle = assertNonEmpty(title, "title");
39
+ const cleanObjective = assertNonEmpty(objective, "objective");
40
+ return { id: normalizeGoalId(cleanTitle, index), title: cleanTitle, objective: cleanObjective, status: "pending", successCriteria: seedDefaultSuccessCriteria(index, cleanObjective), attempt: 0, createdAt: now, updatedAt: now };
41
+ }
42
+
43
+ function appendGoalToPlan(plan: UlwLoopPlan, title: string, objective: string, now: string): UlwLoopItem {
44
+ const goal = makeGoal(title, objective, plan.goals.length, now);
45
+ plan.goals.push(goal);
46
+ plan.updatedAt = now;
47
+ return goal;
48
+ }
49
+
50
+ function isScheduleEligible(goal: UlwLoopItem): boolean { return goal.steeringStatus !== "superseded" && goal.steeringStatus !== "blocked"; }
51
+
52
+ function clearGoalBlockerFields(goal: UlwLoopItem): void {
53
+ for (const key of ["blockedReason", "blockerSignature", "blockerOccurrenceCount", "requiredExternalDecision", "nonRetriable", "failedAt", "failureReason"] as const) delete goal[key];
54
+ }
55
+
56
+ export async function createUlwLoopPlan(repoRoot: string, args: { brief: string; codexGoalMode?: UlwLoopCodexGoalMode; force?: boolean }, scope?: UlwLoopScope): Promise<UlwLoopPlan> {
57
+ return withUlwLoopMutationLock(repoRoot, scope, async () => {
58
+ if (!args.force && existsSync(ulwLoopGoalsPath(repoRoot, scope))) throw new UlwLoopError(`Refusing to overwrite existing ${ulwLoopGoalsRelativePath(scope)}; pass --force to recreate it.`, "ULW_LOOP_PLAN_EXISTS");
59
+ const now = iso();
60
+ const goals = deriveGoalCandidates(args.brief).map((goal, index) => makeGoal(goal.title, goal.objective, index, now));
61
+ const plan: UlwLoopPlan = { version: 1, createdAt: now, updatedAt: now, briefPath: ulwLoopBriefRelativePath(scope), goalsPath: ulwLoopGoalsRelativePath(scope), ledgerPath: ulwLoopLedgerRelativePath(scope), codexGoalMode: args.codexGoalMode ?? "aggregate", goals };
62
+ if (plan.codexGoalMode === "aggregate") plan.codexObjective = aggregateCodexObjectiveForScope(scope);
63
+ await mkdir(ulwLoopDir(repoRoot, scope), { recursive: true });
64
+ await writeFile(ulwLoopBriefPath(repoRoot, scope), args.brief.endsWith("\n") ? args.brief : `${args.brief}\n`, "utf8");
65
+ await writePlan(repoRoot, plan, scope);
66
+ await writeFile(ulwLoopLedgerPath(repoRoot, scope), "", "utf8");
67
+ await appendLedger(repoRoot, { at: now, kind: "plan_created", message: `${goals.length} goal(s) created` }, scope);
68
+ return plan;
69
+ });
70
+ }
71
+
72
+ export async function addUlwLoopGoal(repoRoot: string, args: { title: string; objective: string }, scope?: UlwLoopScope): Promise<{ plan: UlwLoopPlan; goal: UlwLoopItem }> {
73
+ return withUlwLoopMutationLock(repoRoot, scope, async () => {
74
+ const plan = await readUlwLoopPlan(repoRoot, scope);
75
+ const now = iso();
76
+ const goal = appendGoalToPlan(plan, args.title, args.objective, now);
77
+ await writePlan(repoRoot, plan, scope);
78
+ await appendLedger(repoRoot, { at: now, kind: "goal_added", goalId: goal.id, status: goal.status, message: goal.title }, scope);
79
+ return { plan, goal };
80
+ });
81
+ }
82
+
83
+ export async function startNextUlwLoop(repoRoot: string, args: { retryFailed?: boolean } = {}, scope?: UlwLoopScope): Promise<{ plan: UlwLoopPlan; goal: UlwLoopItem; resumed: boolean } | { done: true; plan: UlwLoopPlan }> {
84
+ return withUlwLoopMutationLock(repoRoot, scope, async () => {
85
+ const plan = await readUlwLoopPlan(repoRoot, scope);
86
+ const now = iso();
87
+ if (plan.aggregateCompletion?.status === "complete") return { done: true, plan };
88
+ const existing = plan.goals.find((goal) => goal.status === "in_progress" && isScheduleEligible(goal));
89
+ if (existing) { await appendLedger(repoRoot, { at: now, kind: "goal_resumed", goalId: existing.id, status: existing.status, message: "Resuming active ulw-loop" }, scope); return { plan, goal: existing, resumed: true }; }
90
+ let next = plan.goals.find((goal) => goal.status === "pending" && isScheduleEligible(goal));
91
+ if (!next && args.retryFailed) {
92
+ next = plan.goals.find((goal) => goal.status === "failed" && !goal.nonRetriable && isScheduleEligible(goal));
93
+ if (next) await appendLedger(repoRoot, { at: now, kind: "goal_retried", goalId: next.id, status: "pending", ...(next.failureReason ? { message: next.failureReason } : {}) }, scope);
94
+ }
95
+ if (!next) return { done: true, plan };
96
+ next.status = "in_progress";
97
+ next.attempt += 1;
98
+ next.startedAt = now;
99
+ clearGoalBlockerFields(next);
100
+ next.updatedAt = now;
101
+ plan.activeGoalId = next.id;
102
+ plan.updatedAt = now;
103
+ await writePlan(repoRoot, plan, scope);
104
+ await appendLedger(repoRoot, { at: now, kind: "goal_started", goalId: next.id, status: next.status, message: `Attempt ${next.attempt}` }, scope);
105
+ return { plan, goal: next, resumed: false };
106
+ });
107
+ }
108
+
109
+ export function summarizeUlwLoopPlan(plan: UlwLoopPlan): UlwLoopPlanSummary {
110
+ const countStatus = (status: UlwLoopItem["status"]): number => plan.goals.filter((goal) => goal.status === status).length;
111
+ const countCriteria = (status: UlwLoopSuccessCriterion["status"]): number => plan.goals.reduce((sum, goal) => sum + goal.successCriteria.filter((criterion) => criterion.status === status).length, 0);
112
+ return { total: plan.goals.length, pending: countStatus("pending"), in_progress: countStatus("in_progress"), complete: countStatus("complete"), failed: countStatus("failed"), blocked: countStatus("blocked"), review_blocked: countStatus("review_blocked"), needs_user_decision: countStatus("needs_user_decision"), superseded: plan.goals.filter((goal) => goal.steeringStatus === "superseded").length, criteria: { total: plan.goals.reduce((sum, goal) => sum + goal.successCriteria.length, 0), pass: countCriteria("pass"), pending: countCriteria("pending"), fail: countCriteria("fail"), blocked: countCriteria("blocked") } };
113
+ }