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.
- package/Cargo.lock +5 -5
- package/Cargo.toml +1 -1
- package/README.md +2 -2
- package/crates/omx-explore/src/main.rs +434 -28
- package/dist/agents/__tests__/native-config.test.js +50 -0
- package/dist/agents/__tests__/native-config.test.js.map +1 -1
- package/dist/agents/native-config.d.ts.map +1 -1
- package/dist/agents/native-config.js +3 -2
- package/dist/agents/native-config.js.map +1 -1
- package/dist/cli/__tests__/codex-plugin-layout.test.js +1 -0
- package/dist/cli/__tests__/codex-plugin-layout.test.js.map +1 -1
- package/dist/cli/__tests__/doctor-warning-copy.test.js +1 -1
- package/dist/cli/__tests__/doctor-warning-copy.test.js.map +1 -1
- package/dist/cli/__tests__/explore.test.js +120 -3
- package/dist/cli/__tests__/explore.test.js.map +1 -1
- package/dist/cli/__tests__/imagegen-continuation.test.d.ts +2 -0
- package/dist/cli/__tests__/imagegen-continuation.test.d.ts.map +1 -0
- package/dist/cli/__tests__/imagegen-continuation.test.js +135 -0
- package/dist/cli/__tests__/imagegen-continuation.test.js.map +1 -0
- package/dist/cli/__tests__/index.test.js +182 -18
- package/dist/cli/__tests__/index.test.js.map +1 -1
- package/dist/cli/__tests__/launch-fallback.test.js +88 -2
- package/dist/cli/__tests__/launch-fallback.test.js.map +1 -1
- package/dist/cli/__tests__/ralph.test.js +62 -0
- package/dist/cli/__tests__/ralph.test.js.map +1 -1
- package/dist/cli/__tests__/setup-install-mode.test.js +48 -0
- package/dist/cli/__tests__/setup-install-mode.test.js.map +1 -1
- package/dist/cli/__tests__/setup-scope.test.js +12 -0
- package/dist/cli/__tests__/setup-scope.test.js.map +1 -1
- package/dist/cli/__tests__/team.test.js +465 -12
- package/dist/cli/__tests__/team.test.js.map +1 -1
- package/dist/cli/__tests__/ultragoal.test.js +50 -5
- package/dist/cli/__tests__/ultragoal.test.js.map +1 -1
- package/dist/cli/__tests__/uninstall.test.js +6 -2
- package/dist/cli/__tests__/uninstall.test.js.map +1 -1
- package/dist/cli/explore.d.ts.map +1 -1
- package/dist/cli/explore.js +211 -12
- package/dist/cli/explore.js.map +1 -1
- package/dist/cli/index.d.ts +11 -3
- package/dist/cli/index.d.ts.map +1 -1
- package/dist/cli/index.js +124 -18
- package/dist/cli/index.js.map +1 -1
- package/dist/cli/ralph.d.ts.map +1 -1
- package/dist/cli/ralph.js +37 -3
- package/dist/cli/ralph.js.map +1 -1
- package/dist/cli/setup.d.ts.map +1 -1
- package/dist/cli/setup.js +100 -9
- package/dist/cli/setup.js.map +1 -1
- package/dist/cli/team.d.ts +1 -0
- package/dist/cli/team.d.ts.map +1 -1
- package/dist/cli/team.js +42 -7
- package/dist/cli/team.js.map +1 -1
- package/dist/cli/ultragoal.d.ts +1 -1
- package/dist/cli/ultragoal.d.ts.map +1 -1
- package/dist/cli/ultragoal.js +29 -8
- package/dist/cli/ultragoal.js.map +1 -1
- package/dist/cli/uninstall.d.ts.map +1 -1
- package/dist/cli/uninstall.js +2 -1
- package/dist/cli/uninstall.js.map +1 -1
- package/dist/config/__tests__/codex-hooks.test.js +97 -2
- package/dist/config/__tests__/codex-hooks.test.js.map +1 -1
- package/dist/config/__tests__/generator-idempotent.test.js +1 -1
- package/dist/config/__tests__/generator-idempotent.test.js.map +1 -1
- package/dist/config/__tests__/generator-notify.test.js +22 -0
- package/dist/config/__tests__/generator-notify.test.js.map +1 -1
- package/dist/config/__tests__/models.test.js +18 -1
- package/dist/config/__tests__/models.test.js.map +1 -1
- package/dist/config/__tests__/wiki-config-contract.test.js +2 -1
- package/dist/config/__tests__/wiki-config-contract.test.js.map +1 -1
- package/dist/config/codex-hooks.d.ts +17 -3
- package/dist/config/codex-hooks.d.ts.map +1 -1
- package/dist/config/codex-hooks.js +102 -2
- package/dist/config/codex-hooks.js.map +1 -1
- package/dist/config/generator.d.ts +4 -1
- package/dist/config/generator.d.ts.map +1 -1
- package/dist/config/generator.js +69 -12
- package/dist/config/generator.js.map +1 -1
- package/dist/config/models.d.ts +6 -0
- package/dist/config/models.d.ts.map +1 -1
- package/dist/config/models.js +37 -0
- package/dist/config/models.js.map +1 -1
- package/dist/exec/followup.d.ts +1 -0
- package/dist/exec/followup.d.ts.map +1 -1
- package/dist/exec/followup.js +9 -3
- package/dist/exec/followup.js.map +1 -1
- package/dist/hooks/__tests__/anti-slop-workflow.test.js +19 -0
- package/dist/hooks/__tests__/anti-slop-workflow.test.js.map +1 -1
- package/dist/hooks/__tests__/consensus-execution-handoff.test.js +19 -2
- package/dist/hooks/__tests__/consensus-execution-handoff.test.js.map +1 -1
- package/dist/hooks/__tests__/deep-interview-contract.test.js +40 -0
- package/dist/hooks/__tests__/deep-interview-contract.test.js.map +1 -1
- package/dist/hooks/__tests__/foreground-isolation-contract.test.d.ts +2 -0
- package/dist/hooks/__tests__/foreground-isolation-contract.test.d.ts.map +1 -0
- package/dist/hooks/__tests__/foreground-isolation-contract.test.js +28 -0
- package/dist/hooks/__tests__/foreground-isolation-contract.test.js.map +1 -0
- package/dist/hooks/__tests__/keyword-detector.test.js +37 -25
- package/dist/hooks/__tests__/keyword-detector.test.js.map +1 -1
- package/dist/hooks/__tests__/notify-hook-auto-nudge.test.js +10 -4
- package/dist/hooks/__tests__/notify-hook-auto-nudge.test.js.map +1 -1
- package/dist/hooks/__tests__/session.test.js +32 -0
- package/dist/hooks/__tests__/session.test.js.map +1 -1
- package/dist/hooks/__tests__/wiki-docs-contract.test.js +6 -4
- package/dist/hooks/__tests__/wiki-docs-contract.test.js.map +1 -1
- package/dist/hooks/codebase-map.d.ts.map +1 -1
- package/dist/hooks/codebase-map.js +3 -2
- package/dist/hooks/codebase-map.js.map +1 -1
- package/dist/hooks/extensibility/dispatcher.d.ts.map +1 -1
- package/dist/hooks/extensibility/dispatcher.js +6 -4
- package/dist/hooks/extensibility/dispatcher.js.map +1 -1
- package/dist/hooks/extensibility/logging.d.ts.map +1 -1
- package/dist/hooks/extensibility/logging.js +3 -2
- package/dist/hooks/extensibility/logging.js.map +1 -1
- package/dist/hooks/extensibility/sdk/paths.d.ts.map +1 -1
- package/dist/hooks/extensibility/sdk/paths.js +4 -3
- package/dist/hooks/extensibility/sdk/paths.js.map +1 -1
- package/dist/hooks/keyword-detector.d.ts.map +1 -1
- package/dist/hooks/keyword-detector.js +2 -4
- package/dist/hooks/keyword-detector.js.map +1 -1
- package/dist/hooks/session.d.ts.map +1 -1
- package/dist/hooks/session.js +22 -12
- package/dist/hooks/session.js.map +1 -1
- package/dist/hud/__tests__/hud-tmux-injection.test.js +8 -7
- package/dist/hud/__tests__/hud-tmux-injection.test.js.map +1 -1
- package/dist/hud/__tests__/reconcile.test.js +1 -1
- package/dist/hud/__tests__/state.test.js +24 -0
- package/dist/hud/__tests__/state.test.js.map +1 -1
- package/dist/hud/index.js +1 -1
- package/dist/hud/index.js.map +1 -1
- package/dist/hud/state.d.ts.map +1 -1
- package/dist/hud/state.js +22 -8
- package/dist/hud/state.js.map +1 -1
- package/dist/hud/tmux.js +1 -1
- package/dist/hud/tmux.js.map +1 -1
- package/dist/imagegen/continuation.d.ts +44 -0
- package/dist/imagegen/continuation.d.ts.map +1 -0
- package/dist/imagegen/continuation.js +220 -0
- package/dist/imagegen/continuation.js.map +1 -0
- package/dist/mcp/__tests__/bootstrap.test.js +47 -2
- package/dist/mcp/__tests__/bootstrap.test.js.map +1 -1
- package/dist/mcp/__tests__/server-lifecycle.test.js +49 -1
- package/dist/mcp/__tests__/server-lifecycle.test.js.map +1 -1
- package/dist/mcp/__tests__/state-server.test.js +145 -6
- package/dist/mcp/__tests__/state-server.test.js.map +1 -1
- package/dist/mcp/__tests__/wiki-server.test.js +97 -1
- package/dist/mcp/__tests__/wiki-server.test.js.map +1 -1
- package/dist/mcp/bootstrap.d.ts +2 -0
- package/dist/mcp/bootstrap.d.ts.map +1 -1
- package/dist/mcp/bootstrap.js +95 -15
- package/dist/mcp/bootstrap.js.map +1 -1
- package/dist/mcp/lifecycle-telemetry.d.ts +16 -0
- package/dist/mcp/lifecycle-telemetry.d.ts.map +1 -0
- package/dist/mcp/lifecycle-telemetry.js +95 -0
- package/dist/mcp/lifecycle-telemetry.js.map +1 -0
- package/dist/mcp/wiki-server.d.ts.map +1 -1
- package/dist/mcp/wiki-server.js +11 -2
- package/dist/mcp/wiki-server.js.map +1 -1
- package/dist/pipeline/__tests__/stages.test.js +274 -5
- package/dist/pipeline/__tests__/stages.test.js.map +1 -1
- package/dist/pipeline/stages/team-exec.d.ts +2 -0
- package/dist/pipeline/stages/team-exec.d.ts.map +1 -1
- package/dist/pipeline/stages/team-exec.js +51 -26
- package/dist/pipeline/stages/team-exec.js.map +1 -1
- package/dist/planning/__tests__/artifacts.test.js +138 -3
- package/dist/planning/__tests__/artifacts.test.js.map +1 -1
- package/dist/planning/__tests__/context-pack-status.test.d.ts +2 -0
- package/dist/planning/__tests__/context-pack-status.test.d.ts.map +1 -0
- package/dist/planning/__tests__/context-pack-status.test.js +271 -0
- package/dist/planning/__tests__/context-pack-status.test.js.map +1 -0
- package/dist/planning/artifacts.d.ts +12 -1
- package/dist/planning/artifacts.d.ts.map +1 -1
- package/dist/planning/artifacts.js +32 -9
- package/dist/planning/artifacts.js.map +1 -1
- package/dist/planning/context-pack-status.d.ts +42 -0
- package/dist/planning/context-pack-status.d.ts.map +1 -0
- package/dist/planning/context-pack-status.js +479 -0
- package/dist/planning/context-pack-status.js.map +1 -0
- package/dist/runtime/__tests__/process-tree.test.d.ts +2 -0
- package/dist/runtime/__tests__/process-tree.test.d.ts.map +1 -0
- package/dist/runtime/__tests__/process-tree.test.js +107 -0
- package/dist/runtime/__tests__/process-tree.test.js.map +1 -0
- package/dist/runtime/process-tree.d.ts +28 -0
- package/dist/runtime/process-tree.d.ts.map +1 -0
- package/dist/runtime/process-tree.js +230 -0
- package/dist/runtime/process-tree.js.map +1 -0
- package/dist/scripts/__tests__/codex-native-hook.test.js +267 -2
- package/dist/scripts/__tests__/codex-native-hook.test.js.map +1 -1
- package/dist/scripts/__tests__/notify-state-io.test.d.ts +2 -0
- package/dist/scripts/__tests__/notify-state-io.test.d.ts.map +1 -0
- package/dist/scripts/__tests__/notify-state-io.test.js +40 -0
- package/dist/scripts/__tests__/notify-state-io.test.js.map +1 -0
- package/dist/scripts/codex-execution-surface.d.ts +1 -1
- package/dist/scripts/codex-execution-surface.d.ts.map +1 -1
- package/dist/scripts/codex-execution-surface.js.map +1 -1
- package/dist/scripts/codex-native-hook.d.ts +1 -1
- package/dist/scripts/codex-native-hook.d.ts.map +1 -1
- package/dist/scripts/codex-native-hook.js +141 -9
- package/dist/scripts/codex-native-hook.js.map +1 -1
- package/dist/scripts/notify-hook/managed-tmux.d.ts.map +1 -1
- package/dist/scripts/notify-hook/managed-tmux.js +6 -9
- package/dist/scripts/notify-hook/managed-tmux.js.map +1 -1
- package/dist/scripts/notify-hook/process-runner.d.ts.map +1 -1
- package/dist/scripts/notify-hook/process-runner.js +4 -1
- package/dist/scripts/notify-hook/process-runner.js.map +1 -1
- package/dist/scripts/notify-hook/state-io.d.ts.map +1 -1
- package/dist/scripts/notify-hook/state-io.js +4 -7
- package/dist/scripts/notify-hook/state-io.js.map +1 -1
- package/dist/scripts/notify-hook.js +25 -3
- package/dist/scripts/notify-hook.js.map +1 -1
- package/dist/scripts/verify-native-agents.d.ts.map +1 -1
- package/dist/scripts/verify-native-agents.js +3 -1
- package/dist/scripts/verify-native-agents.js.map +1 -1
- package/dist/sidecar/__tests__/tmux.test.js +1 -1
- package/dist/sidecar/__tests__/tmux.test.js.map +1 -1
- package/dist/sidecar/tmux.js +1 -1
- package/dist/sidecar/tmux.js.map +1 -1
- package/dist/state/__tests__/operations.test.js +79 -0
- package/dist/state/__tests__/operations.test.js.map +1 -1
- package/dist/state/__tests__/skill-active.test.js +10 -18
- package/dist/state/__tests__/skill-active.test.js.map +1 -1
- package/dist/state/__tests__/workflow-transition.test.js +45 -1
- package/dist/state/__tests__/workflow-transition.test.js.map +1 -1
- package/dist/state/operations.d.ts.map +1 -1
- package/dist/state/operations.js +1 -20
- package/dist/state/operations.js.map +1 -1
- package/dist/state/skill-active.d.ts +1 -0
- package/dist/state/skill-active.d.ts.map +1 -1
- package/dist/state/skill-active.js +28 -18
- package/dist/state/skill-active.js.map +1 -1
- package/dist/state/workflow-transition-reconcile.d.ts.map +1 -1
- package/dist/state/workflow-transition-reconcile.js +3 -2
- package/dist/state/workflow-transition-reconcile.js.map +1 -1
- package/dist/state/workflow-transition.js +2 -2
- package/dist/state/workflow-transition.js.map +1 -1
- package/dist/team/__tests__/approved-execution.test.js +96 -0
- package/dist/team/__tests__/approved-execution.test.js.map +1 -1
- package/dist/team/__tests__/followup-planner.test.js +16 -0
- package/dist/team/__tests__/followup-planner.test.js.map +1 -1
- package/dist/team/__tests__/model-contract.test.js +16 -0
- package/dist/team/__tests__/model-contract.test.js.map +1 -1
- package/dist/team/__tests__/repo-aware-decomposition.test.js +20 -0
- package/dist/team/__tests__/repo-aware-decomposition.test.js.map +1 -1
- package/dist/team/__tests__/runtime-cli.test.js +16 -0
- package/dist/team/__tests__/runtime-cli.test.js.map +1 -1
- package/dist/team/__tests__/runtime.test.js +209 -11
- package/dist/team/__tests__/runtime.test.js.map +1 -1
- package/dist/team/__tests__/scaling.test.js +110 -0
- package/dist/team/__tests__/scaling.test.js.map +1 -1
- package/dist/team/__tests__/tmux-session.test.js +9 -0
- package/dist/team/__tests__/tmux-session.test.js.map +1 -1
- package/dist/team/__tests__/worker-runtime-identity.test.js +6 -0
- package/dist/team/__tests__/worker-runtime-identity.test.js.map +1 -1
- package/dist/team/approved-execution.d.ts +13 -0
- package/dist/team/approved-execution.d.ts.map +1 -1
- package/dist/team/approved-execution.js +40 -22
- package/dist/team/approved-execution.js.map +1 -1
- package/dist/team/followup-planner.d.ts +1 -0
- package/dist/team/followup-planner.d.ts.map +1 -1
- package/dist/team/followup-planner.js +9 -9
- package/dist/team/followup-planner.js.map +1 -1
- package/dist/team/model-contract.d.ts +1 -1
- package/dist/team/model-contract.d.ts.map +1 -1
- package/dist/team/model-contract.js +4 -3
- package/dist/team/model-contract.js.map +1 -1
- package/dist/team/repo-aware-decomposition.d.ts +1 -0
- package/dist/team/repo-aware-decomposition.d.ts.map +1 -1
- package/dist/team/repo-aware-decomposition.js +5 -1
- package/dist/team/repo-aware-decomposition.js.map +1 -1
- package/dist/team/runtime-cli.d.ts +4 -0
- package/dist/team/runtime-cli.d.ts.map +1 -1
- package/dist/team/runtime-cli.js +14 -1
- package/dist/team/runtime-cli.js.map +1 -1
- package/dist/team/runtime.d.ts +1 -0
- package/dist/team/runtime.d.ts.map +1 -1
- package/dist/team/runtime.js +46 -16
- package/dist/team/runtime.js.map +1 -1
- package/dist/team/scaling.d.ts.map +1 -1
- package/dist/team/scaling.js +13 -6
- package/dist/team/scaling.js.map +1 -1
- package/dist/team/tmux-session.d.ts.map +1 -1
- package/dist/team/tmux-session.js +7 -0
- package/dist/team/tmux-session.js.map +1 -1
- package/dist/ultragoal/__tests__/artifacts.test.js +129 -4
- package/dist/ultragoal/__tests__/artifacts.test.js.map +1 -1
- package/dist/ultragoal/__tests__/docs-contract.test.d.ts +2 -0
- package/dist/ultragoal/__tests__/docs-contract.test.d.ts.map +1 -0
- package/dist/ultragoal/__tests__/docs-contract.test.js +31 -0
- package/dist/ultragoal/__tests__/docs-contract.test.js.map +1 -0
- package/dist/ultragoal/artifacts.d.ts +6 -2
- package/dist/ultragoal/artifacts.d.ts.map +1 -1
- package/dist/ultragoal/artifacts.js +108 -4
- package/dist/ultragoal/artifacts.js.map +1 -1
- package/dist/utils/paths.d.ts +3 -1
- package/dist/utils/paths.d.ts.map +1 -1
- package/dist/utils/paths.js +6 -2
- package/dist/utils/paths.js.map +1 -1
- package/dist/verification/__tests__/ci-rust-gates.test.js +44 -14
- package/dist/verification/__tests__/ci-rust-gates.test.js.map +1 -1
- package/dist/wiki/__tests__/ingest.test.js +35 -1
- package/dist/wiki/__tests__/ingest.test.js.map +1 -1
- package/dist/wiki/__tests__/lint.test.js +14 -1
- package/dist/wiki/__tests__/lint.test.js.map +1 -1
- package/dist/wiki/__tests__/query.test.js +28 -3
- package/dist/wiki/__tests__/query.test.js.map +1 -1
- package/dist/wiki/__tests__/session-hooks.test.js +30 -2
- package/dist/wiki/__tests__/session-hooks.test.js.map +1 -1
- package/dist/wiki/__tests__/storage.test.js +62 -22
- package/dist/wiki/__tests__/storage.test.js.map +1 -1
- package/dist/wiki/index.d.ts +2 -2
- package/dist/wiki/index.d.ts.map +1 -1
- package/dist/wiki/index.js +2 -2
- package/dist/wiki/index.js.map +1 -1
- package/dist/wiki/ingest.js +2 -2
- package/dist/wiki/ingest.js.map +1 -1
- package/dist/wiki/lifecycle.d.ts +5 -0
- package/dist/wiki/lifecycle.d.ts.map +1 -1
- package/dist/wiki/lifecycle.js +31 -4
- package/dist/wiki/lifecycle.js.map +1 -1
- package/dist/wiki/lint.d.ts.map +1 -1
- package/dist/wiki/lint.js +12 -8
- package/dist/wiki/lint.js.map +1 -1
- package/dist/wiki/query.d.ts.map +1 -1
- package/dist/wiki/query.js +3 -2
- package/dist/wiki/query.js.map +1 -1
- package/dist/wiki/storage.d.ts +4 -0
- package/dist/wiki/storage.d.ts.map +1 -1
- package/dist/wiki/storage.js +54 -18
- package/dist/wiki/storage.js.map +1 -1
- package/package.json +1 -1
- package/plugins/oh-my-codex/.codex-plugin/plugin.json +1 -1
- package/plugins/oh-my-codex/skills/ai-slop-cleaner/SKILL.md +9 -0
- package/plugins/oh-my-codex/skills/deep-interview/SKILL.md +25 -2
- package/plugins/oh-my-codex/skills/omx-setup/SKILL.md +1 -1
- package/plugins/oh-my-codex/skills/plan/SKILL.md +7 -4
- package/plugins/oh-my-codex/skills/ralplan/SKILL.md +13 -3
- package/plugins/oh-my-codex/skills/team/SKILL.md +2 -2
- package/plugins/oh-my-codex/skills/ultragoal/SKILL.md +11 -7
- package/plugins/oh-my-codex/skills/visual-ralph/SKILL.md +8 -0
- package/plugins/oh-my-codex/skills/wiki/SKILL.md +5 -5
- package/prompts/planner.md +1 -1
- package/skills/ai-slop-cleaner/SKILL.md +9 -0
- package/skills/deep-interview/SKILL.md +25 -2
- package/skills/omx-setup/SKILL.md +1 -1
- package/skills/plan/SKILL.md +7 -4
- package/skills/ralplan/SKILL.md +13 -3
- package/skills/team/SKILL.md +2 -2
- package/skills/ultragoal/SKILL.md +11 -7
- package/skills/visual-ralph/SKILL.md +8 -0
- package/skills/wiki/SKILL.md +5 -5
- package/src/scripts/__tests__/codex-native-hook.test.ts +302 -2
- package/src/scripts/__tests__/notify-state-io.test.ts +73 -0
- package/src/scripts/codex-execution-surface.ts +2 -0
- package/src/scripts/codex-native-hook.ts +163 -16
- package/src/scripts/notify-hook/managed-tmux.ts +6 -7
- package/src/scripts/notify-hook/process-runner.ts +4 -1
- package/src/scripts/notify-hook/state-io.ts +5 -7
- package/src/scripts/notify-hook.ts +26 -3
- 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
|
-
|
|
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 {
|