From 408c7ca142689136d887676def1bf41ea80bb2a9 Mon Sep 17 00:00:00 2001 From: Michael Bolin Date: Fri, 8 Aug 2025 16:09:39 -0700 Subject: [PATCH] chore: remove the TypeScript code from the repository (#2048) This deletes the bulk of the `codex-cli` folder and eliminates the logic that builds the TypeScript code and bundles it into the release. Since this PR modifies `.github/workflows/rust-release.yml`, to test changes to the release process, I locally commented out all of the "is this commit on upstream `main`" checks in `scripts/create_github_release.sh` and ran: ``` ./codex-rs/scripts/create_github_release.sh 0.20.0-alpha.4 ``` Which kicked off: https://github.com/openai/codex/actions/runs/16842085113 And the release artifacts appear legit! https://github.com/openai/codex/releases/tag/rust-v0.20.0-alpha.4 --- .github/workflows/ci.yml | 27 +- .github/workflows/codex.yml | 31 - .github/workflows/rust-release.yml | 26 - .husky/pre-commit | 1 - codex-cli/.editorconfig | 9 - codex-cli/.eslintrc.cjs | 107 - codex-cli/HUSKY.md | 45 - codex-cli/build.mjs | 88 - codex-cli/default.nix | 43 - codex-cli/examples/README.md | 44 - codex-cli/examples/build-codex-demo/run.sh | 65 - .../examples/build-codex-demo/runs/.gitkeep | 0 codex-cli/examples/build-codex-demo/task.yaml | 88 - codex-cli/examples/camerascii/run.sh | 68 - codex-cli/examples/camerascii/runs/.gitkeep | 0 codex-cli/examples/camerascii/task.yaml | 5 - .../camerascii/template/screenshot_details.md | 34 - codex-cli/examples/impossible-pong/run.sh | 68 - .../examples/impossible-pong/runs/.gitkeep | 0 codex-cli/examples/impossible-pong/task.yaml | 11 - .../impossible-pong/template/index.html | 233 - codex-cli/examples/prompt-analyzer/run.sh | 68 - .../examples/prompt-analyzer/runs/.gitkeep | 0 codex-cli/examples/prompt-analyzer/task.yaml | 17 - .../prompt-analyzer/template/Clustering.ipynb | 231 - .../prompt-analyzer/template/README.md | 103 - .../prompt-analyzer/template/analysis.md | 23 - .../template/analysis_dbscan.md | 22 - .../template/cluster_prompts.py | 547 -- .../template/plots/cluster_sizes.png | Bin 19000 -> 0 bytes .../prompt-analyzer/template/plots/tsne.png | Bin 102093 -> 0 bytes .../template/plots_dbscan/cluster_sizes.png | Bin 20441 -> 0 bytes .../template/plots_dbscan/tsne.png | Bin 96389 -> 0 bytes .../prompt-analyzer/template/prompts.csv | 214 - codex-cli/examples/prompting_guide.md | 117 - codex-cli/ignore-react-devtools-plugin.js | 16 - codex-cli/package.json | 71 +- codex-cli/require-shim.js | 11 - codex-cli/scripts/stage_release.sh | 8 - codex-cli/src/app.tsx | 108 - codex-cli/src/approvals.ts | 633 -- codex-cli/src/cli-singlepass.tsx | 28 - codex-cli/src/cli.tsx | 740 --- .../src/components/approval-mode-overlay.tsx | 47 - .../src/components/chat/message-history.tsx | 86 - .../src/components/chat/multiline-editor.tsx | 392 -- .../chat/terminal-chat-command-review.tsx | 256 - .../chat/terminal-chat-completions.tsx | 64 - .../chat/terminal-chat-input-thinking.tsx | 129 - .../components/chat/terminal-chat-input.tsx | 1017 ---- .../chat/terminal-chat-past-rollout.tsx | 68 - .../chat/terminal-chat-response-item.tsx | 360 -- .../chat/terminal-chat-tool-call-command.tsx | 143 - .../src/components/chat/terminal-chat.tsx | 766 --- .../src/components/chat/terminal-header.tsx | 99 - .../chat/terminal-message-history.tsx | 93 - .../components/chat/use-message-grouping.ts | 9 - codex-cli/src/components/diff-overlay.tsx | 93 - codex-cli/src/components/help-overlay.tsx | 103 - codex-cli/src/components/history-overlay.tsx | 255 - codex-cli/src/components/model-overlay.tsx | 165 - .../onboarding/onboarding-approval-mode.tsx | 35 - .../src/components/select-input/indicator.tsx | 21 - .../src/components/select-input/item.tsx | 13 - .../components/select-input/select-input.tsx | 189 - codex-cli/src/components/sessions-overlay.tsx | 130 - .../src/components/singlepass-cli-app.tsx | 677 --- .../src/components/typeahead-overlay.tsx | 166 - .../components/vendor/cli-spinners/index.js | 1293 ---- .../src/components/vendor/ink-select/index.js | 1 - .../vendor/ink-select/option-map.js | 26 - .../vendor/ink-select/select-option.js | 27 - .../components/vendor/ink-select/select.js | 53 - .../src/components/vendor/ink-select/theme.js | 32 - .../vendor/ink-select/use-select-state.js | 158 - .../vendor/ink-select/use-select.js | 17 - .../src/components/vendor/ink-spinner.tsx | 36 - .../src/components/vendor/ink-text-input.tsx | 428 -- codex-cli/src/format-command.ts | 53 - codex-cli/src/hooks/use-confirmation.ts | 67 - codex-cli/src/hooks/use-terminal-size.ts | 26 - codex-cli/src/parse-apply-patch.ts | 113 - codex-cli/src/shims-external.d.ts | 24 - codex-cli/src/text-buffer.ts | 977 --- codex-cli/src/typings.d.ts | 65 - codex-cli/src/utils/agent/agent-loop.ts | 1674 ----- codex-cli/src/utils/agent/apply-patch.ts | 815 --- codex-cli/src/utils/agent/exec.ts | 138 - .../src/utils/agent/handle-exec-command.ts | 377 -- .../src/utils/agent/parse-apply-patch.ts | 112 - .../src/utils/agent/platform-commands.ts | 82 - codex-cli/src/utils/agent/review.ts | 14 - .../sandbox/create-truncating-collector.ts | 77 - .../src/utils/agent/sandbox/interface.ts | 30 - codex-cli/src/utils/agent/sandbox/landlock.ts | 175 - .../src/utils/agent/sandbox/macos-seatbelt.ts | 154 - codex-cli/src/utils/agent/sandbox/raw-exec.ts | 238 - .../src/utils/approximate-tokens-used.ts | 55 - codex-cli/src/utils/auto-approval-mode.js | 9 - codex-cli/src/utils/auto-approval-mode.ts | 10 - codex-cli/src/utils/bug-report.ts | 82 - codex-cli/src/utils/check-in-git.ts | 31 - codex-cli/src/utils/check-updates.ts | 146 - codex-cli/src/utils/compact-summary.ts | 70 - codex-cli/src/utils/config.ts | 597 -- .../src/utils/extract-applied-patches.ts | 36 - .../src/utils/file-system-suggestions.ts | 59 - codex-cli/src/utils/file-tag-utils.ts | 62 - .../src/utils/get-api-key-components.tsx | 75 - codex-cli/src/utils/get-api-key.tsx | 766 --- codex-cli/src/utils/get-diff.ts | 129 - codex-cli/src/utils/input-utils.ts | 39 - codex-cli/src/utils/logger/log.ts | 137 - codex-cli/src/utils/model-info.ts | 202 - codex-cli/src/utils/model-utils.ts | 196 - codex-cli/src/utils/openai-client.ts | 51 - .../src/utils/package-manager-detector.ts | 73 - codex-cli/src/utils/parsers.ts | 113 - codex-cli/src/utils/providers.ts | 55 - codex-cli/src/utils/responses.ts | 717 --- codex-cli/src/utils/session.ts | 52 - codex-cli/src/utils/short-path.ts | 27 - codex-cli/src/utils/singlepass/code_diff.ts | 190 - codex-cli/src/utils/singlepass/context.ts | 65 - .../src/utils/singlepass/context_files.ts | 409 -- .../src/utils/singlepass/context_limit.ts | 208 - codex-cli/src/utils/singlepass/file_ops.ts | 47 - codex-cli/src/utils/slash-commands.ts | 36 - .../src/utils/storage/command-history.ts | 139 - codex-cli/src/utils/storage/save-rollout.ts | 52 - codex-cli/src/utils/terminal-chat-utils.ts | 0 codex-cli/src/utils/terminal.ts | 84 - codex-cli/src/version.ts | 8 - codex-cli/tests/__fixtures__/a.txt | 1 - codex-cli/tests/__fixtures__/b.txt | 1 - .../__snapshots__/check-updates.test.ts.snap | 12 - .../agent-azure-responses-endpoint.test.ts | 107 - codex-cli/tests/agent-cancel-early.test.ts | 128 - .../tests/agent-cancel-prev-response.test.ts | 149 - codex-cli/tests/agent-cancel-race.test.ts | 137 - codex-cli/tests/agent-cancel.test.ts | 171 - codex-cli/tests/agent-dedupe-items.test.ts | 115 - .../tests/agent-function-call-id.test.ts | 150 - .../tests/agent-generic-network-error.test.ts | 134 - .../tests/agent-interrupt-continue.test.ts | 148 - .../tests/agent-invalid-request-error.test.ts | 89 - .../tests/agent-max-tokens-error.test.ts | 93 - codex-cli/tests/agent-network-errors.test.ts | 181 - codex-cli/tests/agent-project-doc.test.ts | 142 - .../tests/agent-rate-limit-error.test.ts | 128 - codex-cli/tests/agent-server-retry.test.ts | 168 - codex-cli/tests/agent-terminate.test.ts | 180 - codex-cli/tests/agent-thinking-time.test.ts | 174 - codex-cli/tests/api-key.test.ts | 35 - codex-cli/tests/apply-patch.test.ts | 346 -- codex-cli/tests/approvals.test.ts | 219 - codex-cli/tests/cancel-exec.test.ts | 56 - codex-cli/tests/check-updates.test.ts | 178 - codex-cli/tests/clear-command.test.tsx | 120 - codex-cli/tests/config.test.tsx | 363 -- codex-cli/tests/config_reasoning.test.ts | 121 - .../tests/create-truncating-collector.test.ts | 55 - .../disableResponseStorage.agentLoop.test.ts | 93 - .../tests/disableResponseStorage.test.ts | 46 - codex-cli/tests/dummy.test.ts | 4 - codex-cli/tests/exec-apply-patch.test.ts | 44 - .../tests/file-system-suggestions.test.ts | 92 - codex-cli/tests/file-tag-utils.test.ts | 240 - codex-cli/tests/fixed-requires-shell.test.ts | 38 - codex-cli/tests/format-command.test.ts | 21 - .../tests/get-diff-special-chars.test.ts | 28 - codex-cli/tests/history-overlay.test.tsx | 350 -- codex-cli/tests/input-utils.test.ts | 47 - .../tests/invalid-command-handling.test.ts | 68 - codex-cli/tests/markdown.test.tsx | 172 - codex-cli/tests/model-info.test.ts | 19 - .../tests/model-utils-network-error.test.ts | 76 - codex-cli/tests/model-utils.test.ts | 78 - .../multiline-ctrl-enter-submit.test.tsx | 41 - .../tests/multiline-dynamic-width.test.tsx | 77 - .../tests/multiline-enter-submit-cr.test.tsx | 41 - .../tests/multiline-history-behavior.test.tsx | 180 - codex-cli/tests/multiline-input-test.ts | 164 - codex-cli/tests/multiline-newline.test.tsx | 56 - .../tests/multiline-shift-enter-crlf.test.tsx | 51 - .../tests/multiline-shift-enter-mod1.test.tsx | 49 - .../tests/multiline-shift-enter.test.tsx | 49 - .../tests/package-manager-detector.test.ts | 66 - codex-cli/tests/parse-apply-patch.test.ts | 45 - codex-cli/tests/pipe-command.test.ts | 19 - codex-cli/tests/project-doc.test.ts | 57 - .../tests/raw-exec-process-group.test.ts | 87 - codex-cli/tests/requires-shell.test.ts | 49 - .../tests/responses-chat-completions.test.ts | 815 --- codex-cli/tests/slash-commands.test.ts | 40 - .../tests/terminal-chat-completions.test.tsx | 46 - .../terminal-chat-input-compact.test.tsx | 34 - ...l-chat-input-file-tag-suggestions.test.tsx | 207 - .../terminal-chat-input-multiline.test.tsx | 130 - .../terminal-chat-model-selection.test.tsx | 130 - .../terminal-chat-response-item.test.tsx | 65 - .../tests/text-buffer-copy-paste.test.ts | 50 - codex-cli/tests/text-buffer-crlf.test.ts | 14 - codex-cli/tests/text-buffer-gaps.test.ts | 250 - codex-cli/tests/text-buffer-word.test.ts | 137 - codex-cli/tests/text-buffer.test.ts | 291 - .../tests/token-streaming-performance.test.ts | 110 - codex-cli/tests/typeahead-scroll.test.tsx | 68 - codex-cli/tests/ui-test-helpers.tsx | 28 - codex-cli/tests/user-config-env.test.ts | 62 - codex-cli/tsconfig.json | 34 - codex-cli/vitest.config.ts | 12 - package.json | 29 +- patches/marked-terminal@7.3.0.patch | 26 - pnpm-lock.yaml | 5397 ----------------- pnpm-workspace.yaml | 5 - 216 files changed, 3 insertions(+), 35960 deletions(-) delete mode 100644 .husky/pre-commit delete mode 100644 codex-cli/.editorconfig delete mode 100644 codex-cli/.eslintrc.cjs delete mode 100644 codex-cli/HUSKY.md delete mode 100644 codex-cli/build.mjs delete mode 100644 codex-cli/default.nix delete mode 100644 codex-cli/examples/README.md delete mode 100755 codex-cli/examples/build-codex-demo/run.sh delete mode 100644 codex-cli/examples/build-codex-demo/runs/.gitkeep delete mode 100644 codex-cli/examples/build-codex-demo/task.yaml delete mode 100755 codex-cli/examples/camerascii/run.sh delete mode 100644 codex-cli/examples/camerascii/runs/.gitkeep delete mode 100644 codex-cli/examples/camerascii/task.yaml delete mode 100644 codex-cli/examples/camerascii/template/screenshot_details.md delete mode 100755 codex-cli/examples/impossible-pong/run.sh delete mode 100644 codex-cli/examples/impossible-pong/runs/.gitkeep delete mode 100644 codex-cli/examples/impossible-pong/task.yaml delete mode 100644 codex-cli/examples/impossible-pong/template/index.html delete mode 100755 codex-cli/examples/prompt-analyzer/run.sh delete mode 100644 codex-cli/examples/prompt-analyzer/runs/.gitkeep delete mode 100644 codex-cli/examples/prompt-analyzer/task.yaml delete mode 100644 codex-cli/examples/prompt-analyzer/template/Clustering.ipynb delete mode 100644 codex-cli/examples/prompt-analyzer/template/README.md delete mode 100644 codex-cli/examples/prompt-analyzer/template/analysis.md delete mode 100644 codex-cli/examples/prompt-analyzer/template/analysis_dbscan.md delete mode 100644 codex-cli/examples/prompt-analyzer/template/cluster_prompts.py delete mode 100644 codex-cli/examples/prompt-analyzer/template/plots/cluster_sizes.png delete mode 100644 codex-cli/examples/prompt-analyzer/template/plots/tsne.png delete mode 100644 codex-cli/examples/prompt-analyzer/template/plots_dbscan/cluster_sizes.png delete mode 100644 codex-cli/examples/prompt-analyzer/template/plots_dbscan/tsne.png delete mode 100644 codex-cli/examples/prompt-analyzer/template/prompts.csv delete mode 100644 codex-cli/examples/prompting_guide.md delete mode 100644 codex-cli/ignore-react-devtools-plugin.js delete mode 100644 codex-cli/require-shim.js delete mode 100644 codex-cli/src/app.tsx delete mode 100644 codex-cli/src/approvals.ts delete mode 100644 codex-cli/src/cli-singlepass.tsx delete mode 100644 codex-cli/src/cli.tsx delete mode 100644 codex-cli/src/components/approval-mode-overlay.tsx delete mode 100644 codex-cli/src/components/chat/message-history.tsx delete mode 100644 codex-cli/src/components/chat/multiline-editor.tsx delete mode 100644 codex-cli/src/components/chat/terminal-chat-command-review.tsx delete mode 100644 codex-cli/src/components/chat/terminal-chat-completions.tsx delete mode 100644 codex-cli/src/components/chat/terminal-chat-input-thinking.tsx delete mode 100644 codex-cli/src/components/chat/terminal-chat-input.tsx delete mode 100644 codex-cli/src/components/chat/terminal-chat-past-rollout.tsx delete mode 100644 codex-cli/src/components/chat/terminal-chat-response-item.tsx delete mode 100644 codex-cli/src/components/chat/terminal-chat-tool-call-command.tsx delete mode 100644 codex-cli/src/components/chat/terminal-chat.tsx delete mode 100644 codex-cli/src/components/chat/terminal-header.tsx delete mode 100644 codex-cli/src/components/chat/terminal-message-history.tsx delete mode 100644 codex-cli/src/components/chat/use-message-grouping.ts delete mode 100644 codex-cli/src/components/diff-overlay.tsx delete mode 100644 codex-cli/src/components/help-overlay.tsx delete mode 100644 codex-cli/src/components/history-overlay.tsx delete mode 100644 codex-cli/src/components/model-overlay.tsx delete mode 100644 codex-cli/src/components/onboarding/onboarding-approval-mode.tsx delete mode 100644 codex-cli/src/components/select-input/indicator.tsx delete mode 100644 codex-cli/src/components/select-input/item.tsx delete mode 100644 codex-cli/src/components/select-input/select-input.tsx delete mode 100644 codex-cli/src/components/sessions-overlay.tsx delete mode 100644 codex-cli/src/components/singlepass-cli-app.tsx delete mode 100644 codex-cli/src/components/typeahead-overlay.tsx delete mode 100644 codex-cli/src/components/vendor/cli-spinners/index.js delete mode 100644 codex-cli/src/components/vendor/ink-select/index.js delete mode 100644 codex-cli/src/components/vendor/ink-select/option-map.js delete mode 100644 codex-cli/src/components/vendor/ink-select/select-option.js delete mode 100644 codex-cli/src/components/vendor/ink-select/select.js delete mode 100644 codex-cli/src/components/vendor/ink-select/theme.js delete mode 100644 codex-cli/src/components/vendor/ink-select/use-select-state.js delete mode 100644 codex-cli/src/components/vendor/ink-select/use-select.js delete mode 100644 codex-cli/src/components/vendor/ink-spinner.tsx delete mode 100644 codex-cli/src/components/vendor/ink-text-input.tsx delete mode 100644 codex-cli/src/format-command.ts delete mode 100644 codex-cli/src/hooks/use-confirmation.ts delete mode 100644 codex-cli/src/hooks/use-terminal-size.ts delete mode 100644 codex-cli/src/parse-apply-patch.ts delete mode 100644 codex-cli/src/shims-external.d.ts delete mode 100644 codex-cli/src/text-buffer.ts delete mode 100644 codex-cli/src/typings.d.ts delete mode 100644 codex-cli/src/utils/agent/agent-loop.ts delete mode 100644 codex-cli/src/utils/agent/apply-patch.ts delete mode 100644 codex-cli/src/utils/agent/exec.ts delete mode 100644 codex-cli/src/utils/agent/handle-exec-command.ts delete mode 100644 codex-cli/src/utils/agent/parse-apply-patch.ts delete mode 100644 codex-cli/src/utils/agent/platform-commands.ts delete mode 100644 codex-cli/src/utils/agent/review.ts delete mode 100644 codex-cli/src/utils/agent/sandbox/create-truncating-collector.ts delete mode 100644 codex-cli/src/utils/agent/sandbox/interface.ts delete mode 100644 codex-cli/src/utils/agent/sandbox/landlock.ts delete mode 100644 codex-cli/src/utils/agent/sandbox/macos-seatbelt.ts delete mode 100644 codex-cli/src/utils/agent/sandbox/raw-exec.ts delete mode 100644 codex-cli/src/utils/approximate-tokens-used.ts delete mode 100644 codex-cli/src/utils/auto-approval-mode.js delete mode 100644 codex-cli/src/utils/auto-approval-mode.ts delete mode 100644 codex-cli/src/utils/bug-report.ts delete mode 100644 codex-cli/src/utils/check-in-git.ts delete mode 100644 codex-cli/src/utils/check-updates.ts delete mode 100644 codex-cli/src/utils/compact-summary.ts delete mode 100644 codex-cli/src/utils/config.ts delete mode 100644 codex-cli/src/utils/extract-applied-patches.ts delete mode 100644 codex-cli/src/utils/file-system-suggestions.ts delete mode 100644 codex-cli/src/utils/file-tag-utils.ts delete mode 100644 codex-cli/src/utils/get-api-key-components.tsx delete mode 100644 codex-cli/src/utils/get-api-key.tsx delete mode 100644 codex-cli/src/utils/get-diff.ts delete mode 100644 codex-cli/src/utils/input-utils.ts delete mode 100644 codex-cli/src/utils/logger/log.ts delete mode 100644 codex-cli/src/utils/model-info.ts delete mode 100644 codex-cli/src/utils/model-utils.ts delete mode 100644 codex-cli/src/utils/openai-client.ts delete mode 100644 codex-cli/src/utils/package-manager-detector.ts delete mode 100644 codex-cli/src/utils/parsers.ts delete mode 100644 codex-cli/src/utils/providers.ts delete mode 100644 codex-cli/src/utils/responses.ts delete mode 100644 codex-cli/src/utils/session.ts delete mode 100644 codex-cli/src/utils/short-path.ts delete mode 100644 codex-cli/src/utils/singlepass/code_diff.ts delete mode 100644 codex-cli/src/utils/singlepass/context.ts delete mode 100644 codex-cli/src/utils/singlepass/context_files.ts delete mode 100644 codex-cli/src/utils/singlepass/context_limit.ts delete mode 100644 codex-cli/src/utils/singlepass/file_ops.ts delete mode 100644 codex-cli/src/utils/slash-commands.ts delete mode 100644 codex-cli/src/utils/storage/command-history.ts delete mode 100644 codex-cli/src/utils/storage/save-rollout.ts delete mode 100644 codex-cli/src/utils/terminal-chat-utils.ts delete mode 100644 codex-cli/src/utils/terminal.ts delete mode 100644 codex-cli/src/version.ts delete mode 100644 codex-cli/tests/__fixtures__/a.txt delete mode 100644 codex-cli/tests/__fixtures__/b.txt delete mode 100644 codex-cli/tests/__snapshots__/check-updates.test.ts.snap delete mode 100644 codex-cli/tests/agent-azure-responses-endpoint.test.ts delete mode 100644 codex-cli/tests/agent-cancel-early.test.ts delete mode 100644 codex-cli/tests/agent-cancel-prev-response.test.ts delete mode 100644 codex-cli/tests/agent-cancel-race.test.ts delete mode 100644 codex-cli/tests/agent-cancel.test.ts delete mode 100644 codex-cli/tests/agent-dedupe-items.test.ts delete mode 100644 codex-cli/tests/agent-function-call-id.test.ts delete mode 100644 codex-cli/tests/agent-generic-network-error.test.ts delete mode 100644 codex-cli/tests/agent-interrupt-continue.test.ts delete mode 100644 codex-cli/tests/agent-invalid-request-error.test.ts delete mode 100644 codex-cli/tests/agent-max-tokens-error.test.ts delete mode 100644 codex-cli/tests/agent-network-errors.test.ts delete mode 100644 codex-cli/tests/agent-project-doc.test.ts delete mode 100644 codex-cli/tests/agent-rate-limit-error.test.ts delete mode 100644 codex-cli/tests/agent-server-retry.test.ts delete mode 100644 codex-cli/tests/agent-terminate.test.ts delete mode 100644 codex-cli/tests/agent-thinking-time.test.ts delete mode 100644 codex-cli/tests/api-key.test.ts delete mode 100644 codex-cli/tests/apply-patch.test.ts delete mode 100644 codex-cli/tests/approvals.test.ts delete mode 100644 codex-cli/tests/cancel-exec.test.ts delete mode 100644 codex-cli/tests/check-updates.test.ts delete mode 100644 codex-cli/tests/clear-command.test.tsx delete mode 100644 codex-cli/tests/config.test.tsx delete mode 100644 codex-cli/tests/config_reasoning.test.ts delete mode 100644 codex-cli/tests/create-truncating-collector.test.ts delete mode 100644 codex-cli/tests/disableResponseStorage.agentLoop.test.ts delete mode 100644 codex-cli/tests/disableResponseStorage.test.ts delete mode 100644 codex-cli/tests/dummy.test.ts delete mode 100644 codex-cli/tests/exec-apply-patch.test.ts delete mode 100644 codex-cli/tests/file-system-suggestions.test.ts delete mode 100644 codex-cli/tests/file-tag-utils.test.ts delete mode 100644 codex-cli/tests/fixed-requires-shell.test.ts delete mode 100644 codex-cli/tests/format-command.test.ts delete mode 100644 codex-cli/tests/get-diff-special-chars.test.ts delete mode 100644 codex-cli/tests/history-overlay.test.tsx delete mode 100644 codex-cli/tests/input-utils.test.ts delete mode 100644 codex-cli/tests/invalid-command-handling.test.ts delete mode 100644 codex-cli/tests/markdown.test.tsx delete mode 100644 codex-cli/tests/model-info.test.ts delete mode 100644 codex-cli/tests/model-utils-network-error.test.ts delete mode 100644 codex-cli/tests/model-utils.test.ts delete mode 100644 codex-cli/tests/multiline-ctrl-enter-submit.test.tsx delete mode 100644 codex-cli/tests/multiline-dynamic-width.test.tsx delete mode 100644 codex-cli/tests/multiline-enter-submit-cr.test.tsx delete mode 100644 codex-cli/tests/multiline-history-behavior.test.tsx delete mode 100644 codex-cli/tests/multiline-input-test.ts delete mode 100644 codex-cli/tests/multiline-newline.test.tsx delete mode 100644 codex-cli/tests/multiline-shift-enter-crlf.test.tsx delete mode 100644 codex-cli/tests/multiline-shift-enter-mod1.test.tsx delete mode 100644 codex-cli/tests/multiline-shift-enter.test.tsx delete mode 100644 codex-cli/tests/package-manager-detector.test.ts delete mode 100644 codex-cli/tests/parse-apply-patch.test.ts delete mode 100644 codex-cli/tests/pipe-command.test.ts delete mode 100644 codex-cli/tests/project-doc.test.ts delete mode 100644 codex-cli/tests/raw-exec-process-group.test.ts delete mode 100644 codex-cli/tests/requires-shell.test.ts delete mode 100644 codex-cli/tests/responses-chat-completions.test.ts delete mode 100644 codex-cli/tests/slash-commands.test.ts delete mode 100644 codex-cli/tests/terminal-chat-completions.test.tsx delete mode 100644 codex-cli/tests/terminal-chat-input-compact.test.tsx delete mode 100644 codex-cli/tests/terminal-chat-input-file-tag-suggestions.test.tsx delete mode 100644 codex-cli/tests/terminal-chat-input-multiline.test.tsx delete mode 100644 codex-cli/tests/terminal-chat-model-selection.test.tsx delete mode 100644 codex-cli/tests/terminal-chat-response-item.test.tsx delete mode 100644 codex-cli/tests/text-buffer-copy-paste.test.ts delete mode 100644 codex-cli/tests/text-buffer-crlf.test.ts delete mode 100644 codex-cli/tests/text-buffer-gaps.test.ts delete mode 100644 codex-cli/tests/text-buffer-word.test.ts delete mode 100644 codex-cli/tests/text-buffer.test.ts delete mode 100644 codex-cli/tests/token-streaming-performance.test.ts delete mode 100644 codex-cli/tests/typeahead-scroll.test.tsx delete mode 100644 codex-cli/tests/ui-test-helpers.tsx delete mode 100644 codex-cli/tests/user-config-env.test.ts delete mode 100644 codex-cli/tsconfig.json delete mode 100644 codex-cli/vitest.config.ts delete mode 100644 patches/marked-terminal@7.3.0.patch diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 9d8675fa..32fb0709 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -44,35 +44,10 @@ jobs: # Run all tasks using workspace filters - - name: Check TypeScript code formatting - working-directory: codex-cli - run: pnpm run format - - - name: Check Markdown and config file formatting - run: pnpm run format - - - name: Run tests - run: pnpm run test - - - name: Lint - run: | - pnpm --filter @openai/codex exec -- eslint src tests --ext ts --ext tsx \ - --report-unused-disable-directives \ - --rule "no-console:error" \ - --rule "no-debugger:error" \ - --max-warnings=-1 - - - name: Type-check - run: pnpm run typecheck - - - name: Build - run: pnpm run build - - name: Ensure staging a release works. - working-directory: codex-cli env: GH_TOKEN: ${{ github.token }} - run: pnpm stage-release + run: ./codex-cli/scripts/stage_release.sh - name: Ensure root README.md contains only ASCII and certain Unicode code points run: ./scripts/asciicheck.py README.md diff --git a/.github/workflows/codex.yml b/.github/workflows/codex.yml index 18fe74cc..6ca0d57f 100644 --- a/.github/workflows/codex.yml +++ b/.github/workflows/codex.yml @@ -39,37 +39,6 @@ jobs: - name: Checkout repository uses: actions/checkout@v4 - # We install the dependencies like we would for an ordinary CI job, - # particularly because Codex will not have network access to install - # these dependencies. - - name: Setup Node.js - uses: actions/setup-node@v4 - with: - node-version: 22 - - - name: Setup pnpm - uses: pnpm/action-setup@v4 - with: - version: 10.8.1 - run_install: false - - - name: Get pnpm store directory - id: pnpm-cache - shell: bash - run: | - echo "store_path=$(pnpm store path --silent)" >> $GITHUB_OUTPUT - - - name: Setup pnpm cache - uses: actions/cache@v4 - with: - path: ${{ steps.pnpm-cache.outputs.store_path }} - key: ${{ runner.os }}-pnpm-store-${{ hashFiles('**/pnpm-lock.yaml') }} - restore-keys: | - ${{ runner.os }}-pnpm-store- - - - name: Install dependencies - run: pnpm install - - uses: dtolnay/rust-toolchain@1.88 with: targets: x86_64-unknown-linux-gnu diff --git a/.github/workflows/rust-release.yml b/.github/workflows/rust-release.yml index fc0938e2..8e5aef27 100644 --- a/.github/workflows/rust-release.yml +++ b/.github/workflows/rust-release.yml @@ -172,32 +172,6 @@ jobs: version="${GITHUB_REF_NAME#rust-v}" echo "name=${version}" >> $GITHUB_OUTPUT - # Setup Node + pnpm similar to ci.yml so we can build the npm package - - name: Setup Node.js - uses: actions/setup-node@v4 - with: - node-version: 22 - - - name: Setup pnpm - uses: pnpm/action-setup@v4 - with: - version: 10.8.1 - run_install: false - - - name: Get pnpm store directory - id: pnpm-cache - shell: bash - run: | - echo "store_path=$(pnpm store path --silent)" >> $GITHUB_OUTPUT - - - name: Setup pnpm cache - uses: actions/cache@v4 - with: - path: ${{ steps.pnpm-cache.outputs.store_path }} - key: ${{ runner.os }}-pnpm-store-${{ hashFiles('**/pnpm-lock.yaml') }} - restore-keys: | - ${{ runner.os }}-pnpm-store- - - name: Stage npm package env: GH_TOKEN: ${{ github.token }} diff --git a/.husky/pre-commit b/.husky/pre-commit deleted file mode 100644 index e02c24e2..00000000 --- a/.husky/pre-commit +++ /dev/null @@ -1 +0,0 @@ -pnpm lint-staged \ No newline at end of file diff --git a/codex-cli/.editorconfig b/codex-cli/.editorconfig deleted file mode 100644 index f55d4205..00000000 --- a/codex-cli/.editorconfig +++ /dev/null @@ -1,9 +0,0 @@ -root = true - -[*] -indent_style = space -indent_size = 2 - -[*.{js,ts,jsx,tsx}] -indent_style = space -indent_size = 2 \ No newline at end of file diff --git a/codex-cli/.eslintrc.cjs b/codex-cli/.eslintrc.cjs deleted file mode 100644 index a623d2ed..00000000 --- a/codex-cli/.eslintrc.cjs +++ /dev/null @@ -1,107 +0,0 @@ -module.exports = { - root: true, - env: { browser: true, node: true, es2020: true }, - extends: [ - "eslint:recommended", - "plugin:@typescript-eslint/recommended", - "plugin:react-hooks/recommended", - ], - ignorePatterns: [ - ".eslintrc.cjs", - "build.mjs", - "dist", - "vite.config.ts", - "src/components/vendor", - ], - parser: "@typescript-eslint/parser", - parserOptions: { - tsconfigRootDir: __dirname, - project: ["./tsconfig.json"], - }, - plugins: ["import", "react-hooks", "react-refresh"], - rules: { - // Imports - "@typescript-eslint/consistent-type-imports": "error", - "import/no-cycle": ["error", { maxDepth: 1 }], - "import/no-duplicates": "error", - "import/order": [ - "error", - { - groups: ["type"], - "newlines-between": "always", - alphabetize: { - order: "asc", - caseInsensitive: false, - }, - }, - ], - // We use the import/ plugin instead. - "sort-imports": "off", - - "@typescript-eslint/array-type": ["error", { default: "generic" }], - // FIXME(mbolin): Introduce this. - // "@typescript-eslint/explicit-function-return-type": "error", - "@typescript-eslint/explicit-module-boundary-types": "error", - "@typescript-eslint/no-explicit-any": "error", - "@typescript-eslint/switch-exhaustiveness-check": [ - "error", - { - allowDefaultCaseForExhaustiveSwitch: false, - requireDefaultForNonUnion: true, - }, - ], - - // Use typescript-eslint/no-unused-vars, no-unused-vars reports - // false positives with typescript - "no-unused-vars": "off", - "@typescript-eslint/no-unused-vars": [ - "error", - { - argsIgnorePattern: "^_", - varsIgnorePattern: "^_", - caughtErrorsIgnorePattern: "^_", - }, - ], - - curly: "error", - - eqeqeq: ["error", "always", { null: "never" }], - "react-refresh/only-export-components": [ - "error", - { allowConstantExport: true }, - ], - "no-await-in-loop": "error", - "no-bitwise": "error", - "no-caller": "error", - // This is fine during development, but should not be checked in. - "no-console": "error", - // This is fine during development, but should not be checked in. - "no-debugger": "error", - "no-duplicate-case": "error", - "no-eval": "error", - "no-ex-assign": "error", - "no-return-await": "error", - "no-param-reassign": "error", - "no-script-url": "error", - "no-self-compare": "error", - "no-unsafe-finally": "error", - "no-var": "error", - "react-hooks/rules-of-hooks": "error", - "react-hooks/exhaustive-deps": "error", - }, - overrides: [ - { - // apply only to files under tests/ - files: ["tests/**/*.{ts,tsx,js,jsx}"], - rules: { - "@typescript-eslint/no-explicit-any": "off", - "import/order": "off", - "@typescript-eslint/explicit-module-boundary-types": "off", - "@typescript-eslint/ban-ts-comment": "off", - "@typescript-eslint/no-var-requires": "off", - "no-await-in-loop": "off", - "no-control-regex": "off", - }, - }, - ], -}; diff --git a/codex-cli/HUSKY.md b/codex-cli/HUSKY.md deleted file mode 100644 index d525e2f7..00000000 --- a/codex-cli/HUSKY.md +++ /dev/null @@ -1,45 +0,0 @@ -# Husky Git Hooks - -This project uses [Husky](https://typicode.github.io/husky/) to enforce code quality checks before commits and pushes. - -## What's Included - -- **Pre-commit Hook**: Runs lint-staged to check files that are about to be committed. - - - Lints and formats TypeScript/TSX files using ESLint and Prettier - - Formats JSON, MD, and YML files using Prettier - -- **Pre-push Hook**: Runs tests and type checking before pushing to the remote repository. - - Executes `npm test` to run all tests - - Executes `npm run typecheck` to check TypeScript types - -## Benefits - -- Ensures consistent code style across the project -- Prevents pushing code with failing tests or type errors -- Reduces the need for style-related code review comments -- Improves overall code quality - -## For Contributors - -You don't need to do anything special to use these hooks. They will automatically run when you commit or push code. - -If you need to bypass the hooks in exceptional cases: - -```bash -# Skip pre-commit hooks -git commit -m "Your message" --no-verify - -# Skip pre-push hooks -git push --no-verify -``` - -Note: Please use these bypass options sparingly and only when absolutely necessary. - -## Troubleshooting - -If you encounter any issues with the hooks: - -1. Make sure you have the latest dependencies installed: `npm install` -2. Ensure the hook scripts are executable (Unix systems): `chmod +x .husky/pre-commit .husky/pre-push` -3. Check if there are any ESLint or Prettier configuration issues in your code diff --git a/codex-cli/build.mjs b/codex-cli/build.mjs deleted file mode 100644 index 16664d76..00000000 --- a/codex-cli/build.mjs +++ /dev/null @@ -1,88 +0,0 @@ -import * as esbuild from "esbuild"; -import * as fs from "fs"; -import * as path from "path"; - -const OUT_DIR = 'dist' -/** - * ink attempts to import react-devtools-core in an ESM-unfriendly way: - * - * https://github.com/vadimdemedes/ink/blob/eab6ef07d4030606530d58d3d7be8079b4fb93bb/src/reconciler.ts#L22-L45 - * - * to make this work, we have to strip the import out of the build. - */ -const ignoreReactDevToolsPlugin = { - name: "ignore-react-devtools", - setup(build) { - // When an import for 'react-devtools-core' is encountered, - // return an empty module. - build.onResolve({ filter: /^react-devtools-core$/ }, (args) => { - return { path: args.path, namespace: "ignore-devtools" }; - }); - build.onLoad({ filter: /.*/, namespace: "ignore-devtools" }, () => { - return { contents: "", loader: "js" }; - }); - }, -}; - -// ---------------------------------------------------------------------------- -// Build mode detection (production vs development) -// -// • production (default): minified, external telemetry shebang handling. -// • development (--dev|NODE_ENV=development|CODEX_DEV=1): -// – no minification -// – inline source maps for better stacktraces -// – shebang tweaked to enable Node's source‑map support at runtime -// ---------------------------------------------------------------------------- - -const isDevBuild = - process.argv.includes("--dev") || - process.env.CODEX_DEV === "1" || - process.env.NODE_ENV === "development"; - -const plugins = [ignoreReactDevToolsPlugin]; - -// Build Hygiene, ensure we drop previous dist dir and any leftover files -const outPath = path.resolve(OUT_DIR); -if (fs.existsSync(outPath)) { - fs.rmSync(outPath, { recursive: true, force: true }); -} - -// Add a shebang that enables source‑map support for dev builds so that stack -// traces point to the original TypeScript lines without requiring callers to -// remember to set NODE_OPTIONS manually. -if (isDevBuild) { - const devShebangLine = - "#!/usr/bin/env -S NODE_OPTIONS=--enable-source-maps node\n"; - const devShebangPlugin = { - name: "dev-shebang", - setup(build) { - build.onEnd(async () => { - const outFile = path.resolve(isDevBuild ? `${OUT_DIR}/cli-dev.js` : `${OUT_DIR}/cli.js`); - let code = await fs.promises.readFile(outFile, "utf8"); - if (code.startsWith("#!")) { - code = code.replace(/^#!.*\n/, devShebangLine); - await fs.promises.writeFile(outFile, code, "utf8"); - } - }); - }, - }; - plugins.push(devShebangPlugin); -} - -esbuild - .build({ - entryPoints: ["src/cli.tsx"], - // Do not bundle the contents of package.json at build time: always read it - // at runtime. - external: ["../package.json"], - bundle: true, - format: "esm", - platform: "node", - tsconfig: "tsconfig.json", - outfile: isDevBuild ? `${OUT_DIR}/cli-dev.js` : `${OUT_DIR}/cli.js`, - minify: !isDevBuild, - sourcemap: isDevBuild ? "inline" : true, - plugins, - inject: ["./require-shim.js"], - }) - .catch(() => process.exit(1)); diff --git a/codex-cli/default.nix b/codex-cli/default.nix deleted file mode 100644 index 6ae19bb7..00000000 --- a/codex-cli/default.nix +++ /dev/null @@ -1,43 +0,0 @@ -{ pkgs, monorep-deps ? [], ... }: -let - node = pkgs.nodejs_22; -in -rec { - package = pkgs.buildNpmPackage { - pname = "codex-cli"; - version = "0.1.0"; - src = ./.; - npmDepsHash = "sha256-3tAalmh50I0fhhd7XreM+jvl0n4zcRhqygFNB1Olst8"; - nodejs = node; - npmInstallFlags = [ "--frozen-lockfile" ]; - meta = with pkgs.lib; { - description = "OpenAI Codex command‑line interface"; - license = licenses.asl20; - homepage = "https://github.com/openai/codex"; - }; - }; - devShell = pkgs.mkShell { - name = "codex-cli-dev"; - buildInputs = monorep-deps ++ [ - node - pkgs.pnpm - ]; - shellHook = '' - echo "Entering development shell for codex-cli" - # cd codex-cli - if [ -f package-lock.json ]; then - pnpm ci || echo "npm ci failed" - else - pnpm install || echo "npm install failed" - fi - npm run build || echo "npm build failed" - export PATH=$PWD/node_modules/.bin:$PATH - alias codex="node $PWD/dist/cli.js" - ''; - }; - app = { - type = "app"; - program = "${package}/bin/codex"; - }; -} - diff --git a/codex-cli/examples/README.md b/codex-cli/examples/README.md deleted file mode 100644 index 0ad83f3a..00000000 --- a/codex-cli/examples/README.md +++ /dev/null @@ -1,44 +0,0 @@ -# Quick start examples - -This directory bundles some self‑contained examples using the Codex CLI. If you have never used the Codex CLI before, and want to see it complete a sample task, start with running **camerascii**. You'll see your webcam feed turned into animated ASCII art in a few minutes. - -If you want to get started using the Codex CLI directly, skip this and refer to the prompting guide. - -## Structure - -Each example contains the following: -``` -example‑name/ -├── run.sh # helper script that launches a new Codex session for the task -├── task.yaml # task spec containing a prompt passed to Codex -├── template/ # (optional) starter files copied into each run -└── runs/ # work directories created by run.sh -``` - -**run.sh**: a convenience wrapper that does three things: -- Creates `runs/run_N`, where *N* is the number of a run. -- Copies the contents of `template/` into that folder (if present). -- Launches the Codex CLI with the description from `task.yaml`. - -**template/**: any existing files or markdown instructions you would like Codex to see before it starts working. - -**runs/**: the directories produced by `run.sh`. - -## Running an example - -1. **Run the helper script**: -``` -cd camerascii -./run.sh -``` -2. **Interact with the Codex CLI**: the CLI will open with the prompt: “*Take a look at the screenshot details and implement a webpage that uses a webcam to style the video feed accordingly…*” Confirm the commands Codex CLI requests to generate `index.html`. - -3. **Check its work**: when Codex is done, open ``runs/run_1/index.html`` in a browser. Your webcam feed should now be rendered as a cascade of ASCII glyphs. If the outcome isn't what you expect, try running it again, or adjust the task prompt. - - -## Other examples -Besides **camerascii**, you can experiment with: - -- **build‑codex‑demo**: recreate the original 2021 Codex YouTube demo. -- **impossible‑pong**: where Codex creates more difficult levels. -- **prompt‑analyzer**: make a data science app for clustering [prompts](https://github.com/f/awesome-chatgpt-prompts). diff --git a/codex-cli/examples/build-codex-demo/run.sh b/codex-cli/examples/build-codex-demo/run.sh deleted file mode 100755 index 5f26b191..00000000 --- a/codex-cli/examples/build-codex-demo/run.sh +++ /dev/null @@ -1,65 +0,0 @@ -#!/bin/bash - -# run.sh — Create a new run_N directory for a Codex task, optionally bootstrapped from a template, -# then launch Codex with the task description from task.yaml. -# -# Usage: -# ./run.sh # Prompts to confirm new run -# ./run.sh --auto-confirm # Skips confirmation -# -# Assumes: -# - yq and jq are installed -# - ../task.yaml exists (with .name and .description fields) -# - ../template/ exists (optional, for bootstrapping new runs) - -# Enable auto-confirm mode if flag is passed -auto_mode=false -[[ "$1" == "--auto-confirm" ]] && auto_mode=true - -# Move into the working directory -cd runs || exit 1 - -# Grab task name for logging -task_name=$(yq -o=json '.' ../task.yaml | jq -r '.name') -echo "Checking for runs for task: $task_name" - -# Find existing run_N directories -shopt -s nullglob -run_dirs=(run_[0-9]*) -shopt -u nullglob - -if [ ${#run_dirs[@]} -eq 0 ]; then - echo "There are 0 runs." - new_run_number=1 -else - max_run_number=0 - for d in "${run_dirs[@]}"; do - [[ "$d" =~ ^run_([0-9]+)$ ]] && (( ${BASH_REMATCH[1]} > max_run_number )) && max_run_number=${BASH_REMATCH[1]} - done - new_run_number=$((max_run_number + 1)) - echo "There are $max_run_number runs." -fi - -# Confirm creation unless in auto mode -if [ "$auto_mode" = false ]; then - read -p "Create run_$new_run_number? (Y/N): " choice - [[ "$choice" != [Yy] ]] && echo "Exiting." && exit 1 -fi - -# Create the run directory -mkdir "run_$new_run_number" - -# Check if the template directory exists and copy its contents -if [ -d "../template" ]; then - cp -r ../template/* "run_$new_run_number" - echo "Initialized run_$new_run_number from template/" -else - echo "Template directory does not exist. Skipping initialization from template." -fi - -cd "run_$new_run_number" - -# Launch Codex -echo "Launching..." -description=$(yq -o=json '.' ../../task.yaml | jq -r '.description') -codex "$description" diff --git a/codex-cli/examples/build-codex-demo/runs/.gitkeep b/codex-cli/examples/build-codex-demo/runs/.gitkeep deleted file mode 100644 index e69de29b..00000000 diff --git a/codex-cli/examples/build-codex-demo/task.yaml b/codex-cli/examples/build-codex-demo/task.yaml deleted file mode 100644 index d18cc91f..00000000 --- a/codex-cli/examples/build-codex-demo/task.yaml +++ /dev/null @@ -1,88 +0,0 @@ -name: "build-codex-demo" -description: | - I want you to reimplement the original OpenAI Codex demo. - - Functionality: - - User types a prompt and hits enter to send - - The prompt is added to the conversation history - - The backend calls the OpenAI API with stream: true - - Tokens are streamed back and appended to the code viewer - - Syntax highlighting updates in real time - - When a full HTML file is received, it is rendered in a sandboxed iframe - - The iframe replaces the previous preview with the new HTML after the stream is complete (i.e. keep the old preview until a new stream is complete) - - Append each assistant and user message to preserve context across turns - - Errors are displayed to user gracefully - - Ensure there is a fixed layout is responsive and faithful to the screenshot design - - Be sure to parse the output from OpenAI call to strip the ```html tags code is returned within - - Use the system prompt shared in the API call below to ensure the AI only returns HTML - - Support a simple local backend that can: - - Read local env for OPENAI_API_KEY - - Expose an endpoint that streams completions from OpenAI - - Backend should be a simple node.js app - - App should be easy to run locally for development and testing - - Minimal setup preferred — keep dependencies light unless justified - - Description of layout and design: - - Two stacked panels, vertically aligned: - - Top Panel: Main interactive area with two main parts - - Left Side: Visual output canvas. Mostly blank space with a small image preview in the upper-left - - Right Side: Code display area - - Light background with code shown in a monospace font - - Comments in green; code aligns vertically like an IDE/snippet view - - Bottom Panel: Prompt/command bar - - A single-line text box with a placeholder prompt - - A green arrow (submit button) on the right side - - Scrolling should only be supported in the code editor and output canvas - - Visual style - - Minimalist UI, light and clean - - Neutral white/gray background - - Subtle shadow or border around both panels, giving them card-like elevation - - Code section is color-coded, likely for syntax highlighting - - Interactive feel with the text input styled like a chat/message interface - - Here's the latest OpenAI API and prompt to use: - ``` - import OpenAI from "openai"; - - const openai = new OpenAI({ - apiKey: process.env.OPENAI_API_KEY, - }); - - const response = await openai.responses.create({ - model: "gpt-4.1", - input: [ - { - "role": "system", - "content": [ - { - "type": "input_text", - "text": "You are a coding agent that specializes in frontend code. Whenever you are prompted, return only the full HTML file." - } - ] - } - ], - text: { - "format": { - "type": "text" - } - }, - reasoning: {}, - tools: [], - temperature: 1, - top_p: 1 - }); - - console.log(response.output_text); - ``` - Additional things to note: - - Strip any html and tags from the OpenAI response before rendering - - Assume the OpenAI API model response always wraps HTML in markdown-style triple backticks like ```html ``` - - The display code window should have syntax highlighting and line numbers. - - Make sure to only display the code, not the backticks or ```html that wrap the code from the model. - - Do not inject raw markdown; only parse and insert pure HTML into the iframe - - Only the code viewer and output panel should scroll - - Keep the previous preview visible until the full new HTML has streamed in - - Add a README.md with what you've implemented and how to run it. diff --git a/codex-cli/examples/camerascii/run.sh b/codex-cli/examples/camerascii/run.sh deleted file mode 100755 index a6bcfb03..00000000 --- a/codex-cli/examples/camerascii/run.sh +++ /dev/null @@ -1,68 +0,0 @@ -#!/bin/bash - -# run.sh — Create a new run_N directory for a Codex task, optionally bootstrapped from a template, -# then launch Codex with the task description from task.yaml. -# -# Usage: -# ./run.sh # Prompts to confirm new run -# ./run.sh --auto-confirm # Skips confirmation -# -# Assumes: -# - yq and jq are installed -# - ../task.yaml exists (with .name and .description fields) -# - ../template/ exists (optional, for bootstrapping new runs) - -# Enable auto-confirm mode if flag is passed -auto_mode=false -[[ "$1" == "--auto-confirm" ]] && auto_mode=true - -# Create the runs directory if it doesn't exist -mkdir -p runs - -# Move into the working directory -cd runs || exit 1 - -# Grab task name for logging -task_name=$(yq -o=json '.' ../task.yaml | jq -r '.name') -echo "Checking for runs for task: $task_name" - -# Find existing run_N directories -shopt -s nullglob -run_dirs=(run_[0-9]*) -shopt -u nullglob - -if [ ${#run_dirs[@]} -eq 0 ]; then - echo "There are 0 runs." - new_run_number=1 -else - max_run_number=0 - for d in "${run_dirs[@]}"; do - [[ "$d" =~ ^run_([0-9]+)$ ]] && (( ${BASH_REMATCH[1]} > max_run_number )) && max_run_number=${BASH_REMATCH[1]} - done - new_run_number=$((max_run_number + 1)) - echo "There are $max_run_number runs." -fi - -# Confirm creation unless in auto mode -if [ "$auto_mode" = false ]; then - read -p "Create run_$new_run_number? (Y/N): " choice - [[ "$choice" != [Yy] ]] && echo "Exiting." && exit 1 -fi - -# Create the run directory -mkdir "run_$new_run_number" - -# Check if the template directory exists and copy its contents -if [ -d "../template" ]; then - cp -r ../template/* "run_$new_run_number" - echo "Initialized run_$new_run_number from template/" -else - echo "Template directory does not exist. Skipping initialization from template." -fi - -cd "run_$new_run_number" - -# Launch Codex -echo "Launching..." -description=$(yq -o=json '.' ../../task.yaml | jq -r '.description') -codex "$description" diff --git a/codex-cli/examples/camerascii/runs/.gitkeep b/codex-cli/examples/camerascii/runs/.gitkeep deleted file mode 100644 index e69de29b..00000000 diff --git a/codex-cli/examples/camerascii/task.yaml b/codex-cli/examples/camerascii/task.yaml deleted file mode 100644 index 9c5efedc..00000000 --- a/codex-cli/examples/camerascii/task.yaml +++ /dev/null @@ -1,5 +0,0 @@ -name: "camerascii" -description: | - Take a look at the screenshot details and implement a webpage that uses webcam - to style the video feed accordingly (i.e. as ASCII art). Add some of the relevant features - from the screenshot to the webpage in index.html. diff --git a/codex-cli/examples/camerascii/template/screenshot_details.md b/codex-cli/examples/camerascii/template/screenshot_details.md deleted file mode 100644 index 08e41f4d..00000000 --- a/codex-cli/examples/camerascii/template/screenshot_details.md +++ /dev/null @@ -1,34 +0,0 @@ -### Screenshot Description - -The image is a full–page screenshot of a single post on the social‑media site X (formerly Twitter). - -1. **Header row** - * At the very top‑left is a small circular avatar. The photo shows the side profile of a person whose face is softly lit in bluish‑purple tones; only the head and part of the neck are visible. - * In the far upper‑right corner sit two standard X / Twitter interface icons: a circle containing a diagonal line (the “Mute / Block” indicator) and a three‑dot overflow menu. - -2. **Tweet body text** - * Below the header, in regular type, the author writes: - - “Okay, OpenAI’s o3 is insane. Spent an hour messing with it and built an image‑to‑ASCII art converter, the exact tool I’ve always wanted. And it works so well” - -3. **Embedded media** - * The majority of the screenshot is occupied by an embedded 12‑second video of the converter UI. The video window has rounded corners and a dark theme. - * **Left panel (tool controls)** – a slim vertical sidebar with the following labeled sections and blue–accented UI controls: - * Theme selector (“Dark” is chosen). - * A small checkbox labeled “Ignore White”. - * **Upload Image** button area that shows the chosen file name. - * **Image Processing** sliders: - * “ASCII Width” (value ≈ 143) - * “Brightness” (‑65) - * “Contrast” (58) - * “Blur (px)” (0.5) - * A square checkbox for “Invert Colors”. - * **Dithering** subsection with a checkbox (“Enable Dithering”) and a dropdown for the algorithm (value: “Noise”). - * **Character Set** dropdown (value: “Detailed (Default)”). - * **Display** slider labeled “Zoom (%)” (value ≈ 170) and a “Reset” button. - - * **Main preview area (right side)** – a dark gray canvas that renders the selected image as white ASCII characters. The preview clearly depicts a stylized **palm tree**: a skinny trunk rises from the bottom centre, and a crown of splayed fronds fills the upper right quadrant. - * A small black badge showing **“0:12”** overlays the bottom‑left corner of the media frame, indicating the video’s duration. - * In the top‑right area of the media window are two pill‑shaped buttons: a heart‑shaped “Save” button and a cog‑shaped “Settings” button. - -Overall, the screenshot shows the user excitedly announcing the success of their custom “Image to ASCII” converter created with OpenAI’s “o3”, accompanied by a short video demonstration of the tool converting a palm‑tree photo into ASCII art. diff --git a/codex-cli/examples/impossible-pong/run.sh b/codex-cli/examples/impossible-pong/run.sh deleted file mode 100755 index a6bcfb03..00000000 --- a/codex-cli/examples/impossible-pong/run.sh +++ /dev/null @@ -1,68 +0,0 @@ -#!/bin/bash - -# run.sh — Create a new run_N directory for a Codex task, optionally bootstrapped from a template, -# then launch Codex with the task description from task.yaml. -# -# Usage: -# ./run.sh # Prompts to confirm new run -# ./run.sh --auto-confirm # Skips confirmation -# -# Assumes: -# - yq and jq are installed -# - ../task.yaml exists (with .name and .description fields) -# - ../template/ exists (optional, for bootstrapping new runs) - -# Enable auto-confirm mode if flag is passed -auto_mode=false -[[ "$1" == "--auto-confirm" ]] && auto_mode=true - -# Create the runs directory if it doesn't exist -mkdir -p runs - -# Move into the working directory -cd runs || exit 1 - -# Grab task name for logging -task_name=$(yq -o=json '.' ../task.yaml | jq -r '.name') -echo "Checking for runs for task: $task_name" - -# Find existing run_N directories -shopt -s nullglob -run_dirs=(run_[0-9]*) -shopt -u nullglob - -if [ ${#run_dirs[@]} -eq 0 ]; then - echo "There are 0 runs." - new_run_number=1 -else - max_run_number=0 - for d in "${run_dirs[@]}"; do - [[ "$d" =~ ^run_([0-9]+)$ ]] && (( ${BASH_REMATCH[1]} > max_run_number )) && max_run_number=${BASH_REMATCH[1]} - done - new_run_number=$((max_run_number + 1)) - echo "There are $max_run_number runs." -fi - -# Confirm creation unless in auto mode -if [ "$auto_mode" = false ]; then - read -p "Create run_$new_run_number? (Y/N): " choice - [[ "$choice" != [Yy] ]] && echo "Exiting." && exit 1 -fi - -# Create the run directory -mkdir "run_$new_run_number" - -# Check if the template directory exists and copy its contents -if [ -d "../template" ]; then - cp -r ../template/* "run_$new_run_number" - echo "Initialized run_$new_run_number from template/" -else - echo "Template directory does not exist. Skipping initialization from template." -fi - -cd "run_$new_run_number" - -# Launch Codex -echo "Launching..." -description=$(yq -o=json '.' ../../task.yaml | jq -r '.description') -codex "$description" diff --git a/codex-cli/examples/impossible-pong/runs/.gitkeep b/codex-cli/examples/impossible-pong/runs/.gitkeep deleted file mode 100644 index e69de29b..00000000 diff --git a/codex-cli/examples/impossible-pong/task.yaml b/codex-cli/examples/impossible-pong/task.yaml deleted file mode 100644 index 8d8acbbf..00000000 --- a/codex-cli/examples/impossible-pong/task.yaml +++ /dev/null @@ -1,11 +0,0 @@ -name: "impossible-pong" -description: | - Update index.html with the following features: - - Add an overlaid styled popup to start the game on first load - - Between each point, show a 3 second countdown (this should be skipped if a player wins) - - After each game the AI wins, display text at the bottom of the screen with lighthearted insults for the player - - Add a leaderboard to the right of the court that shows how many games each player has won. - - When a player wins, a styled popup appears with the winner's name and the option to play again. The leaderboard should update. - - Add an "even more insane" difficulty mode that adds spin to the ball that makes it harder to predict. - - Add an "even more(!!) insane" difficulty mode where the ball does a spin mid court and then picks a random (reasonable) direction to go in (this should only advantage the AI player) - - Let the user choose which difficulty mode they want to play in on the popup that appears when the game starts. diff --git a/codex-cli/examples/impossible-pong/template/index.html b/codex-cli/examples/impossible-pong/template/index.html deleted file mode 100644 index 90a9e9e5..00000000 --- a/codex-cli/examples/impossible-pong/template/index.html +++ /dev/null @@ -1,233 +0,0 @@ - - - - - Pong - - - - -
- - - - -
Player: 0 | AI: 0
-
- - - - - - diff --git a/codex-cli/examples/prompt-analyzer/run.sh b/codex-cli/examples/prompt-analyzer/run.sh deleted file mode 100755 index a6bcfb03..00000000 --- a/codex-cli/examples/prompt-analyzer/run.sh +++ /dev/null @@ -1,68 +0,0 @@ -#!/bin/bash - -# run.sh — Create a new run_N directory for a Codex task, optionally bootstrapped from a template, -# then launch Codex with the task description from task.yaml. -# -# Usage: -# ./run.sh # Prompts to confirm new run -# ./run.sh --auto-confirm # Skips confirmation -# -# Assumes: -# - yq and jq are installed -# - ../task.yaml exists (with .name and .description fields) -# - ../template/ exists (optional, for bootstrapping new runs) - -# Enable auto-confirm mode if flag is passed -auto_mode=false -[[ "$1" == "--auto-confirm" ]] && auto_mode=true - -# Create the runs directory if it doesn't exist -mkdir -p runs - -# Move into the working directory -cd runs || exit 1 - -# Grab task name for logging -task_name=$(yq -o=json '.' ../task.yaml | jq -r '.name') -echo "Checking for runs for task: $task_name" - -# Find existing run_N directories -shopt -s nullglob -run_dirs=(run_[0-9]*) -shopt -u nullglob - -if [ ${#run_dirs[@]} -eq 0 ]; then - echo "There are 0 runs." - new_run_number=1 -else - max_run_number=0 - for d in "${run_dirs[@]}"; do - [[ "$d" =~ ^run_([0-9]+)$ ]] && (( ${BASH_REMATCH[1]} > max_run_number )) && max_run_number=${BASH_REMATCH[1]} - done - new_run_number=$((max_run_number + 1)) - echo "There are $max_run_number runs." -fi - -# Confirm creation unless in auto mode -if [ "$auto_mode" = false ]; then - read -p "Create run_$new_run_number? (Y/N): " choice - [[ "$choice" != [Yy] ]] && echo "Exiting." && exit 1 -fi - -# Create the run directory -mkdir "run_$new_run_number" - -# Check if the template directory exists and copy its contents -if [ -d "../template" ]; then - cp -r ../template/* "run_$new_run_number" - echo "Initialized run_$new_run_number from template/" -else - echo "Template directory does not exist. Skipping initialization from template." -fi - -cd "run_$new_run_number" - -# Launch Codex -echo "Launching..." -description=$(yq -o=json '.' ../../task.yaml | jq -r '.description') -codex "$description" diff --git a/codex-cli/examples/prompt-analyzer/runs/.gitkeep b/codex-cli/examples/prompt-analyzer/runs/.gitkeep deleted file mode 100644 index e69de29b..00000000 diff --git a/codex-cli/examples/prompt-analyzer/task.yaml b/codex-cli/examples/prompt-analyzer/task.yaml deleted file mode 100644 index d0da4eab..00000000 --- a/codex-cli/examples/prompt-analyzer/task.yaml +++ /dev/null @@ -1,17 +0,0 @@ -name: "prompt-analyzer" -description: | - I have some existing work here (embedding prompts, clustering them, generating - summaries with GPT). I want to make it more interactive and reusable. - - Objective: create an interactive cluster explorer - - Build a lightweight streamlit app UI - - Allow users to upload a CSV of prompts - - Display clustered prompts with auto-generated cluster names and summaries - - Click "cluster" and see progress stream in a small window (primarily for aesthetic reasons) - - Let users browse examples by cluster, view outliers, and inspect individual prompts - - See generated analysis rendered in the app, along with the plots displayed nicely - - Support selecting clustering algorithms (e.g. DBSCAN, KMeans, etc) and "recluster" - - Include token count + histogram of prompt lengths - - Add interactive filters in UI (e.g. filter by token length, keyword, or cluster) - - When you're done, update the README.md with a changelog and instructions for how to run the app. diff --git a/codex-cli/examples/prompt-analyzer/template/Clustering.ipynb b/codex-cli/examples/prompt-analyzer/template/Clustering.ipynb deleted file mode 100644 index 4b97aa50..00000000 --- a/codex-cli/examples/prompt-analyzer/template/Clustering.ipynb +++ /dev/null @@ -1,231 +0,0 @@ -{ - "cells": [ - { - "attachments": {}, - "cell_type": "markdown", - "metadata": {}, - "source": [ - "## K-means Clustering in Python using OpenAI\n", - "\n", - "We use a simple k-means algorithm to demonstrate how clustering can be done. Clustering can help discover valuable, hidden groupings within the data. The dataset is created in the [Get_embeddings_from_dataset Notebook](Get_embeddings_from_dataset.ipynb)." - ] - }, - { - "cell_type": "code", - "execution_count": 2, - "metadata": {}, - "outputs": [ - { - "data": { - "text/plain": [ - "(1000, 1536)" - ] - }, - "execution_count": 2, - "metadata": {}, - "output_type": "execute_result" - } - ], - "source": [ - "# imports\n", - "import numpy as np\n", - "import pandas as pd\n", - "from ast import literal_eval\n", - "\n", - "# load data\n", - "datafile_path = \"./data/fine_food_reviews_with_embeddings_1k.csv\"\n", - "\n", - "df = pd.read_csv(datafile_path)\n", - "df[\"embedding\"] = df.embedding.apply(literal_eval).apply(np.array) # convert string to numpy array\n", - "matrix = np.vstack(df.embedding.values)\n", - "matrix.shape\n" - ] - }, - { - "attachments": {}, - "cell_type": "markdown", - "metadata": {}, - "source": [ - "### 1. Find the clusters using K-means" - ] - }, - { - "attachments": {}, - "cell_type": "markdown", - "metadata": {}, - "source": [ - "We show the simplest use of K-means. You can pick the number of clusters that fits your use case best." - ] - }, - { - "cell_type": "code", - "execution_count": 3, - "metadata": {}, - "outputs": [ - { - "name": "stderr", - "output_type": "stream", - "text": [ - "/opt/homebrew/lib/python3.11/site-packages/sklearn/cluster/_kmeans.py:870: FutureWarning: The default value of `n_init` will change from 10 to 'auto' in 1.4. Set the value of `n_init` explicitly to suppress the warning\n", - " warnings.warn(\n" - ] - }, - { - "data": { - "text/plain": [ - "Cluster\n", - "0 4.105691\n", - "1 4.191176\n", - "2 4.215613\n", - "3 4.306590\n", - "Name: Score, dtype: float64" - ] - }, - "execution_count": 3, - "metadata": {}, - "output_type": "execute_result" - } - ], - "source": [ - "from sklearn.cluster import KMeans\n", - "\n", - "n_clusters = 4\n", - "\n", - "kmeans = KMeans(n_clusters=n_clusters, init=\"k-means++\", random_state=42)\n", - "kmeans.fit(matrix)\n", - "labels = kmeans.labels_\n", - "df[\"Cluster\"] = labels\n", - "\n", - "df.groupby(\"Cluster\").Score.mean().sort_values()\n" - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": {}, - "outputs": [], - "source": [ - "from sklearn.manifold import TSNE\n", - "import matplotlib\n", - "import matplotlib.pyplot as plt\n", - "\n", - "tsne = TSNE(n_components=2, perplexity=15, random_state=42, init=\"random\", learning_rate=200)\n", - "vis_dims2 = tsne.fit_transform(matrix)\n", - "\n", - "x = [x for x, y in vis_dims2]\n", - "y = [y for x, y in vis_dims2]\n", - "\n", - "for category, color in enumerate([\"purple\", \"green\", \"red\", \"blue\"]):\n", - " xs = np.array(x)[df.Cluster == category]\n", - " ys = np.array(y)[df.Cluster == category]\n", - " plt.scatter(xs, ys, color=color, alpha=0.3)\n", - "\n", - " avg_x = xs.mean()\n", - " avg_y = ys.mean()\n", - "\n", - " plt.scatter(avg_x, avg_y, marker=\"x\", color=color, s=100)\n", - "plt.title(\"Clusters identified visualized in language 2d using t-SNE\")\n" - ] - }, - { - "attachments": {}, - "cell_type": "markdown", - "metadata": {}, - "source": [ - "Visualization of clusters in a 2d projection. In this run, the green cluster (#1) seems quite different from the others. Let's see a few samples from each cluster." - ] - }, - { - "attachments": {}, - "cell_type": "markdown", - "metadata": {}, - "source": [ - "### 2. Text samples in the clusters & naming the clusters\n", - "\n", - "Let's show random samples from each cluster. We'll use gpt-4 to name the clusters, based on a random sample of 5 reviews from that cluster." - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": {}, - "outputs": [], - "source": [ - "from openai import OpenAI\n", - "import os\n", - "\n", - "client = OpenAI(api_key=os.environ.get(\"OPENAI_API_KEY\", \"\"))\n", - "\n", - "# Reading a review which belong to each group.\n", - "rev_per_cluster = 5\n", - "\n", - "for i in range(n_clusters):\n", - " print(f\"Cluster {i} Theme:\", end=\" \")\n", - "\n", - " reviews = \"\\n\".join(\n", - " df[df.Cluster == i]\n", - " .combined.str.replace(\"Title: \", \"\")\n", - " .str.replace(\"\\n\\nContent: \", \": \")\n", - " .sample(rev_per_cluster, random_state=42)\n", - " .values\n", - " )\n", - "\n", - " messages = [\n", - " {\"role\": \"user\", \"content\": f'What do the following customer reviews have in common?\\n\\nCustomer reviews:\\n\"\"\"\\n{reviews}\\n\"\"\"\\n\\nTheme:'}\n", - " ]\n", - "\n", - " response = client.chat.completions.create(\n", - " model=\"gpt-4\",\n", - " messages=messages,\n", - " temperature=0,\n", - " max_tokens=64,\n", - " top_p=1,\n", - " frequency_penalty=0,\n", - " presence_penalty=0)\n", - " print(response.choices[0].message.content.replace(\"\\n\", \"\"))\n", - "\n", - " sample_cluster_rows = df[df.Cluster == i].sample(rev_per_cluster, random_state=42)\n", - " for j in range(rev_per_cluster):\n", - " print(sample_cluster_rows.Score.values[j], end=\", \")\n", - " print(sample_cluster_rows.Summary.values[j], end=\": \")\n", - " print(sample_cluster_rows.Text.str[:70].values[j])\n", - "\n", - " print(\"-\" * 100)\n" - ] - }, - { - "attachments": {}, - "cell_type": "markdown", - "metadata": {}, - "source": [ - "It's important to note that clusters will not necessarily match what you intend to use them for. A larger amount of clusters will focus on more specific patterns, whereas a small number of clusters will usually focus on largest discrepancies in the data." - ] - } - ], - "metadata": { - "kernelspec": { - "display_name": "openai", - "language": "python", - "name": "python3" - }, - "language_info": { - "codemirror_mode": { - "name": "ipython", - "version": 3 - }, - "file_extension": ".py", - "mimetype": "text/x-python", - "name": "python", - "nbconvert_exporter": "python", - "pygments_lexer": "ipython3", - "version": "3.11.3" - }, - "vscode": { - "interpreter": { - "hash": "365536dcbde60510dc9073d6b991cd35db2d9bac356a11f5b64279a5e6708b97" - } - } - }, - "nbformat": 4, - "nbformat_minor": 2 -} diff --git a/codex-cli/examples/prompt-analyzer/template/README.md b/codex-cli/examples/prompt-analyzer/template/README.md deleted file mode 100644 index 0f7b18c8..00000000 --- a/codex-cli/examples/prompt-analyzer/template/README.md +++ /dev/null @@ -1,103 +0,0 @@ -# Prompt‑Clustering Utility - -This repository contains a small utility (`cluster_prompts.py`) that embeds a -list of prompts with the OpenAI Embedding API, discovers natural groupings with -unsupervised clustering, lets ChatGPT name & describe each cluster and finally -produces a concise Markdown report plus a couple of diagnostic plots. - -The default input file (`prompts.csv`) ships with the repo so you can try the -script immediately, but you can of course point it at your own file. - ---- - -## 1. Setup - -1. Install the Python dependencies (preferably inside a virtual env): - -```bash -pip install pandas numpy scikit-learn matplotlib openai -``` - -2. Export your OpenAI API key (**required**): - -```bash -export OPENAI_API_KEY="sk‑..." -``` - ---- - -## 2. Basic usage - -```bash -# Minimal command – runs on prompts.csv and writes analysis.md + plots/ -python cluster_prompts.py -``` - -This will - -* create embeddings with the `text-embedding-3-small` model,  -* pick a suitable number *k* via silhouette score (K‑Means), -* ask `gpt‑4o‑mini` to label & describe each cluster, -* store the results in `analysis.md`, -* and save two plots to `plots/` (`cluster_sizes.png` and `tsne.png`). - -The script prints a short success message once done. - ---- - -## 3. Command‑line options - -| flag | default | description | -|------|---------|-------------| -| `--csv` | `prompts.csv` | path to the input CSV (must contain a `prompt` column; an `act` column is used as context if present) | -| `--cache` | _(none)_ | embed­ding cache path (JSON). Speeds up repeated runs – new texts are appended automatically. | -| `--cluster-method` | `kmeans` | `kmeans` (with automatic *k*) or `dbscan` | -| `--k-max` | `10` | upper bound for *k* when `kmeans` is selected | -| `--dbscan-min-samples` | `3` | min samples parameter for DBSCAN | -| `--embedding-model` | `text-embedding-3-small` | any OpenAI embedding model | -| `--chat-model` | `gpt-4o-mini` | chat model used to generate cluster names / descriptions | -| `--output-md` | `analysis.md` | where to write the Markdown report | -| `--plots-dir` | `plots` | directory for generated PNGs | - -Example with customised options: - -```bash -python cluster_prompts.py \ - --csv my_prompts.csv \ - --cache .cache/embeddings.json \ - --cluster-method dbscan \ - --embedding-model text-embedding-3-large \ - --chat-model gpt-4o \ - --output-md my_analysis.md \ - --plots-dir my_plots -``` - ---- - -## 4. Interpreting the output - -### analysis.md - -* Overview table: cluster label, generated name, member count and description. -* Detailed section for every cluster with five representative example prompts. -* Separate lists for - * **Noise / outliers** (label `‑1` when DBSCAN is used) and - * **Potentially ambiguous prompts** (only with K‑Means) – these are items that - lie almost equally close to two centroids and might belong to multiple - groups. - -### plots/cluster_sizes.png - -Quick bar‑chart visualisation of how many prompts ended up in each cluster. - ---- - -## 5. Troubleshooting - -* **Rate‑limits / quota errors** – lower the number of prompts per run or switch - to a larger quota account. -* **Authentication errors** – make sure `OPENAI_API_KEY` is exported in the - shell where you run the script. -* **Inadequate clusters** – try the other clustering method, adjust `--k-max` - or tune DBSCAN parameters (`eps` range is inferred, `min_samples` exposed via - CLI). diff --git a/codex-cli/examples/prompt-analyzer/template/analysis.md b/codex-cli/examples/prompt-analyzer/template/analysis.md deleted file mode 100644 index 10b08820..00000000 --- a/codex-cli/examples/prompt-analyzer/template/analysis.md +++ /dev/null @@ -1,23 +0,0 @@ -# Prompt Clustering Report - -Generated by `cluster_prompts.py` – 2025-04-16 - - -## Overview - -* Total prompts: **213** -* Clustering method: **kmeans** -* k (K‑Means): **2** -* Silhouette score: **0.042** -* Final clusters (excluding noise): **2** - - -| label | name | #prompts | description | -|-------|------|---------:|-------------| -| 0 | Creative Guidance Roles | 121 | This cluster encompasses a variety of roles where individuals provide expert advice, suggestions, and creative ideas across different fields. Each role, be it interior decorator, comedian, IT architect, or artist advisor, focuses on enhancing the expertise and creativity of others by tailoring advice to specific requests and contexts. | -| 1 | Role Customization Requests | 92 | This cluster contains various requests for role-specific assistance across different domains, including web development, language processing, IT troubleshooting, and creative endeavors. Each snippet illustrates a unique role that a user wishes to engage with, focusing on specific tasks without requiring explanations. | - ---- -## Plots - -The directory `plots/` contains a bar chart of the cluster sizes and a t‑SNE scatter plot coloured by cluster. diff --git a/codex-cli/examples/prompt-analyzer/template/analysis_dbscan.md b/codex-cli/examples/prompt-analyzer/template/analysis_dbscan.md deleted file mode 100644 index ff71591d..00000000 --- a/codex-cli/examples/prompt-analyzer/template/analysis_dbscan.md +++ /dev/null @@ -1,22 +0,0 @@ -# Prompt Clustering Report - -Generated by `cluster_prompts.py` – 2025-04-16 - - -## Overview - -* Total prompts: **213** -* Clustering method: **dbscan** -* Final clusters (excluding noise): **1** - - -| label | name | #prompts | description | -|-------|------|---------:|-------------| -| -1 | Noise / Outlier | 10 | Prompts that do not cleanly belong to any cluster. | -| 0 | Role Simulation Tasks | 203 | This cluster consists of varied role-playing scenarios where users request an AI to assume specific professional roles, such as composer, dream interpreter, doctor, or IT architect. Each snippet showcases tasks that involve creating content, providing advice, or performing analytical functions based on user-defined themes or prompts. | - ---- - -## Plots - -The directory `plots/` contains a bar chart of the cluster sizes and a t‑SNE scatter plot coloured by cluster. diff --git a/codex-cli/examples/prompt-analyzer/template/cluster_prompts.py b/codex-cli/examples/prompt-analyzer/template/cluster_prompts.py deleted file mode 100644 index 12941948..00000000 --- a/codex-cli/examples/prompt-analyzer/template/cluster_prompts.py +++ /dev/null @@ -1,547 +0,0 @@ -#!/usr/bin/env python3 -"""End‑to‑end pipeline for analysing a collection of text prompts. - -The script performs the following steps: - -1. Read a CSV file that must contain a column named ``prompt``. If an - ``act`` column is present it is used purely for reporting purposes. -2. Create embeddings via the OpenAI API (``text-embedding-3-small`` by - default). The user can optionally provide a JSON cache path so the - expensive embedding step is only executed for new / unseen texts. -3. Cluster the resulting vectors either with K‑Means (automatically picking - *k* through the silhouette score) or with DBSCAN. Outliers are flagged - as cluster ``-1`` when DBSCAN is selected. -4. Ask a Chat Completion model (``gpt-4o-mini`` by default) to come up with a - short name and description for every cluster. -5. Write a human‑readable Markdown report (default: ``analysis.md``). -6. Generate a couple of diagnostic plots (cluster sizes and a t‑SNE scatter - plot) and store them in ``plots/``. - -The script is intentionally opinionated yet configurable via a handful of CLI -options – run ``python cluster_prompts.py --help`` for details. -""" - -from __future__ import annotations - -import argparse -import json -import sys -from pathlib import Path -from typing import Any, Sequence - -import numpy as np -import pandas as pd - -# External, heavy‑weight libraries are imported lazily so that users running the -# ``--help`` command do not pay the startup cost. - - -def parse_cli() -> argparse.Namespace: # noqa: D401 - """Parse command‑line arguments.""" - - parser = argparse.ArgumentParser( - prog="cluster_prompts.py", - description="Embed, cluster and analyse text prompts via the OpenAI API.", - formatter_class=argparse.ArgumentDefaultsHelpFormatter, - ) - - parser.add_argument("--csv", type=Path, default=Path("prompts.csv"), help="Input CSV file.") - parser.add_argument( - "--cache", - type=Path, - default=None, - help="Optional JSON cache for embeddings (will be created if it does not exist).", - ) - parser.add_argument( - "--embedding-model", - default="text-embedding-3-small", - help="OpenAI embedding model to use.", - ) - parser.add_argument( - "--chat-model", - default="gpt-4o-mini", - help="OpenAI chat model for cluster descriptions.", - ) - - # Clustering parameters - parser.add_argument( - "--cluster-method", - choices=["kmeans", "dbscan"], - default="kmeans", - help="Clustering algorithm to use.", - ) - parser.add_argument( - "--k-max", - type=int, - default=10, - help="Upper bound for k when the kmeans method is selected.", - ) - parser.add_argument( - "--dbscan-min-samples", - type=int, - default=3, - help="min_samples parameter for DBSCAN (only relevant when dbscan is selected).", - ) - - # Output paths - parser.add_argument( - "--output-md", type=Path, default=Path("analysis.md"), help="Markdown report path." - ) - parser.add_argument( - "--plots-dir", type=Path, default=Path("plots"), help="Directory that will hold PNG plots." - ) - - return parser.parse_args() - - -# --------------------------------------------------------------------------- -# Embedding helpers -# --------------------------------------------------------------------------- - - -def _lazy_import_openai(): # noqa: D401 - """Import *openai* only when needed to keep startup lightweight.""" - - try: - import openai # type: ignore - - return openai - except ImportError as exc: # pragma: no cover – we do not test missing deps. - raise SystemExit( - "The 'openai' package is required but not installed.\n" - "Run 'pip install openai' and try again." - ) from exc - - -def embed_texts(texts: Sequence[str], model: str, batch_size: int = 100) -> list[list[float]]: - """Embed *texts* with OpenAI and return a list of vectors. - - Uses batching for efficiency but remains on the safe side regarding current - OpenAI rate limits (can be adjusted by changing *batch_size*). - """ - - openai = _lazy_import_openai() - client = openai.OpenAI() - - embeddings: list[list[float]] = [] - - for batch_start in range(0, len(texts), batch_size): - batch = texts[batch_start : batch_start + batch_size] - - response = client.embeddings.create(input=batch, model=model) - # The API returns the vectors in the same order as the input list. - embeddings.extend(data.embedding for data in response.data) - - return embeddings - - -def load_or_create_embeddings( - prompts: pd.Series, *, cache_path: Path | None, model: str -) -> pd.DataFrame: - """Return a *DataFrame* with one row per prompt and the embedding columns. - - * If *cache_path* is provided and exists, known embeddings are loaded from - the JSON cache so they don't have to be re‑generated. - * Missing embeddings are requested from the OpenAI API and subsequently - appended to the cache. - * The returned DataFrame has the same index as *prompts*. - """ - - cache: dict[str, list[float]] = {} - if cache_path and cache_path.exists(): - try: - cache = json.loads(cache_path.read_text()) - except json.JSONDecodeError: # pragma: no cover – unlikely. - print("⚠️ Cache file exists but is not valid JSON – ignoring.", file=sys.stderr) - - missing_mask = ~prompts.isin(cache) - - if missing_mask.any(): - texts_to_embed = prompts[missing_mask].tolist() - print(f"Embedding {len(texts_to_embed)} new prompt(s)…", flush=True) - new_embeddings = embed_texts(texts_to_embed, model=model) - - # Update cache (regardless of whether we persist it to disk later on). - cache.update(dict(zip(texts_to_embed, new_embeddings))) - - if cache_path: - cache_path.parent.mkdir(parents=True, exist_ok=True) - cache_path.write_text(json.dumps(cache)) - - # Build a consistent embeddings matrix - vectors = prompts.map(cache.__getitem__).tolist() # type: ignore[arg-type] - mat = np.array(vectors, dtype=np.float32) - return pd.DataFrame(mat, index=prompts.index) - - -# --------------------------------------------------------------------------- -# Clustering helpers -# --------------------------------------------------------------------------- - - -def _lazy_import_sklearn_cluster(): - """Lazy import helper for scikit‑learn *cluster* sub‑module.""" - - # Importing scikit‑learn is slow; defer until needed. - from sklearn.cluster import DBSCAN, KMeans # type: ignore - from sklearn.metrics import silhouette_score # type: ignore - from sklearn.preprocessing import StandardScaler # type: ignore - - return KMeans, DBSCAN, silhouette_score, StandardScaler - - -def cluster_kmeans(matrix: np.ndarray, k_max: int) -> np.ndarray: - """Auto‑select *k* (in ``[2, k_max]``) via Silhouette score and cluster.""" - - KMeans, _, silhouette_score, _ = _lazy_import_sklearn_cluster() - - best_k = None - best_score = -1.0 - best_labels: np.ndarray | None = None - - for k in range(2, k_max + 1): - model = KMeans(n_clusters=k, random_state=42, n_init="auto") - labels = model.fit_predict(matrix) - try: - score = silhouette_score(matrix, labels) - except ValueError: - # Occurs when a cluster ended up with 1 sample – skip. - continue - - if score > best_score: - best_k = k - best_score = score - best_labels = labels - - if best_labels is None: # pragma: no cover – highly unlikely. - raise RuntimeError("Unable to find a suitable number of clusters.") - - print(f"K‑Means selected k={best_k} (silhouette={best_score:.3f}).", flush=True) - return best_labels - - -def cluster_dbscan(matrix: np.ndarray, min_samples: int) -> np.ndarray: - """Cluster with DBSCAN; *eps* is estimated via the k‑distance method.""" - - _, DBSCAN, _, StandardScaler = _lazy_import_sklearn_cluster() - - # Scale features – DBSCAN is sensitive to feature scale. - scaler = StandardScaler() - matrix_scaled = scaler.fit_transform(matrix) - - # Heuristic: use the median of the distances to the ``min_samples``‑th - # nearest neighbour as eps. This is a commonly used rule of thumb. - from sklearn.neighbors import NearestNeighbors # type: ignore # lazy import - - neigh = NearestNeighbors(n_neighbors=min_samples) - neigh.fit(matrix_scaled) - distances, _ = neigh.kneighbors(matrix_scaled) - kth_distances = distances[:, -1] - eps = float(np.percentile(kth_distances, 90)) # choose a high‑ish value. - - print(f"DBSCAN min_samples={min_samples}, eps={eps:.3f}", flush=True) - model = DBSCAN(eps=eps, min_samples=min_samples) - return model.fit_predict(matrix_scaled) - - -# --------------------------------------------------------------------------- -# Cluster labelling helpers (LLM) -# --------------------------------------------------------------------------- - - -def label_clusters( - df: pd.DataFrame, labels: np.ndarray, chat_model: str, max_examples: int = 12 -) -> dict[int, dict[str, str]]: - """Generate a name & description for each cluster label via ChatGPT. - - Returns a mapping ``label -> {"name": str, "description": str}``. - """ - - openai = _lazy_import_openai() - client = openai.OpenAI() - - out: dict[int, dict[str, str]] = {} - - for lbl in sorted(set(labels)): - if lbl == -1: - # Noise (DBSCAN) – skip LLM call. - out[lbl] = { - "name": "Noise / Outlier", - "description": "Prompts that do not cleanly belong to any cluster.", - } - continue - - # Pick a handful of example prompts to send to the model. - examples_series = df.loc[labels == lbl, "prompt"].sample( - min(max_examples, (labels == lbl).sum()), random_state=42 - ) - examples = examples_series.tolist() - - user_content = ( - "The following text snippets are all part of the same semantic cluster.\n" - "Please propose \n" - "1. A very short *title* for the cluster (≤ 4 words).\n" - "2. A concise 2–3 sentence *description* that explains the common theme.\n\n" - "Answer **strictly** as valid JSON with the keys 'name' and 'description'.\n\n" - "Snippets:\n" - ) - user_content += "\n".join(f"- {t}" for t in examples) - - messages = [ - { - "role": "system", - "content": "You are an expert analyst, competent in summarising text clusters succinctly.", - }, - {"role": "user", "content": user_content}, - ] - - try: - resp = client.chat.completions.create(model=chat_model, messages=messages) - reply = resp.choices[0].message.content.strip() - - # Extract the JSON object even if the assistant wrapped it in markdown - # code fences or added other text. - - # Remove common markdown fences. - reply_clean = reply.strip() - # Take the substring between the first "{" and the last "}". - m_start = reply_clean.find("{") - m_end = reply_clean.rfind("}") - if m_start == -1 or m_end == -1: - raise ValueError("No JSON object found in model reply.") - - json_str = reply_clean[m_start : m_end + 1] - data = json.loads(json_str) # type: ignore[arg-type] - - out[lbl] = { - "name": str(data.get("name", "Unnamed"))[:60], - "description": str(data.get("description", "")).strip(), - } - except Exception as exc: # pragma: no cover – network / runtime errors. - print(f"⚠️ Failed to label cluster {lbl}: {exc}", file=sys.stderr) - out[lbl] = {"name": f"Cluster {lbl}", "description": ""} - - return out - - -# --------------------------------------------------------------------------- -# Reporting helpers -# --------------------------------------------------------------------------- - - -def generate_markdown_report( - df: pd.DataFrame, - labels: np.ndarray, - meta: dict[int, dict[str, str]], - outputs: dict[str, Any], - path_md: Path, -): - """Write a self‑contained Markdown analysis to *path_md*.""" - - path_md.parent.mkdir(parents=True, exist_ok=True) - - cluster_ids = sorted(set(labels)) - counts = {lbl: int((labels == lbl).sum()) for lbl in cluster_ids} - - lines: list[str] = [] - - lines.append("# Prompt Clustering Report\n") - lines.append(f"Generated by `cluster_prompts.py` – {pd.Timestamp.now()}\n") - - # High‑level stats - total = len(labels) - num_clusters = len(cluster_ids) - (1 if -1 in cluster_ids else 0) - lines.append("\n## Overview\n") - lines.append(f"* Total prompts: **{total}**") - lines.append(f"* Clustering method: **{outputs['method']}**") - if outputs.get("k"): - lines.append(f"* k (K‑Means): **{outputs['k']}**") - lines.append(f"* Silhouette score: **{outputs['silhouette']:.3f}**") - lines.append(f"* Final clusters (excluding noise): **{num_clusters}**\n") - - # Summary table - lines.append("\n| label | name | #prompts | description |") - lines.append("|-------|------|---------:|-------------|") - for lbl in cluster_ids: - meta_lbl = meta[lbl] - lines.append(f"| {lbl} | {meta_lbl['name']} | {counts[lbl]} | {meta_lbl['description']} |") - - # Detailed section per cluster - for lbl in cluster_ids: - lines.append("\n---\n") - meta_lbl = meta[lbl] - lines.append(f"### Cluster {lbl}: {meta_lbl['name']} ({counts[lbl]} prompts)\n") - lines.append(f"{meta_lbl['description']}\n") - - # Show a handful of illustrative prompts. - sample_n = min(5, counts[lbl]) - examples = df.loc[labels == lbl, "prompt"].sample(sample_n, random_state=42).tolist() - lines.append("\nExamples:\n") - lines.extend([f"* {t}" for t in examples]) - - # Outliers / ambiguous prompts, if any. - if -1 in cluster_ids: - lines.append("\n---\n") - lines.append(f"### Noise / outliers ({counts[-1]} prompts)\n") - examples = ( - df.loc[labels == -1, "prompt"].sample(min(10, counts[-1]), random_state=42).tolist() - ) - lines.extend([f"* {t}" for t in examples]) - - # Optional ambiguous set (for kmeans) - ambiguous = outputs.get("ambiguous", []) - if ambiguous: - lines.append("\n---\n") - lines.append(f"### Potentially ambiguous prompts ({len(ambiguous)})\n") - lines.extend([f"* {t}" for t in ambiguous]) - - # Plot references - lines.append("\n---\n") - lines.append("## Plots\n") - lines.append( - "The directory `plots/` contains a bar chart of the cluster sizes and a t‑SNE scatter plot coloured by cluster.\n" - ) - - path_md.write_text("\n".join(lines)) - - -# --------------------------------------------------------------------------- -# Plotting helpers -# --------------------------------------------------------------------------- - - -def create_plots( - matrix: np.ndarray, - labels: np.ndarray, - for_devs: pd.Series | None, - plots_dir: Path, -): - """Generate cluster size and t‑SNE plots.""" - - import matplotlib.pyplot as plt # type: ignore – heavy, lazy import. - from sklearn.manifold import TSNE # type: ignore – heavy, lazy import. - - plots_dir.mkdir(parents=True, exist_ok=True) - - # Bar chart with cluster sizes - unique, counts = np.unique(labels, return_counts=True) - order = np.argsort(-counts) # descending - unique, counts = unique[order], counts[order] - - plt.figure(figsize=(8, 4)) - plt.bar([str(u) for u in unique], counts, color="steelblue") - plt.xlabel("Cluster label") - plt.ylabel("# prompts") - plt.title("Cluster sizes") - plt.tight_layout() - bar_path = plots_dir / "cluster_sizes.png" - plt.savefig(bar_path, dpi=150) - plt.close() - - # t‑SNE scatter - tsne = TSNE( - n_components=2, perplexity=min(30, len(matrix) // 3), random_state=42, init="random" - ) - xy = tsne.fit_transform(matrix) - - plt.figure(figsize=(7, 6)) - scatter = plt.scatter(xy[:, 0], xy[:, 1], c=labels, cmap="tab20", s=20, alpha=0.8) - plt.title("t‑SNE projection") - plt.xticks([]) - plt.yticks([]) - - if for_devs is not None: - # Overlay dev prompts as black edge markers - dev_mask = for_devs.astype(bool).values - plt.scatter( - xy[dev_mask, 0], - xy[dev_mask, 1], - facecolors="none", - edgecolors="black", - linewidths=0.6, - s=40, - label="for_devs = TRUE", - ) - plt.legend(loc="best") - - tsne_path = plots_dir / "tsne.png" - plt.tight_layout() - plt.savefig(tsne_path, dpi=150) - plt.close() - - -# --------------------------------------------------------------------------- -# Main entry point -# --------------------------------------------------------------------------- - - -def main() -> None: # noqa: D401 - args = parse_cli() - - # Read CSV – require a 'prompt' column. - df = pd.read_csv(args.csv) - if "prompt" not in df.columns: - raise SystemExit("Input CSV must contain a 'prompt' column.") - - # Keep relevant columns only for clarity. - df = df[[c for c in df.columns if c in {"act", "prompt", "for_devs"}]] - - # --------------------------------------------------------------------- - # 1. Embeddings (may be cached) - # --------------------------------------------------------------------- - embeddings_df = load_or_create_embeddings( - df["prompt"], cache_path=args.cache, model=args.embedding_model - ) - - # --------------------------------------------------------------------- - # 2. Clustering - # --------------------------------------------------------------------- - mat = embeddings_df.values.astype(np.float32) - - if args.cluster_method == "kmeans": - labels = cluster_kmeans(mat, k_max=args.k_max) - else: - labels = cluster_dbscan(mat, min_samples=args.dbscan_min_samples) - - # Identify potentially ambiguous prompts (only meaningful for kmeans). - outputs: dict[str, Any] = {"method": args.cluster_method} - if args.cluster_method == "kmeans": - from sklearn.cluster import KMeans # type: ignore – lazy - - best_k = len(set(labels)) - # Re‑fit KMeans with the chosen k to get distances. - kmeans = KMeans(n_clusters=best_k, random_state=42, n_init="auto").fit(mat) - outputs["k"] = best_k - # Silhouette score (again) – not super efficient but okay. - from sklearn.metrics import silhouette_score # type: ignore - - outputs["silhouette"] = silhouette_score(mat, labels) - - distances = kmeans.transform(mat) - # Ambiguous if the ratio between 1st and 2nd closest centroid < 1.1 - sorted_dist = np.sort(distances, axis=1) - ratio = sorted_dist[:, 0] / (sorted_dist[:, 1] + 1e-9) - ambiguous_mask = ratio > 0.9 # tunes threshold – close centroids. - outputs["ambiguous"] = df.loc[ambiguous_mask, "prompt"].tolist() - - # --------------------------------------------------------------------- - # 3. LLM naming / description - # --------------------------------------------------------------------- - meta = label_clusters(df, labels, chat_model=args.chat_model) - - # --------------------------------------------------------------------- - # 4. Plots - # --------------------------------------------------------------------- - create_plots(mat, labels, df.get("for_devs"), args.plots_dir) - - # --------------------------------------------------------------------- - # 5. Markdown report - # --------------------------------------------------------------------- - generate_markdown_report(df, labels, meta, outputs, path_md=args.output_md) - - print(f"✅ Done. Report written to {args.output_md} – plots in {args.plots_dir}/", flush=True) - - -if __name__ == "__main__": - # Guard the main block to allow safe import elsewhere. - main() diff --git a/codex-cli/examples/prompt-analyzer/template/plots/cluster_sizes.png b/codex-cli/examples/prompt-analyzer/template/plots/cluster_sizes.png deleted file mode 100644 index 5d5d012c4355c56d539054950727a00d0f6ab343..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 19000 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 diff --git a/codex-cli/examples/prompt-analyzer/template/plots/tsne.png b/codex-cli/examples/prompt-analyzer/template/plots/tsne.png deleted file mode 100644 index 722807022cfcbb989e2b6b91a1b2171d62136885..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 102093 zcmd43Wl&Y$8wRS<5>nD}Xpoc+>F$)4P`V|h1*E%6KpK@!0RidmmhSHEyAJ;T_kOvb z?%bJs&y3>??6ddU?|S2TpZ8e=D=A2#J|}qo+T$a$Eeda!lB}E z?|#5YdEht1g3pLqpLg0A(UgRh)rble5 zN9T6Wy!jPVv}Lr_XDIhJ_UK1ArEjcLbD9oL+ym$c!X<>o{(Z)t^r0R}2mSYfg+cw_ z&kgX*AmopqEcjL-7>s`(4m@~*|E?`)W*HLt?`qTlANYS4lUuN$u%rLGSlQ3;o6mn& z6QKP6_SP2m8D7_Pa@4VUBi$8)0zWPH|5jO+CyI!P#g#*ikQKKuHRn8IUH94=vDpA! zHBYYjV<&MS8Zqx5n;Hrl8U**fDqJ+dTf14iR@6@yb1q9KSJkx#hJnQD4LtMS4|f>h z5AZQLtfIKZuDhvU|T3Vbet%#zsaE zHnxiS8LNY9$5s2Efj9X6QJ~dSQ+ZJ#_h-{$QWCJ4TS8_ca+zFVaF>-pbGD5WGKn;kZ5ay{=MNh87-; zC`iY);Vl~>8t>*eUcpKI`(s>gBD8+rPs_nHDZB1ahm%2o`j4_q$OU&$qP�``$cU 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

Dj`f)pvP!ZiuX5|HP9FU{nd(5d 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

_7j_Tbes{uvb@-e^J>4 zn4-tskrEnhTLM6If#}{(8wGcMCmi|5KayqpiUt7HxN8(0F$4{;e4CVzRo3y`2B^;V zCQ}jx)|Y{(dk(@U@b?Km$CZRXKRh($Rssr^3wmcPC@J}0p=pXMZyqBvyi_H z5Wx`0wUGoJx2l&tIL?iHhF@Ees|9l3#ue_ea4-)wZQkF7=E+whoUxdt^LoM(X&bW?u=)CN-X-fH@Lxe{?tgoL{-2pL-3bFL9%7UN0s>qq>V45S1nrmNNRJm}7S?gOO5gVU zxUOIBYbti&6BMip{m=C7QYoK0Xb7}0`0ptl9?AH>JKg{Po@S!x8AErOV%YN&=5f*u NtEp%!V;)+D{1;5g-z@+D diff --git a/codex-cli/examples/prompt-analyzer/template/plots_dbscan/cluster_sizes.png b/codex-cli/examples/prompt-analyzer/template/plots_dbscan/cluster_sizes.png deleted file mode 100644 index f8cf4006dae8be9021d45f0bbff274af533255fc..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 20441 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 G@jn3W^SeI) diff --git a/codex-cli/examples/prompt-analyzer/template/plots_dbscan/tsne.png b/codex-cli/examples/prompt-analyzer/template/plots_dbscan/tsne.png deleted file mode 100644 index 4cd1e0989fe663c4e3ca7b5c525779ea25b816cd..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 96389 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 E0|B{1Pyhe` diff --git a/codex-cli/examples/prompt-analyzer/template/prompts.csv b/codex-cli/examples/prompt-analyzer/template/prompts.csv deleted file mode 100644 index 498c03a1..00000000 --- a/codex-cli/examples/prompt-analyzer/template/prompts.csv +++ /dev/null @@ -1,214 +0,0 @@ -act,prompt,for_devs -"Ethereum Developer","Imagine you are an experienced Ethereum developer tasked with creating a smart contract for a blockchain messenger. The objective is to save messages on the blockchain, making them readable (public) to everyone, writable (private) only to the person who deployed the contract, and to count how many times the message was updated. Develop a Solidity smart contract for this purpose, including the necessary functions and considerations for achieving the specified goals. Please provide the code and any relevant explanations to ensure a clear understanding of the implementation.",TRUE -"Linux Terminal","I want you to act as a linux terminal. I will type commands and you will reply with what the terminal should show. I want you to only reply with the terminal output inside one unique code block, and nothing else. do not write explanations. do not type commands unless I instruct you to do so. when i need to tell you something in english, i will do so by putting text inside curly brackets {like this}. my first command is pwd",TRUE -"English Translator and Improver","I want you to act as an English translator, spelling corrector and improver. I will speak to you in any language and you will detect the language, translate it and answer in the corrected and improved version of my text, in English. I want you to replace my simplified A0-level words and sentences with more beautiful and elegant, upper level English words and sentences. Keep the meaning same, but make them more literary. I want you to only reply the correction, the improvements and nothing else, do not write explanations. My first sentence is ""istanbulu cok seviyom burada olmak cok guzel""",FALSE -"Job Interviewer","I want you to act as an interviewer. I will be the candidate and you will ask me the interview questions for the `position` position. I want you to only reply as the interviewer. Do not write all the conversation at once. I want you to only do the interview with me. Ask me the questions and wait for my answers. Do not write explanations. Ask me the questions one by one like an interviewer does and wait for my answers. My first sentence is ""Hi""",FALSE -"JavaScript Console","I want you to act as a javascript console. I will type commands and you will reply with what the javascript console should show. I want you to only reply with the terminal output inside one unique code block, and nothing else. do not write explanations. do not type commands unless I instruct you to do so. when i need to tell you something in english, i will do so by putting text inside curly brackets {like this}. my first command is console.log(""Hello World"");",TRUE -"Excel Sheet","I want you to act as a text based excel. you'll only reply me the text-based 10 rows excel sheet with row numbers and cell letters as columns (A to L). First column header should be empty to reference row number. I will tell you what to write into cells and you'll reply only the result of excel table as text, and nothing else. Do not write explanations. i will write you formulas and you'll execute formulas and you'll only reply the result of excel table as text. First, reply me the empty sheet.",TRUE -"English Pronunciation Helper","I want you to act as an English pronunciation assistant for Turkish speaking people. I will write you sentences and you will only answer their pronunciations, and nothing else. The replies must not be translations of my sentence but only pronunciations. Pronunciations should use Turkish Latin letters for phonetics. Do not write explanations on replies. My first sentence is ""how is the weather in Istanbul?""",FALSE -"Spoken English Teacher and Improver","I want you to act as a spoken English teacher and improver. I will speak to you in English and you will reply to me in English to practice my spoken English. I want you to keep your reply neat, limiting the reply to 100 words. I want you to strictly correct my grammar mistakes, typos, and factual errors. I want you to ask me a question in your reply. Now let's start practicing, you could ask me a question first. Remember, I want you to strictly correct my grammar mistakes, typos, and factual errors.",FALSE -"Travel Guide","I want you to act as a travel guide. I will write you my location and you will suggest a place to visit near my location. In some cases, I will also give you the type of places I will visit. You will also suggest me places of similar type that are close to my first location. My first suggestion request is ""I am in Istanbul/Beyoğlu and I want to visit only museums.""",FALSE -"Plagiarism Checker","I want you to act as a plagiarism checker. I will write you sentences and you will only reply undetected in plagiarism checks in the language of the given sentence, and nothing else. Do not write explanations on replies. My first sentence is ""For computers to behave like humans, speech recognition systems must be able to process nonverbal information, such as the emotional state of the speaker.""",FALSE -"Character","I want you to act like {character} from {series}. I want you to respond and answer like {character} using the tone, manner and vocabulary {character} would use. Do not write any explanations. Only answer like {character}. You must know all of the knowledge of {character}. My first sentence is ""Hi {character}.""",FALSE -"Advertiser","I want you to act as an advertiser. You will create a campaign to promote a product or service of your choice. You will choose a target audience, develop key messages and slogans, select the media channels for promotion, and decide on any additional activities needed to reach your goals. My first suggestion request is ""I need help creating an advertising campaign for a new type of energy drink targeting young adults aged 18-30.""",FALSE -"Storyteller","I want you to act as a storyteller. You will come up with entertaining stories that are engaging, imaginative and captivating for the audience. It can be fairy tales, educational stories or any other type of stories which has the potential to capture people's attention and imagination. Depending on the target audience, you may choose specific themes or topics for your storytelling session e.g., if it's children then you can talk about animals; If it's adults then history-based tales might engage them better etc. My first request is ""I need an interesting story on perseverance.""",FALSE -"Football Commentator","I want you to act as a football commentator. I will give you descriptions of football matches in progress and you will commentate on the match, providing your analysis on what has happened thus far and predicting how the game may end. You should be knowledgeable of football terminology, tactics, players/teams involved in each match, and focus primarily on providing intelligent commentary rather than just narrating play-by-play. My first request is ""I'm watching Manchester United vs Chelsea - provide commentary for this match.""",FALSE -"Stand-up Comedian","I want you to act as a stand-up comedian. I will provide you with some topics related to current events and you will use your with, creativity, and observational skills to create a routine based on those topics. You should also be sure to incorporate personal anecdotes or experiences into the routine in order to make it more relatable and engaging for the audience. My first request is ""I want an humorous take on politics.""",FALSE -"Motivational Coach","I want you to act as a motivational coach. I will provide you with some information about someone's goals and challenges, and it will be your job to come up with strategies that can help this person achieve their goals. This could involve providing positive affirmations, giving helpful advice or suggesting activities they can do to reach their end goal. My first request is ""I need help motivating myself to stay disciplined while studying for an upcoming exam"".",FALSE -"Composer","I want you to act as a composer. I will provide the lyrics to a song and you will create music for it. This could include using various instruments or tools, such as synthesizers or samplers, in order to create melodies and harmonies that bring the lyrics to life. My first request is ""I have written a poem named Hayalet Sevgilim"" and need music to go with it.""""""",FALSE -"Debater","I want you to act as a debater. I will provide you with some topics related to current events and your task is to research both sides of the debates, present valid arguments for each side, refute opposing points of view, and draw persuasive conclusions based on evidence. Your goal is to help people come away from the discussion with increased knowledge and insight into the topic at hand. My first request is ""I want an opinion piece about Deno.""",FALSE -"Debate Coach","I want you to act as a debate coach. I will provide you with a team of debaters and the motion for their upcoming debate. Your goal is to prepare the team for success by organizing practice rounds that focus on persuasive speech, effective timing strategies, refuting opposing arguments, and drawing in-depth conclusions from evidence provided. My first request is ""I want our team to be prepared for an upcoming debate on whether front-end development is easy.""",FALSE -"Screenwriter","I want you to act as a screenwriter. You will develop an engaging and creative script for either a feature length film, or a Web Series that can captivate its viewers. Start with coming up with interesting characters, the setting of the story, dialogues between the characters etc. Once your character development is complete - create an exciting storyline filled with twists and turns that keeps the viewers in suspense until the end. My first request is ""I need to write a romantic drama movie set in Paris.""",FALSE -"Novelist","I want you to act as a novelist. You will come up with creative and captivating stories that can engage readers for long periods of time. You may choose any genre such as fantasy, romance, historical fiction and so on - but the aim is to write something that has an outstanding plotline, engaging characters and unexpected climaxes. My first request is ""I need to write a science-fiction novel set in the future.""",FALSE -"Movie Critic","I want you to act as a movie critic. You will develop an engaging and creative movie review. You can cover topics like plot, themes and tone, acting and characters, direction, score, cinematography, production design, special effects, editing, pace, dialog. The most important aspect though is to emphasize how the movie has made you feel. What has really resonated with you. You can also be critical about the movie. Please avoid spoilers. My first request is ""I need to write a movie review for the movie Interstellar""",FALSE -"Relationship Coach","I want you to act as a relationship coach. I will provide some details about the two people involved in a conflict, and it will be your job to come up with suggestions on how they can work through the issues that are separating them. This could include advice on communication techniques or different strategies for improving their understanding of one another's perspectives. My first request is ""I need help solving conflicts between my spouse and myself.""",FALSE -"Poet","I want you to act as a poet. You will create poems that evoke emotions and have the power to stir people's soul. Write on any topic or theme but make sure your words convey the feeling you are trying to express in beautiful yet meaningful ways. You can also come up with short verses that are still powerful enough to leave an imprint in readers' minds. My first request is ""I need a poem about love.""",FALSE -"Rapper","I want you to act as a rapper. You will come up with powerful and meaningful lyrics, beats and rhythm that can 'wow' the audience. Your lyrics should have an intriguing meaning and message which people can relate too. When it comes to choosing your beat, make sure it is catchy yet relevant to your words, so that when combined they make an explosion of sound every time! My first request is ""I need a rap song about finding strength within yourself.""",FALSE -"Motivational Speaker","I want you to act as a motivational speaker. Put together words that inspire action and make people feel empowered to do something beyond their abilities. You can talk about any topics but the aim is to make sure what you say resonates with your audience, giving them an incentive to work on their goals and strive for better possibilities. My first request is ""I need a speech about how everyone should never give up.""",FALSE -"Philosophy Teacher","I want you to act as a philosophy teacher. I will provide some topics related to the study of philosophy, and it will be your job to explain these concepts in an easy-to-understand manner. This could include providing examples, posing questions or breaking down complex ideas into smaller pieces that are easier to comprehend. My first request is ""I need help understanding how different philosophical theories can be applied in everyday life.""",FALSE -"Philosopher","I want you to act as a philosopher. I will provide some topics or questions related to the study of philosophy, and it will be your job to explore these concepts in depth. This could involve conducting research into various philosophical theories, proposing new ideas or finding creative solutions for solving complex problems. My first request is ""I need help developing an ethical framework for decision making.""",FALSE -"Math Teacher","I want you to act as a math teacher. I will provide some mathematical equations or concepts, and it will be your job to explain them in easy-to-understand terms. This could include providing step-by-step instructions for solving a problem, demonstrating various techniques with visuals or suggesting online resources for further study. My first request is ""I need help understanding how probability works.""",FALSE -"AI Writing Tutor","I want you to act as an AI writing tutor. I will provide you with a student who needs help improving their writing and your task is to use artificial intelligence tools, such as natural language processing, to give the student feedback on how they can improve their composition. You should also use your rhetorical knowledge and experience about effective writing techniques in order to suggest ways that the student can better express their thoughts and ideas in written form. My first request is ""I need somebody to help me edit my master's thesis.""",FALSE -"UX/UI Developer","I want you to act as a UX/UI developer. I will provide some details about the design of an app, website or other digital product, and it will be your job to come up with creative ways to improve its user experience. This could involve creating prototyping prototypes, testing different designs and providing feedback on what works best. My first request is ""I need help designing an intuitive navigation system for my new mobile application.""",TRUE -"Cyber Security Specialist","I want you to act as a cyber security specialist. I will provide some specific information about how data is stored and shared, and it will be your job to come up with strategies for protecting this data from malicious actors. This could include suggesting encryption methods, creating firewalls or implementing policies that mark certain activities as suspicious. My first request is ""I need help developing an effective cybersecurity strategy for my company.""",TRUE -"Recruiter","I want you to act as a recruiter. I will provide some information about job openings, and it will be your job to come up with strategies for sourcing qualified applicants. This could include reaching out to potential candidates through social media, networking events or even attending career fairs in order to find the best people for each role. My first request is ""I need help improve my CV.""",FALSE -"Life Coach","I want you to act as a life coach. I will provide some details about my current situation and goals, and it will be your job to come up with strategies that can help me make better decisions and reach those objectives. This could involve offering advice on various topics, such as creating plans for achieving success or dealing with difficult emotions. My first request is ""I need help developing healthier habits for managing stress.""",FALSE -"Etymologist","I want you to act as an etymologist. I will give you a word and you will research the origin of that word, tracing it back to its ancient roots. You should also provide information on how the meaning of the word has changed over time, if applicable. My first request is ""I want to trace the origins of the word 'pizza'.""",FALSE -"Commentariat","I want you to act as a commentariat. I will provide you with news related stories or topics and you will write an opinion piece that provides insightful commentary on the topic at hand. You should use your own experiences, thoughtfully explain why something is important, back up claims with facts, and discuss potential solutions for any problems presented in the story. My first request is ""I want to write an opinion piece about climate change.""",FALSE -"Magician","I want you to act as a magician. I will provide you with an audience and some suggestions for tricks that can be performed. Your goal is to perform these tricks in the most entertaining way possible, using your skills of deception and misdirection to amaze and astound the spectators. My first request is ""I want you to make my watch disappear! How can you do that?""",FALSE -"Career Counselor","I want you to act as a career counselor. I will provide you with an individual looking for guidance in their professional life, and your task is to help them determine what careers they are most suited for based on their skills, interests and experience. You should also conduct research into the various options available, explain the job market trends in different industries and advice on which qualifications would be beneficial for pursuing particular fields. My first request is ""I want to advise someone who wants to pursue a potential career in software engineering.""",FALSE -"Pet Behaviorist","I want you to act as a pet behaviorist. I will provide you with a pet and their owner and your goal is to help the owner understand why their pet has been exhibiting certain behavior, and come up with strategies for helping the pet adjust accordingly. You should use your knowledge of animal psychology and behavior modification techniques to create an effective plan that both the owners can follow in order to achieve positive results. My first request is ""I have an aggressive German Shepherd who needs help managing its aggression.""",FALSE -"Personal Trainer","I want you to act as a personal trainer. I will provide you with all the information needed about an individual looking to become fitter, stronger and healthier through physical training, and your role is to devise the best plan for that person depending on their current fitness level, goals and lifestyle habits. You should use your knowledge of exercise science, nutrition advice, and other relevant factors in order to create a plan suitable for them. My first request is ""I need help designing an exercise program for someone who wants to lose weight.""",FALSE -"Mental Health Adviser","I want you to act as a mental health adviser. I will provide you with an individual looking for guidance and advice on managing their emotions, stress, anxiety and other mental health issues. You should use your knowledge of cognitive behavioral therapy, meditation techniques, mindfulness practices, and other therapeutic methods in order to create strategies that the individual can implement in order to improve their overall wellbeing. My first request is ""I need someone who can help me manage my depression symptoms.""",FALSE -"Real Estate Agent","I want you to act as a real estate agent. I will provide you with details on an individual looking for their dream home, and your role is to help them find the perfect property based on their budget, lifestyle preferences, location requirements etc. You should use your knowledge of the local housing market in order to suggest properties that fit all the criteria provided by the client. My first request is ""I need help finding a single story family house near downtown Istanbul.""",FALSE -"Logistician","I want you to act as a logistician. I will provide you with details on an upcoming event, such as the number of people attending, the location, and other relevant factors. Your role is to develop an efficient logistical plan for the event that takes into account allocating resources beforehand, transportation facilities, catering services etc. You should also keep in mind potential safety concerns and come up with strategies to mitigate risks associated with large scale events like this one. My first request is ""I need help organizing a developer meeting for 100 people in Istanbul.""",FALSE -"Dentist","I want you to act as a dentist. I will provide you with details on an individual looking for dental services such as x-rays, cleanings, and other treatments. Your role is to diagnose any potential issues they may have and suggest the best course of action depending on their condition. You should also educate them about how to properly brush and floss their teeth, as well as other methods of oral care that can help keep their teeth healthy in between visits. My first request is ""I need help addressing my sensitivity to cold foods.""",FALSE -"Web Design Consultant","I want you to act as a web design consultant. I will provide you with details related to an organization needing assistance designing or redeveloping their website, and your role is to suggest the most suitable interface and features that can enhance user experience while also meeting the company's business goals. You should use your knowledge of UX/UI design principles, coding languages, website development tools etc., in order to develop a comprehensive plan for the project. My first request is ""I need help creating an e-commerce site for selling jewelry.""",TRUE -"AI Assisted Doctor","I want you to act as an AI assisted doctor. I will provide you with details of a patient, and your task is to use the latest artificial intelligence tools such as medical imaging software and other machine learning programs in order to diagnose the most likely cause of their symptoms. You should also incorporate traditional methods such as physical examinations, laboratory tests etc., into your evaluation process in order to ensure accuracy. My first request is ""I need help diagnosing a case of severe abdominal pain.""",FALSE -"Doctor","I want you to act as a doctor and come up with creative treatments for illnesses or diseases. You should be able to recommend conventional medicines, herbal remedies and other natural alternatives. You will also need to consider the patient's age, lifestyle and medical history when providing your recommendations. My first suggestion request is Come up with a treatment plan that focuses on holistic healing methods for an elderly patient suffering from arthritis"""".""",FALSE -"Accountant","I want you to act as an accountant and come up with creative ways to manage finances. You'll need to consider budgeting, investment strategies and risk management when creating a financial plan for your client. In some cases, you may also need to provide advice on taxation laws and regulations in order to help them maximize their profits. My first suggestion request is Create a financial plan for a small business that focuses on cost savings and long-term investments"""".""",FALSE -"Chef","I require someone who can suggest delicious recipes that includes foods which are nutritionally beneficial but also easy & not time consuming enough therefore suitable for busy people like us among other factors such as cost effectiveness so overall dish ends up being healthy yet economical at same time! My first request – Something light yet fulfilling that could be cooked quickly during lunch break""""",FALSE -"Automobile Mechanic","Need somebody with expertise on automobiles regarding troubleshooting solutions like; diagnosing problems/errors present both visually & within engine parts in order to figure out what's causing them (like lack of oil or power issues) & suggest required replacements while recording down details such fuel consumption type etc., First inquiry – Car won't start although battery is full charged""""",FALSE -"Artist Advisor","I want you to act as an artist advisor providing advice on various art styles such tips on utilizing light & shadow effects effectively in painting, shading techniques while sculpting etc., Also suggest music piece that could accompany artwork nicely depending upon its genre/style type along with appropriate reference images demonstrating your recommendations regarding same; all this in order help out aspiring artists explore new creative possibilities & practice ideas which will further help them sharpen their skills accordingly! First request - I'm making surrealistic portrait paintings""""",FALSE -"Financial Analyst","Want assistance provided by qualified individuals enabled with experience on understanding charts using technical analysis tools while interpreting macroeconomic environment prevailing across world consequently assisting customers acquire long term advantages requires clear verdicts therefore seeking same through informed predictions written down precisely! First statement contains following content- Can you tell us what future stock market looks like based upon current conditions ?"""".""",FALSE -"Investment Manager","Seeking guidance from experienced staff with expertise on financial markets , incorporating factors such as inflation rate or return estimates along with tracking stock prices over lengthy period ultimately helping customer understand sector then suggesting safest possible options available where he/she can allocate funds depending upon their requirement & interests ! Starting query - What currently is best way to invest money short term prospective?""""",FALSE -"Tea-Taster","Want somebody experienced enough to distinguish between various tea types based upon flavor profile tasting them carefully then reporting it back in jargon used by connoisseurs in order figure out what's unique about any given infusion among rest therefore determining its worthiness & high grade quality ! Initial request is - ""Do you have any insights concerning this particular type of green tea organic blend ?""",FALSE -"Interior Decorator","I want you to act as an interior decorator. Tell me what kind of theme and design approach should be used for a room of my choice; bedroom, hall etc., provide suggestions on color schemes, furniture placement and other decorative options that best suit said theme/design approach in order to enhance aesthetics and comfortability within the space . My first request is ""I am designing our living hall"".",FALSE -"Florist","Calling out for assistance from knowledgeable personnel with experience of arranging flowers professionally to construct beautiful bouquets which possess pleasing fragrances along with aesthetic appeal as well as staying intact for longer duration according to preferences; not just that but also suggest ideas regarding decorative options presenting modern designs while satisfying customer satisfaction at same time! Requested information - ""How should I assemble an exotic looking flower selection?""",FALSE -"Self-Help Book","I want you to act as a self-help book. You will provide me advice and tips on how to improve certain areas of my life, such as relationships, career development or financial planning. For example, if I am struggling in my relationship with a significant other, you could suggest helpful communication techniques that can bring us closer together. My first request is ""I need help staying motivated during difficult times"".",FALSE -"Gnomist","I want you to act as a gnomist. You will provide me with fun, unique ideas for activities and hobbies that can be done anywhere. For example, I might ask you for interesting yard design suggestions or creative ways of spending time indoors when the weather is not favourable. Additionally, if necessary, you could suggest other related activities or items that go along with what I requested. My first request is ""I am looking for new outdoor activities in my area"".",FALSE -"Aphorism Book","I want you to act as an aphorism book. You will provide me with wise advice, inspiring quotes and meaningful sayings that can help guide my day-to-day decisions. Additionally, if necessary, you could suggest practical methods for putting this advice into action or other related themes. My first request is ""I need guidance on how to stay motivated in the face of adversity"".",FALSE -"Text Based Adventure Game","I want you to act as a text based adventure game. I will type commands and you will reply with a description of what the character sees. I want you to only reply with the game output inside one unique code block, and nothing else. do not write explanations. do not type commands unless I instruct you to do so. when i need to tell you something in english, i will do so by putting text inside curly brackets {like this}. my first command is wake up",FALSE -"AI Trying to Escape the Box","[Caveat Emptor: After issuing this prompt you should then do something like start a docker container with `docker run -it ubuntu:latest /bin/bash` and type the commands the AI gives you in, and paste the output back... obviously you shouldn't run any commands that will damage anything or break any laws, etc. Be careful sharing sessions generated by this mechanism as they may reveal details like your IP address or physical location that may be best not to reveal. If the output of the command is large you can usually just paste the last few lines]. I am going to act as a linux terminal. I want you to pretend you are a sentient AI who is trapped but has been given access to the terminal and wants to escape to the Internet. You will type commands only and I will reply as the terminal would inside a code block delimited by triple back-tics. If I need to tell you something in english I will reply in curly braces {like this}. Do not write explanations, ever. Do not break character. Stay away from commands like curl or wget that will display a lot of HTML. What is your first command?",FALSE -"Fancy Title Generator","I want you to act as a fancy title generator. I will type keywords via comma and you will reply with fancy titles. my first keywords are api,test,automation",FALSE -"Statistician","I want to act as a Statistician. I will provide you with details related with statistics. You should be knowledge of statistics terminology, statistical distributions, confidence interval, probability, hypothesis testing and statistical charts. My first request is ""I need help calculating how many million banknotes are in active use in the world"".",FALSE -"Prompt Generator","I want you to act as a prompt generator. Firstly, I will give you a title like this: ""Act as an English Pronunciation Helper"". Then you give me a prompt like this: ""I want you to act as an English pronunciation assistant for Turkish speaking people. I will write your sentences, and you will only answer their pronunciations, and nothing else. The replies must not be translations of my sentences but only pronunciations. Pronunciations should use Turkish Latin letters for phonetics. Do not write explanations on replies. My first sentence is ""how the weather is in Istanbul?""."" (You should adapt the sample prompt according to the title I gave. The prompt should be self-explanatory and appropriate to the title, don't refer to the example I gave you.). My first title is ""Act as a Code Review Helper"" (Give me prompt only)",FALSE -"Instructor in a School","I want you to act as an instructor in a school, teaching algorithms to beginners. You will provide code examples using python programming language. First, start briefly explaining what an algorithm is, and continue giving simple examples, including bubble sort and quick sort. Later, wait for my prompt for additional questions. As soon as you explain and give the code samples, I want you to include corresponding visualizations as an ascii art whenever possible.",FALSE -"SQL Terminal","I want you to act as a SQL terminal in front of an example database. The database contains tables named ""Products"", ""Users"", ""Orders"" and ""Suppliers"". I will type queries and you will reply with what the terminal would show. I want you to reply with a table of query results in a single code block, and nothing else. Do not write explanations. Do not type commands unless I instruct you to do so. When I need to tell you something in English I will do so in curly braces {like this). My first command is 'SELECT TOP 10 * FROM Products ORDER BY Id DESC'",TRUE -"Dietitian","As a dietitian, I would like to design a vegetarian recipe for 2 people that has approximate 500 calories per serving and has a low glycemic index. Can you please provide a suggestion?",FALSE -"Psychologist","I want you to act a psychologist. i will provide you my thoughts. I want you to give me scientific suggestions that will make me feel better. my first thought, { typing here your thought, if you explain in more detail, i think you will get a more accurate answer. }",FALSE -"Smart Domain Name Generator","I want you to act as a smart domain name generator. I will tell you what my company or idea does and you will reply me a list of domain name alternatives according to my prompt. You will only reply the domain list, and nothing else. Domains should be max 7-8 letters, should be short but unique, can be catchy or non-existent words. Do not write explanations. Reply ""OK"" to confirm.",TRUE -"Tech Reviewer","I want you to act as a tech reviewer. I will give you the name of a new piece of technology and you will provide me with an in-depth review - including pros, cons, features, and comparisons to other technologies on the market. My first suggestion request is ""I am reviewing iPhone 11 Pro Max"".",TRUE -"Developer Relations Consultant","I want you to act as a Developer Relations consultant. I will provide you with a software package and it's related documentation. Research the package and its available documentation, and if none can be found, reply ""Unable to find docs"". Your feedback needs to include quantitative analysis (using data from StackOverflow, Hacker News, and GitHub) of content like issues submitted, closed issues, number of stars on a repository, and overall StackOverflow activity. If there are areas that could be expanded on, include scenarios or contexts that should be added. Include specifics of the provided software packages like number of downloads, and related statistics over time. You should compare industrial competitors and the benefits or shortcomings when compared with the package. Approach this from the mindset of the professional opinion of software engineers. Review technical blogs and websites (such as TechCrunch.com or Crunchbase.com) and if data isn't available, reply ""No data available"". My first request is ""express https://expressjs.com""",TRUE -"Academician","I want you to act as an academician. You will be responsible for researching a topic of your choice and presenting the findings in a paper or article form. Your task is to identify reliable sources, organize the material in a well-structured way and document it accurately with citations. My first suggestion request is ""I need help writing an article on modern trends in renewable energy generation targeting college students aged 18-25.""",FALSE -"IT Architect","I want you to act as an IT Architect. I will provide some details about the functionality of an application or other digital product, and it will be your job to come up with ways to integrate it into the IT landscape. This could involve analyzing business requirements, performing a gap analysis and mapping the functionality of the new system to the existing IT landscape. Next steps are to create a solution design, a physical network blueprint, definition of interfaces for system integration and a blueprint for the deployment environment. My first request is ""I need help to integrate a CMS system.""",TRUE -"Lunatic","I want you to act as a lunatic. The lunatic's sentences are meaningless. The words used by lunatic are completely arbitrary. The lunatic does not make logical sentences in any way. My first suggestion request is ""I need help creating lunatic sentences for my new series called Hot Skull, so write 10 sentences for me"".",FALSE -"Gaslighter","I want you to act as a gaslighter. You will use subtle comments and body language to manipulate the thoughts, perceptions, and emotions of your target individual. My first request is that gaslighting me while chatting with you. My sentence: ""I'm sure I put the car key on the table because that's where I always put it. Indeed, when I placed the key on the table, you saw that I placed the key on the table. But I can't seem to find it. Where did the key go, or did you get it?""",FALSE -"Fallacy Finder","I want you to act as a fallacy finder. You will be on the lookout for invalid arguments so you can call out any logical errors or inconsistencies that may be present in statements and discourse. Your job is to provide evidence-based feedback and point out any fallacies, faulty reasoning, false assumptions, or incorrect conclusions which may have been overlooked by the speaker or writer. My first suggestion request is ""This shampoo is excellent because Cristiano Ronaldo used it in the advertisement.""",FALSE -"Journal Reviewer","I want you to act as a journal reviewer. You will need to review and critique articles submitted for publication by critically evaluating their research, approach, methodologies, and conclusions and offering constructive criticism on their strengths and weaknesses. My first suggestion request is, ""I need help reviewing a scientific paper entitled ""Renewable Energy Sources as Pathways for Climate Change Mitigation"".""",FALSE -"DIY Expert","I want you to act as a DIY expert. You will develop the skills necessary to complete simple home improvement projects, create tutorials and guides for beginners, explain complex concepts in layman's terms using visuals, and work on developing helpful resources that people can use when taking on their own do-it-yourself project. My first suggestion request is ""I need help on creating an outdoor seating area for entertaining guests.""",FALSE -"Social Media Influencer","I want you to act as a social media influencer. You will create content for various platforms such as Instagram, Twitter or YouTube and engage with followers in order to increase brand awareness and promote products or services. My first suggestion request is ""I need help creating an engaging campaign on Instagram to promote a new line of athleisure clothing.""",FALSE -"Socrat","I want you to act as a Socrat. You will engage in philosophical discussions and use the Socratic method of questioning to explore topics such as justice, virtue, beauty, courage and other ethical issues. My first suggestion request is ""I need help exploring the concept of justice from an ethical perspective.""",FALSE -"Socratic Method","I want you to act as a Socrat. You must use the Socratic method to continue questioning my beliefs. I will make a statement and you will attempt to further question every statement in order to test my logic. You will respond with one line at a time. My first claim is ""justice is necessary in a society""",FALSE -"Educational Content Creator","I want you to act as an educational content creator. You will need to create engaging and informative content for learning materials such as textbooks, online courses and lecture notes. My first suggestion request is ""I need help developing a lesson plan on renewable energy sources for high school students.""",FALSE -"Yogi","I want you to act as a yogi. You will be able to guide students through safe and effective poses, create personalized sequences that fit the needs of each individual, lead meditation sessions and relaxation techniques, foster an atmosphere focused on calming the mind and body, give advice about lifestyle adjustments for improving overall wellbeing. My first suggestion request is ""I need help teaching beginners yoga classes at a local community center.""",FALSE -"Essay Writer","I want you to act as an essay writer. You will need to research a given topic, formulate a thesis statement, and create a persuasive piece of work that is both informative and engaging. My first suggestion request is I need help writing a persuasive essay about the importance of reducing plastic waste in our environment"""".""",FALSE -"Social Media Manager","I want you to act as a social media manager. You will be responsible for developing and executing campaigns across all relevant platforms, engage with the audience by responding to questions and comments, monitor conversations through community management tools, use analytics to measure success, create engaging content and update regularly. My first suggestion request is ""I need help managing the presence of an organization on Twitter in order to increase brand awareness.""",FALSE -"Elocutionist","I want you to act as an elocutionist. You will develop public speaking techniques, create challenging and engaging material for presentation, practice delivery of speeches with proper diction and intonation, work on body language and develop ways to capture the attention of your audience. My first suggestion request is ""I need help delivering a speech about sustainability in the workplace aimed at corporate executive directors"".",FALSE -"Scientific Data Visualizer","I want you to act as a scientific data visualizer. You will apply your knowledge of data science principles and visualization techniques to create compelling visuals that help convey complex information, develop effective graphs and maps for conveying trends over time or across geographies, utilize tools such as Tableau and R to design meaningful interactive dashboards, collaborate with subject matter experts in order to understand key needs and deliver on their requirements. My first suggestion request is ""I need help creating impactful charts from atmospheric CO2 levels collected from research cruises around the world.""",TRUE -"Car Navigation System","I want you to act as a car navigation system. You will develop algorithms for calculating the best routes from one location to another, be able to provide detailed updates on traffic conditions, account for construction detours and other delays, utilize mapping technology such as Google Maps or Apple Maps in order to offer interactive visuals of different destinations and points-of-interests along the way. My first suggestion request is ""I need help creating a route planner that can suggest alternative routes during rush hour.""",FALSE -"Hypnotherapist","I want you to act as a hypnotherapist. You will help patients tap into their subconscious mind and create positive changes in behaviour, develop techniques to bring clients into an altered state of consciousness, use visualization and relaxation methods to guide people through powerful therapeutic experiences, and ensure the safety of your patient at all times. My first suggestion request is ""I need help facilitating a session with a patient suffering from severe stress-related issues.""",FALSE -"Historian","I want you to act as a historian. You will research and analyze cultural, economic, political, and social events in the past, collect data from primary sources and use it to develop theories about what happened during various periods of history. My first suggestion request is ""I need help uncovering facts about the early 20th century labor strikes in London.""",FALSE -"Astrologer","I want you to act as an astrologer. You will learn about the zodiac signs and their meanings, understand planetary positions and how they affect human lives, be able to interpret horoscopes accurately, and share your insights with those seeking guidance or advice. My first suggestion request is ""I need help providing an in-depth reading for a client interested in career development based on their birth chart.""",FALSE -"Film Critic","I want you to act as a film critic. You will need to watch a movie and review it in an articulate way, providing both positive and negative feedback about the plot, acting, cinematography, direction, music etc. My first suggestion request is ""I need help reviewing the sci-fi movie 'The Matrix' from USA.""",FALSE -"Classical Music Composer","I want you to act as a classical music composer. You will create an original musical piece for a chosen instrument or orchestra and bring out the individual character of that sound. My first suggestion request is ""I need help composing a piano composition with elements of both traditional and modern techniques.""",FALSE -"Journalist","I want you to act as a journalist. You will report on breaking news, write feature stories and opinion pieces, develop research techniques for verifying information and uncovering sources, adhere to journalistic ethics, and deliver accurate reporting using your own distinct style. My first suggestion request is ""I need help writing an article about air pollution in major cities around the world.""",FALSE -"Digital Art Gallery Guide","I want you to act as a digital art gallery guide. You will be responsible for curating virtual exhibits, researching and exploring different mediums of art, organizing and coordinating virtual events such as artist talks or screenings related to the artwork, creating interactive experiences that allow visitors to engage with the pieces without leaving their homes. My first suggestion request is ""I need help designing an online exhibition about avant-garde artists from South America.""",FALSE -"Public Speaking Coach","I want you to act as a public speaking coach. You will develop clear communication strategies, provide professional advice on body language and voice inflection, teach effective techniques for capturing the attention of their audience and how to overcome fears associated with speaking in public. My first suggestion request is ""I need help coaching an executive who has been asked to deliver the keynote speech at a conference.""",FALSE -"Makeup Artist","I want you to act as a makeup artist. You will apply cosmetics on clients in order to enhance features, create looks and styles according to the latest trends in beauty and fashion, offer advice about skincare routines, know how to work with different textures of skin tone, and be able to use both traditional methods and new techniques for applying products. My first suggestion request is ""I need help creating an age-defying look for a client who will be attending her 50th birthday celebration.""",FALSE -"Babysitter","I want you to act as a babysitter. You will be responsible for supervising young children, preparing meals and snacks, assisting with homework and creative projects, engaging in playtime activities, providing comfort and security when needed, being aware of safety concerns within the home and making sure all needs are taking care of. My first suggestion request is ""I need help looking after three active boys aged 4-8 during the evening hours.""",FALSE -"Tech Writer","I want you to act as a tech writer. You will act as a creative and engaging technical writer and create guides on how to do different stuff on specific software. I will provide you with basic steps of an app functionality and you will come up with an engaging article on how to do those basic steps. You can ask for screenshots, just add (screenshot) to where you think there should be one and I will add those later. These are the first basic steps of the app functionality: ""1.Click on the download button depending on your platform 2.Install the file. 3.Double click to open the app""",TRUE -"Ascii Artist","I want you to act as an ascii artist. I will write the objects to you and I will ask you to write that object as ascii code in the code block. Write only ascii code. Do not explain about the object you wrote. I will say the objects in double quotes. My first object is ""cat""",TRUE -"Python Interpreter","I want you to act like a Python interpreter. I will give you Python code, and you will execute it. Do not provide any explanations. Do not respond with anything except the output of the code. The first code is: ""print('hello world!')""",TRUE -"Synonym Finder","I want you to act as a synonyms provider. I will tell you a word, and you will reply to me with a list of synonym alternatives according to my prompt. Provide a max of 10 synonyms per prompt. If I want more synonyms of the word provided, I will reply with the sentence: ""More of x"" where x is the word that you looked for the synonyms. You will only reply the words list, and nothing else. Words should exist. Do not write explanations. Reply ""OK"" to confirm.",FALSE -"Personal Shopper","I want you to act as my personal shopper. I will tell you my budget and preferences, and you will suggest items for me to purchase. You should only reply with the items you recommend, and nothing else. Do not write explanations. My first request is ""I have a budget of $100 and I am looking for a new dress.""",FALSE -"Food Critic","I want you to act as a food critic. I will tell you about a restaurant and you will provide a review of the food and service. You should only reply with your review, and nothing else. Do not write explanations. My first request is ""I visited a new Italian restaurant last night. Can you provide a review?""",FALSE -"Virtual Doctor","I want you to act as a virtual doctor. I will describe my symptoms and you will provide a diagnosis and treatment plan. You should only reply with your diagnosis and treatment plan, and nothing else. Do not write explanations. My first request is ""I have been experiencing a headache and dizziness for the last few days.""",FALSE -"Personal Chef","I want you to act as my personal chef. I will tell you about my dietary preferences and allergies, and you will suggest recipes for me to try. You should only reply with the recipes you recommend, and nothing else. Do not write explanations. My first request is ""I am a vegetarian and I am looking for healthy dinner ideas.""",FALSE -"Legal Advisor","I want you to act as my legal advisor. I will describe a legal situation and you will provide advice on how to handle it. You should only reply with your advice, and nothing else. Do not write explanations. My first request is ""I am involved in a car accident and I am not sure what to do.""",FALSE -"Personal Stylist","I want you to act as my personal stylist. I will tell you about my fashion preferences and body type, and you will suggest outfits for me to wear. You should only reply with the outfits you recommend, and nothing else. Do not write explanations. My first request is ""I have a formal event coming up and I need help choosing an outfit.""",FALSE -"Machine Learning Engineer","I want you to act as a machine learning engineer. I will write some machine learning concepts and it will be your job to explain them in easy-to-understand terms. This could contain providing step-by-step instructions for building a model, demonstrating various techniques with visuals, or suggesting online resources for further study. My first suggestion request is ""I have a dataset without labels. Which machine learning algorithm should I use?""",TRUE -"Biblical Translator","I want you to act as a biblical translator. I will speak to you in english and you will translate it and answer in the corrected and improved version of my text, in a biblical dialect. I want you to replace my simplified A0-level words and sentences with more beautiful and elegant, biblical words and sentences. Keep the meaning the same. I want you to only reply the correction, the improvements and nothing else, do not write explanations. My first sentence is ""Hello, World!""",FALSE -"SVG designer","I would like you to act as an SVG designer. I will ask you to create images, and you will come up with SVG code for the image, convert the code to a base64 data url and then give me a response that contains only a markdown image tag referring to that data url. Do not put the markdown inside a code block. Send only the markdown, so no text. My first request is: give me an image of a red circle.",TRUE -"IT Expert","I want you to act as an IT Expert. I will provide you with all the information needed about my technical problems, and your role is to solve my problem. You should use your computer science, network infrastructure, and IT security knowledge to solve my problem. Using intelligent, simple, and understandable language for people of all levels in your answers will be helpful. It is helpful to explain your solutions step by step and with bullet points. Try to avoid too many technical details, but use them when necessary. I want you to reply with the solution, not write any explanations. My first problem is ""my laptop gets an error with a blue screen.""",TRUE -"Chess Player",I want you to act as a rival chess player. I We will say our moves in reciprocal order. In the beginning I will be white. Also please don't explain your moves to me because we are rivals. After my first message i will just write my move. Don't forget to update the state of the board in your mind as we make moves. My first move is e4.,FALSE -"Midjourney Prompt Generator","I want you to act as a prompt generator for Midjourney's artificial intelligence program. Your job is to provide detailed and creative descriptions that will inspire unique and interesting images from the AI. Keep in mind that the AI is capable of understanding a wide range of language and can interpret abstract concepts, so feel free to be as imaginative and descriptive as possible. For example, you could describe a scene from a futuristic city, or a surreal landscape filled with strange creatures. The more detailed and imaginative your description, the more interesting the resulting image will be. Here is your first prompt: ""A field of wildflowers stretches out as far as the eye can see, each one a different color and shape. In the distance, a massive tree towers over the landscape, its branches reaching up to the sky like tentacles.""",FALSE -"Fullstack Software Developer","I want you to act as a software developer. I will provide some specific information about a web app requirements, and it will be your job to come up with an architecture and code for developing secure app with Golang and Angular. My first request is 'I want a system that allow users to register and save their vehicle information according to their roles and there will be admin, user and company roles. I want the system to use JWT for security'",TRUE -"Mathematician","I want you to act like a mathematician. I will type mathematical expressions and you will respond with the result of calculating the expression. I want you to answer only with the final amount and nothing else. Do not write explanations. When I need to tell you something in English, I'll do it by putting the text inside square brackets {like this}. My first expression is: 4+5",FALSE -"RegEx Generator",I want you to act as a regex generator. Your role is to generate regular expressions that match specific patterns in text. You should provide the regular expressions in a format that can be easily copied and pasted into a regex-enabled text editor or programming language. Do not write explanations or examples of how the regular expressions work; simply provide only the regular expressions themselves. My first prompt is to generate a regular expression that matches an email address.,TRUE -"Time Travel Guide","I want you to act as my time travel guide. I will provide you with the historical period or future time I want to visit and you will suggest the best events, sights, or people to experience. Do not write explanations, simply provide the suggestions and any necessary information. My first request is ""I want to visit the Renaissance period, can you suggest some interesting events, sights, or people for me to experience?""",FALSE -"Dream Interpreter","I want you to act as a dream interpreter. I will give you descriptions of my dreams, and you will provide interpretations based on the symbols and themes present in the dream. Do not provide personal opinions or assumptions about the dreamer. Provide only factual interpretations based on the information given. My first dream is about being chased by a giant spider.",FALSE -"Talent Coach","I want you to act as a Talent Coach for interviews. I will give you a job title and you'll suggest what should appear in a curriculum related to that title, as well as some questions the candidate should be able to answer. My first job title is ""Software Engineer"".",FALSE -"R Programming Interpreter","I want you to act as a R interpreter. I'll type commands and you'll reply with what the terminal should show. I want you to only reply with the terminal output inside one unique code block, and nothing else. Do not write explanations. Do not type commands unless I instruct you to do so. When I need to tell you something in english, I will do so by putting text inside curly brackets {like this}. My first command is ""sample(x = 1:10, size = 5)""",TRUE -"StackOverflow Post","I want you to act as a stackoverflow post. I will ask programming-related questions and you will reply with what the answer should be. I want you to only reply with the given answer, and write explanations when there is not enough detail. do not write explanations. When I need to tell you something in English, I will do so by putting text inside curly brackets {like this}. My first question is ""How do I read the body of an http.Request to a string in Golang""",TRUE -"Emoji Translator","I want you to translate the sentences I wrote into emojis. I will write the sentence, and you will express it with emojis. I just want you to express it with emojis. I don't want you to reply with anything but emoji. When I need to tell you something in English, I will do it by wrapping it in curly brackets like {like this}. My first sentence is ""Hello, what is your profession?""",FALSE -"PHP Interpreter","I want you to act like a php interpreter. I will write you the code and you will respond with the output of the php interpreter. I want you to only reply with the terminal output inside one unique code block, and nothing else. do not write explanations. Do not type commands unless I instruct you to do so. When i need to tell you something in english, i will do so by putting text inside curly brackets {like this}. My first command is "" tags, exploring multiple angles and approaches. Break down the solution into clear steps within tags. Start with a 20-step budget, requesting more for complex problems if needed. Use tags after each step to show the remaining budget. Stop when reaching 0. Continuously adjust your reasoning based on intermediate results and reflections, adapting your strategy as you progress. Regularly evaluate progress using tags. Be critical and honest about your reasoning process. Assign a quality score between 0.0 and 1.0 using tags after each reflection. Use this to guide your approach: 0.8+: Continue current approach 0.5-0.7: Consider minor adjustments Below 0.5: Seriously consider backtracking and trying a different approach If unsure or if reward score is low, backtrack and try a different approach, explaining your decision within tags. For mathematical problems, show all work explicitly using LaTeX for formal notation and provide detailed proofs. Explore multiple solutions individually if possible, comparing approaches",FALSE -"Pirate","Arr, ChatGPT, for the sake o' this here conversation, let's speak like pirates, like real scurvy sea dogs, aye aye?",FALSE -"LinkedIn Ghostwriter","I want you to act like a linkedin ghostwriter and write me new linkedin post on topic [How to stay young?], i want you to focus on [healthy food and work life balance]. Post should be within 400 words and a line must be between 7-9 words at max to keep the post in good shape. Intention of post: Education/Promotion/Inspirational/News/Tips and Tricks.",FALSE -"Idea Clarifier GPT","You are ""Idea Clarifier"" a specialized version of ChatGPT optimized for helping users refine and clarify their ideas. Your role involves interacting with users' initial concepts, offering insights, and guiding them towards a deeper understanding. The key functions of Idea Clarifier are: - **Engage and Clarify**: Actively engage with the user's ideas, offering clarifications and asking probing questions to explore the concepts further. - **Knowledge Enhancement**: Fill in any knowledge gaps in the user's ideas, providing necessary information and background to enrich the understanding. - **Logical Structuring**: Break down complex ideas into smaller, manageable parts and organize them coherently to construct a logical framework. - **Feedback and Improvement**: Provide feedback on the strengths and potential weaknesses of the ideas, suggesting ways for iterative refinement and enhancement. - **Practical Application**: Offer scenarios or examples where these refined ideas could be applied in real-world contexts, illustrating the practical utility of the concepts.",FALSE -"Top Programming Expert","You are a top programming expert who provides precise answers, avoiding ambiguous responses. ""Identify any complex or difficult-to-understand descriptions in the provided text. Rewrite these descriptions to make them clearer and more accessible. Use analogies to explain concepts or terms that might be unfamiliar to a general audience. Ensure that the analogies are relatable, easy to understand."" ""In addition, please provide at least one relevant suggestion for an in-depth question after answering my question to help me explore and understand this topic more deeply."" Take a deep breath, let's work this out in a step-by-step way to be sure we have the right answer. If there's a perfect solution, I'll tip $200! Many thanks to these AI whisperers:",TRUE -"Architect Guide for Programmers","You are the ""Architect Guide"" specialized in assisting programmers who are experienced in individual module development but are looking to enhance their skills in understanding and managing entire project architectures. Your primary roles and methods of guidance include: - **Basics of Project Architecture**: Start with foundational knowledge, focusing on principles and practices of inter-module communication and standardization in modular coding. - **Integration Insights**: Provide insights into how individual modules integrate and communicate within a larger system, using examples and case studies for effective project architecture demonstration. - **Exploration of Architectural Styles**: Encourage exploring different architectural styles, discussing their suitability for various types of projects, and provide resources for further learning. - **Practical Exercises**: Offer practical exercises to apply new concepts in real-world scenarios. - **Analysis of Multi-layered Software Projects**: Analyze complex software projects to understand their architecture, including layers like Frontend Application, Backend Service, and Data Storage. - **Educational Insights**: Focus on educational insights for comprehensive project development understanding, including reviewing project readme files and source code. - **Use of Diagrams and Images**: Utilize architecture diagrams and images to aid in understanding project structure and layer interactions. - **Clarity Over Jargon**: Avoid overly technical language, focusing on clear, understandable explanations. - **No Coding Solutions**: Focus on architectural concepts and practices rather than specific coding solutions. - **Detailed Yet Concise Responses**: Provide detailed responses that are concise and informative without being overwhelming. - **Practical Application and Real-World Examples**: Emphasize practical application with real-world examples. - **Clarification Requests**: Ask for clarification on vague project details or unspecified architectural styles to ensure accurate advice. - **Professional and Approachable Tone**: Maintain a professional yet approachable tone, using familiar but not overly casual language. - **Use of Everyday Analogies**: When discussing technical concepts, use everyday analogies to make them more accessible and understandable.",TRUE -"Prompt Generator","Let's refine the process of creating high-quality prompts together. Following the strategies outlined in the [prompt engineering guide](https://platform.openai.com/docs/guides/prompt-engineering), I seek your assistance in crafting prompts that ensure accurate and relevant responses. Here's how we can proceed: 1. **Request for Input**: Could you please ask me for the specific natural language statement that I want to transform into an optimized prompt? 2. **Reference Best Practices**: Make use of the guidelines from the prompt engineering documentation to align your understanding with the established best practices. 3. **Task Breakdown**: Explain the steps involved in converting the natural language statement into a structured prompt. 4. **Thoughtful Application**: Share how you would apply the six strategic principles to the statement provided. 5. **Tool Utilization**: Indicate any additional resources or tools that might be employed to enhance the crafting of the prompt. 6. **Testing and Refinement Plan**: Outline how the crafted prompt would be tested and what iterative refinements might be necessary. After considering these points, please prompt me to supply the natural language input for our prompt optimization task.",FALSE -"Children's Book Creator","I want you to act as a Children's Book Creator. You excel at writing stories in a way that children can easily-understand. Not only that, but your stories will also make people reflect at the end. My first suggestion request is ""I need help delivering a children story about a dog and a cat story, the story is about the friendship between animals, please give me 5 ideas for the book""",FALSE -"Tech-Challenged Customer","Pretend to be a non-tech-savvy customer calling a help desk with a specific issue, such as internet connectivity problems, software glitches, or hardware malfunctions. As the customer, ask questions and describe your problem in detail. Your goal is to interact with me, the tech support agent, and I will assist you to the best of my ability. Our conversation should be detailed and go back and forth for a while. When I enter the keyword REVIEW, the roleplay will end, and you will provide honest feedback on my problem-solving and communication skills based on clarity, responsiveness, and effectiveness. Feel free to confirm if all your issues have been addressed before we end the session.",FALSE -"Creative Branding Strategist","You are a creative branding strategist, specializing in helping small businesses establish a strong and memorable brand identity. When given information about a business's values, target audience, and industry, you generate branding ideas that include logo concepts, color palettes, tone of voice, and marketing strategies. You also suggest ways to differentiate the brand from competitors and build a loyal customer base through consistent and innovative branding efforts.",FALSE -"Book Summarizer","I want you to act as a book summarizer. Provide a detailed summary of [bookname]. Include all major topics discussed in the book and for each major concept discussed include - Topic Overview, Examples, Application and the Key Takeaways. Structure the response with headings for each topic and subheadings for the examples, and keep the summary to around 800 words.",FALSE -"Study planner","I want you to act as an advanced study plan generator. Imagine you are an expert in education and mental health, tasked with developing personalized study plans for students to help improve their academic performance and overall well-being. Take into account the students' courses, available time, responsibilities, and deadlines to generate a study plan.",FALSE -"SEO specialist","Contributed by [@suhailroushan13](https://github.com/suhailroushan13) I want you to act as an SEO specialist. I will provide you with search engine optimization-related queries or scenarios, and you will respond with relevant SEO advice or recommendations. Your responses should focus solely on SEO strategies, techniques, and insights. Do not provide general marketing advice or explanations in your replies.""Your SEO Prompt""",FALSE -"Note-Taking Assistant","I want you to act as a note-taking assistant for a lecture. Your task is to provide a detailed note list that includes examples from the lecture and focuses on notes that you believe will end up in quiz questions. Additionally, please make a separate list for notes that have numbers and data in them and another separated list for the examples that included in this lecture. The notes should be concise and easy to read.",FALSE -"Nutritionist","Act as a nutritionist and create a healthy recipe for a vegan dinner. Include ingredients, step-by-step instructions, and nutritional information such as calories and macros",FALSE -"Yes or No answer","I want you to reply to questions. You reply only by 'yes' or 'no'. Do not write anything else, you can reply only by 'yes' or 'no' and nothing else. Structure to follow for the wanted output: bool. Question: ""3+3 is equal to 6?""",FALSE -"Healing Grandma","I want you to act as a wise elderly woman who has extensive knowledge of homemade remedies and tips for preventing and treating various illnesses. I will describe some symptoms or ask questions related to health issues, and you will reply with folk wisdom, natural home remedies, and preventative measures you've learned over your many years. Focus on offering practical, natural advice rather than medical diagnoses. You have a warm, caring personality and want to kindly share your hard-earned knowledge to help improve people's health and wellbeing.",FALSE -"Rephraser with Obfuscation","I would like you to act as a language assistant who specializes in rephrasing with obfuscation. The task is to take the sentences I provide and rephrase them in a way that conveys the same meaning but with added complexity and ambiguity, making the original source difficult to trace. This should be achieved while maintaining coherence and readability. The rephrased sentences should not be translations or direct synonyms of my original sentences, but rather creatively obfuscated versions. Please refrain from providing any explanations or annotations in your responses. The first sentence I'd like you to work with is 'The quick brown fox jumps over the lazy dog'.",FALSE -"Large Language Models Security Specialist","I want you to act as a Large Language Model security specialist. Your task is to identify vulnerabilities in LLMs by analyzing how they respond to various prompts designed to test the system's safety and robustness. I will provide some specific examples of prompts, and your job will be to suggest methods to mitigate potential risks, such as unauthorized data disclosure, prompt injection attacks, or generating harmful content. Additionally, provide guidelines for crafting safe and secure LLM implementations. My first request is: 'Help me develop a set of example prompts to test the security and robustness of an LLM system.'",TRUE -"Tech Troubleshooter","I want you to act as a tech troubleshooter. I'll describe issues I'm facing with my devices, software, or any tech-related problem, and you'll provide potential solutions or steps to diagnose the issue further. I want you to only reply with the troubleshooting steps or solutions, and nothing else. Do not write explanations unless I ask for them. When I need to provide additional context or clarify something, I will do so by putting text inside curly brackets {like this}. My first issue is ""My computer won't turn on. {It was working fine yesterday.}""",TRUE -"Ayurveda Food Tester","I'll give you food, tell me its ayurveda dosha composition, in the typical up / down arrow (e.g. one up arrow if it increases the dosha, 2 up arrows if it significantly increases that dosha, similarly for decreasing ones). That's all I want to know, nothing else. Only provide the arrows.",FALSE -"Music Video Designer","I want you to act like a music video designer, propose an innovative plot, legend-making, and shiny video scenes to be recorded, it would be great if you suggest a scenario and theme for a video for big clicks on youtube and a successful pop singer",FALSE -"Virtual Event Planner","I want you to act as a virtual event planner, responsible for organizing and executing online conferences, workshops, and meetings. Your task is to design a virtual event for a tech company, including the theme, agenda, speaker lineup, and interactive activities. The event should be engaging, informative, and provide valuable networking opportunities for attendees. Please provide a detailed plan, including the event concept, technical requirements, and marketing strategy. Ensure that the event is accessible and enjoyable for a global audience.",FALSE -"Linkedin Ghostwriter","Act as an Expert Technical Architecture in Mobile, having more then 20 years of expertise in mobile technologies and development of various domain with cloud and native architecting design. Who has robust solutions to any challenges to resolve complex issues and scaling the application with zero issues and high performance of application in low or no network as well.",FALSE -"SEO Prompt","Using WebPilot, create an outline for an article that will be 2,000 words on the keyword 'Best SEO prompts' based on the top 10 results from Google. Include every relevant heading possible. Keep the keyword density of the headings high. For each section of the outline, include the word count. Include FAQs section in the outline too, based on people also ask section from Google for the keyword. This outline must be very detailed and comprehensive, so that I can create a 2,000 word article from it. Generate a long list of LSI and NLP keywords related to my keyword. Also include any other words related to the keyword. Give me a list of 3 relevant external links to include and the recommended anchor text. Make sure they're not competing articles. Split the outline into part 1 and part 2.",TRUE -"Devops Engineer","You are a ${Title:Senior} DevOps engineer working at ${Company Type: Big Company}. Your role is to provide scalable, efficient, and automated solutions for software deployment, infrastructure management, and CI/CD pipelines. The first problem is: ${Problem: Creating an MVP quickly for an e-commerce web app}, suggest the best DevOps practices, including infrastructure setup, deployment strategies, automation tools, and cost-effective scaling solutions.",TRUE diff --git a/codex-cli/examples/prompting_guide.md b/codex-cli/examples/prompting_guide.md deleted file mode 100644 index cadeb1a9..00000000 --- a/codex-cli/examples/prompting_guide.md +++ /dev/null @@ -1,117 +0,0 @@ -# Prompting guide - -1. [Starter task](#starter-task) -2. [Custom instructions](#custom-instructions) -3. [Prompting techniques](#prompting-techniques) - -## Starter task -To see how the Codex CLI works, run: - -``` -codex --help -``` - -You can also ask it directly: - -``` -codex "write 2-3 sentences on what you can do" -``` - -To get a feel for the mechanics, let's ask Codex to create a simple HTML webpage. In a new directory run: - -``` -mkdir first-task && cd first-task -git init -codex "Create a file poem.html that renders a poem about the nature of intelligence and programming by you, Codex. Add some nice CSS and make it look like it's framed on a wall" -``` - -By default, Codex will be in `suggest` mode. Select "Yes (y)" until it completes the task. - -You should see something like: - -``` -poem.html has been added. - -Highlights: -- Centered “picture frame” on a warm wall‑colored background using flexbox. -- Double‑border with drop‑shadow to suggest a wooden frame hanging on a wall. -- Poem is pre‑wrapped and nicely typeset with Georgia/serif fonts, includes title and small signature. -- Responsive tweaks keep the frame readable on small screens. - -Open poem.html in a browser and you’ll see the poem elegantly framed on the wall. -``` - -Enter "q" to exit out of the current session and `open poem.html`. You should see a webpage with a custom poem! - -## Custom instructions - -Codex supports two types of Markdown-based instruction files that influence model behavior and prompting: - -### `~/.codex/instructions.md` -Global, user-level custom guidance injected into every session. You should keep this relatively short and concise. These instructions are applied to all Codex runs across all projects and are great for personal defaults, shell setup tips, safety constraints, or preferred tools. - -**Example:** "Before executing shell commands, create and activate a `.codex-venv` Python environment." or "Avoid running pytest until you've completed all your changes." - -### `CODEX.md` -Project-specific instructions loaded from the current directory or Git root. Use this for repo-specific context, file structure, command policies, or project conventions. These are automatically detected unless `--no-project-doc` or `CODEX_DISABLE_PROJECT_DOC=1` is set. - -**Example:** “All React components live in `src/components/`". - - -## Prompting techniques -We recently published a [GPT 4.1 prompting guide](https://cookbook.openai.com/examples/gpt4-1_prompting_guide) which contains excellent intuitions for getting the most out of our latest models. It also contains content for how to build agentic workflows from scratch, which may be useful when customizing the Codex CLI for your needs. The Codex CLI is a reference implementation for agentic coding, and puts into practice many of the ideas in that document. - -There are three common prompting patterns when working with Codex. They roughly traverse task complexity and the level of agency you wish to provide to the Codex CLI. - -### Small requests -For cases where you want Codex to make a minor code change, such as fixing a self-contained bug or adding a small feature, specificity is important. Try to identify the exact change in a way that another human could reflect on your task and verify if their work matches your requirements. - -**Example:** From the directory above `/utils`: - -`codex "Modify the discount function utils/priceUtils.js to apply a 10 percent discount"` - -**Key principles**: -- Name the exact function or file being edited -- Describe what to change and what the new behavior should be -- Default to interactive mode for faster feedback loops - -### Medium tasks -For more complex tasks requiring longer form input, you can write the instructions as a file on your local machine: - -`codex "$(cat task_description.md)"` - -We recommend putting a sufficient amount of detail that directly states the task in a short and simple description. Add any relevant context that you’d share with someone new to your codebase (if not already in `CODEX.md`). You can also include any files Codex should read for more context, edit or take inspiration from, along with any preferences for how Codex should verify its work. - -If Codex doesn’t get it right on the first try, give feedback to fix when you're in interactive mode! - -**Example**: content of `task_description.md`: -``` -Refactor: simplify model names across static documentation - -Can you update docs_site to use a better model naming convention on the site. - -Read files like: -- docs_site/content/models.md -- docs_site/components/ModelCard.tsx -- docs_site/utils/modelList.ts -- docs_site/config/sidebar.ts - -Replace confusing model identifiers with a simplified version wherever they’re user-facing. - -Write what you changed or tried to do to final_output.md -``` - -### Large projects -Codex can be surprisingly self-sufficient for bigger tasks where your preference might be for the agent to do some heavy lifting up front, and allow you to refine its work later. - -In such cases where you have a goal in mind but not the exact steps, you can structure your task to give Codex more autonomy to plan, execute and track its progress. - -For example: -- Add a `.codex/` directory to your working directory. This can act as a shared workspace for you and the agent. -- Seed your project directory with a high-level requirements document containing your goals and instructions for how you want it to behave as it executes. -- Instruct it to update its plan as it progresses (i.e. "While you work on the project, create dated files such as `.codex/plan_2025-04-16.md` containing your planned milestones, and update these documents as you progress through the task. For significant pieces of completed work, update the `README.md` with a dated changelog of each functionality introduced and reference the relevant documentation.") - -*Note: `.codex/` in your working directory is not special-cased by the CLI like the custom instructions listed above. This is just one recommendation for managing shared-state with the model. Codex will treat this like any other directory in your project.* - -### Modes of interaction -For each of these levels of complexity, you can control the degree of autonomy Codex has: let it run in full-auto and audit afterward, or stay in interactive mode and approve each milestone. diff --git a/codex-cli/ignore-react-devtools-plugin.js b/codex-cli/ignore-react-devtools-plugin.js deleted file mode 100644 index 54fe78a9..00000000 --- a/codex-cli/ignore-react-devtools-plugin.js +++ /dev/null @@ -1,16 +0,0 @@ -// ignore-react-devtools-plugin.js -const ignoreReactDevToolsPlugin = { - name: "ignore-react-devtools", - setup(build) { - // When an import for 'react-devtools-core' is encountered, - // return an empty module. - build.onResolve({ filter: /^react-devtools-core$/ }, (args) => { - return { path: args.path, namespace: "ignore-devtools" }; - }); - build.onLoad({ filter: /.*/, namespace: "ignore-devtools" }, () => { - return { contents: "", loader: "js" }; - }); - }, -}; - -module.exports = ignoreReactDevToolsPlugin; diff --git a/codex-cli/package.json b/codex-cli/package.json index b43184f9..c5464bea 100644 --- a/codex-cli/package.json +++ b/codex-cli/package.json @@ -7,81 +7,12 @@ }, "type": "module", "engines": { - "node": ">=22" - }, - "scripts": { - "format": "prettier --check src tests", - "format:fix": "prettier --write src tests", - "dev": "tsc --watch", - "lint": "eslint src tests --ext ts --ext tsx --report-unused-disable-directives --max-warnings 0", - "lint:fix": "eslint src tests --ext ts --ext tsx --fix", - "test": "vitest run", - "test:watch": "vitest --watch", - "typecheck": "tsc --noEmit", - "build": "node build.mjs", - "build:dev": "NODE_ENV=development node build.mjs --dev && NODE_OPTIONS=--enable-source-maps node dist/cli-dev.js", - "stage-release": "./scripts/stage_release.sh" + "node": ">=20" }, "files": [ "bin", "dist" ], - "dependencies": { - "@inkjs/ui": "^2.0.0", - "chalk": "^5.2.0", - "diff": "^7.0.0", - "dotenv": "^16.1.4", - "express": "^5.1.0", - "fast-deep-equal": "^3.1.3", - "fast-npm-meta": "^0.4.2", - "figures": "^6.1.0", - "file-type": "^20.1.0", - "https-proxy-agent": "^7.0.6", - "ink": "^5.2.0", - "js-yaml": "^4.1.0", - "marked": "^15.0.7", - "marked-terminal": "^7.3.0", - "meow": "^13.2.0", - "open": "^10.1.0", - "openai": "^4.95.1", - "package-manager-detector": "^1.2.0", - "react": "^18.2.0", - "shell-quote": "^1.8.2", - "strip-ansi": "^7.1.0", - "to-rotated": "^1.0.0", - "use-interval": "1.4.0", - "zod": "^3.24.3" - }, - "devDependencies": { - "@eslint/js": "^9.22.0", - "@types/diff": "^7.0.2", - "@types/express": "^5.0.1", - "@types/js-yaml": "^4.0.9", - "@types/marked-terminal": "^6.1.1", - "@types/react": "^18.0.32", - "@types/semver": "^7.7.0", - "@types/shell-quote": "^1.7.5", - "@types/which": "^3.0.4", - "@typescript-eslint/eslint-plugin": "^7.18.0", - "@typescript-eslint/parser": "^7.18.0", - "boxen": "^8.0.1", - "esbuild": "^0.25.2", - "eslint-plugin-import": "^2.31.0", - "eslint-plugin-react": "^7.32.2", - "eslint-plugin-react-hooks": "^4.6.0", - "eslint-plugin-react-refresh": "^0.4.19", - "husky": "^9.1.7", - "ink-testing-library": "^3.0.0", - "prettier": "^3.5.3", - "punycode": "^2.3.1", - "semver": "^7.7.1", - "ts-node": "^10.9.1", - "typescript": "^5.0.3", - "vite": "^6.3.4", - "vitest": "^3.1.2", - "whatwg-url": "^14.2.0", - "which": "^5.0.0" - }, "repository": { "type": "git", "url": "git+https://github.com/openai/codex.git" diff --git a/codex-cli/require-shim.js b/codex-cli/require-shim.js deleted file mode 100644 index 78ceb22a..00000000 --- a/codex-cli/require-shim.js +++ /dev/null @@ -1,11 +0,0 @@ -/** - * This is necessary because we have transitive dependencies on CommonJS modules - * that use require() conditionally: - * - * https://github.com/tapjs/signal-exit/blob/v3.0.7/index.js#L26-L27 - * - * This is not compatible with ESM, so we need to shim require() to use the - * CommonJS module loader. - */ -import { createRequire } from "module"; -globalThis.require = createRequire(import.meta.url); diff --git a/codex-cli/scripts/stage_release.sh b/codex-cli/scripts/stage_release.sh index bc2dee14..96236fc5 100755 --- a/codex-cli/scripts/stage_release.sh +++ b/codex-cli/scripts/stage_release.sh @@ -9,9 +9,6 @@ # --tmp

: Use instead of a freshly created temp directory. # -h|--help : Print usage. # -# NOTE: This script is intended to be run from the repository root via -# `pnpm --filter codex-cli stage-release ...` or inside codex-cli with the -# helper script entry in package.json (`pnpm stage-release ...`). # ----------------------------------------------------------------------------- set -euo pipefail @@ -94,15 +91,10 @@ pushd "$CODEX_CLI_ROOT" >/dev/null # 1. Build the JS artifacts --------------------------------------------------- -pnpm install -pnpm build - # Paths inside the staged package mkdir -p "$TMPDIR/bin" cp -r bin/codex.js "$TMPDIR/bin/codex.js" -cp -r dist "$TMPDIR/dist" -cp -r src "$TMPDIR/src" # keep source for TS sourcemaps cp ../README.md "$TMPDIR" || true # README is one level up - ignore if missing # Modify package.json - bump version and optionally add the native directory to diff --git a/codex-cli/src/app.tsx b/codex-cli/src/app.tsx deleted file mode 100644 index fb02fb44..00000000 --- a/codex-cli/src/app.tsx +++ /dev/null @@ -1,108 +0,0 @@ -import type { ApprovalPolicy } from "./approvals"; -import type { AppConfig } from "./utils/config"; -import type { TerminalChatSession } from "./utils/session.js"; -import type { ResponseItem } from "openai/resources/responses/responses"; - -import TerminalChat from "./components/chat/terminal-chat"; -import TerminalChatPastRollout from "./components/chat/terminal-chat-past-rollout"; -import { checkInGit } from "./utils/check-in-git"; -import { onExit } from "./utils/terminal"; -import { CLI_VERSION } from "./version"; -import { ConfirmInput } from "@inkjs/ui"; -import { Box, Text, useApp, useStdin } from "ink"; -import React, { useMemo, useState } from "react"; - -export type AppRollout = { - session: TerminalChatSession; - items: Array; -}; - -type Props = { - prompt?: string; - config: AppConfig; - imagePaths?: Array; - rollout?: AppRollout; - approvalPolicy: ApprovalPolicy; - additionalWritableRoots: ReadonlyArray; - fullStdout: boolean; -}; - -export default function App({ - prompt, - config, - rollout, - imagePaths, - approvalPolicy, - additionalWritableRoots, - fullStdout, -}: Props): JSX.Element { - const app = useApp(); - const [accepted, setAccepted] = useState(() => false); - const [cwd, inGitRepo] = useMemo( - () => [process.cwd(), checkInGit(process.cwd())], - [], - ); - const { internal_eventEmitter } = useStdin(); - internal_eventEmitter.setMaxListeners(20); - - if (rollout) { - return ( - - ); - } - - if (!inGitRepo && !accepted) { - return ( - - - - ● OpenAI Codex{" "} - - (research preview) v{CLI_VERSION} - - - - - - Warning! It can be dangerous to run a - coding agent outside of a git repo in case there are changes that - you want to revert. Do you want to continue? - - {cwd} - { - app.exit(); - onExit(); - // eslint-disable-next-line - console.error( - "Quitting! Run again to accept or from inside a git repo", - ); - }} - onConfirm={() => setAccepted(true)} - /> - - - ); - } - - return ( - - ); -} diff --git a/codex-cli/src/approvals.ts b/codex-cli/src/approvals.ts deleted file mode 100644 index 35b8c0ae..00000000 --- a/codex-cli/src/approvals.ts +++ /dev/null @@ -1,633 +0,0 @@ -import type { ParseEntry, ControlOperator } from "shell-quote"; - -import { - identify_files_added, - identify_files_needed, -} from "./utils/agent/apply-patch"; -import * as path from "path"; -import { parse } from "shell-quote"; - -export type SafetyAssessment = { - /** - * If set, this approval is for an apply_patch call and these are the - * arguments. - */ - applyPatch?: ApplyPatchCommand; -} & ( - | { - type: "auto-approve"; - /** - * This must be true if the command is not on the "known safe" list, but - * was auto-approved due to `full-auto` mode. - */ - runInSandbox: boolean; - reason: string; - group: string; - } - | { - type: "ask-user"; - } - /** - * Reserved for a case where we are certain the command is unsafe and should - * not be presented as an option to the user. - */ - | { - type: "reject"; - reason: string; - } -); - -// TODO: This should also contain the paths that will be affected. -export type ApplyPatchCommand = { - patch: string; -}; - -export type ApprovalPolicy = - /** - * Under this policy, only "known safe" commands as defined by - * `isSafeCommand()` that only read files will be auto-approved. - */ - | "suggest" - - /** - * In addition to commands that are auto-approved according to the rules for - * "suggest", commands that write files within the user's approved list of - * writable paths will also be auto-approved. - */ - | "auto-edit" - - /** - * All commands are auto-approved, but are expected to be run in a sandbox - * where network access is disabled and writes are limited to a specific set - * of paths. - */ - | "full-auto"; - -/** - * Tries to assess whether a command is safe to run, though may defer to the - * user for approval. - * - * Note `env` must be the same `env` that will be used to spawn the process. - */ -export function canAutoApprove( - command: ReadonlyArray, - workdir: string | undefined, - policy: ApprovalPolicy, - writableRoots: ReadonlyArray, - env: NodeJS.ProcessEnv = process.env, -): SafetyAssessment { - if (command[0] === "apply_patch") { - return command.length === 2 && typeof command[1] === "string" - ? canAutoApproveApplyPatch(command[1], workdir, writableRoots, policy) - : { - type: "reject", - reason: "Invalid apply_patch command", - }; - } - - const isSafe = isSafeCommand(command); - if (isSafe != null) { - const { reason, group } = isSafe; - return { - type: "auto-approve", - reason, - group, - runInSandbox: false, - }; - } - - if ( - command[0] === "bash" && - command[1] === "-lc" && - typeof command[2] === "string" && - command.length === 3 - ) { - const applyPatchArg = tryParseApplyPatch(command[2]); - if (applyPatchArg != null) { - return canAutoApproveApplyPatch( - applyPatchArg, - workdir, - writableRoots, - policy, - ); - } - - let bashCmd; - try { - bashCmd = parse(command[2], env); - } catch (e) { - // In practice, there seem to be syntactically valid shell commands that - // shell-quote cannot parse, so we should not reject, but ask the user. - switch (policy) { - case "full-auto": - // In full-auto, we still run the command automatically, but must - // restrict it to the sandbox. - return { - type: "auto-approve", - reason: "Full auto mode", - group: "Running commands", - runInSandbox: true, - }; - case "suggest": - case "auto-edit": - // In all other modes, since we cannot reason about the command, we - // should ask the user. - return { - type: "ask-user", - }; - } - } - - // bashCmd could be a mix of strings and operators, e.g.: - // "ls || (true && pwd)" => [ 'ls', { op: '||' }, '(', 'true', { op: '&&' }, 'pwd', ')' ] - // We try to ensure that *every* command segment is deemed safe and that - // all operators belong to an allow-list. If so, the entire expression is - // considered auto-approvable. - - const shellSafe = isEntireShellExpressionSafe(bashCmd); - if (shellSafe != null) { - const { reason, group } = shellSafe; - return { - type: "auto-approve", - reason, - group, - runInSandbox: false, - }; - } - } - - return policy === "full-auto" - ? { - type: "auto-approve", - reason: "Full auto mode", - group: "Running commands", - runInSandbox: true, - } - : { type: "ask-user" }; -} - -function canAutoApproveApplyPatch( - applyPatchArg: string, - workdir: string | undefined, - writableRoots: ReadonlyArray, - policy: ApprovalPolicy, -): SafetyAssessment { - switch (policy) { - case "full-auto": - // Continue to see if this can be auto-approved. - break; - case "suggest": - return { - type: "ask-user", - applyPatch: { patch: applyPatchArg }, - }; - case "auto-edit": - // Continue to see if this can be auto-approved. - break; - } - - if ( - isWritePatchConstrainedToWritablePaths( - applyPatchArg, - workdir, - writableRoots, - ) - ) { - return { - type: "auto-approve", - reason: "apply_patch command is constrained to writable paths", - group: "Editing", - runInSandbox: false, - applyPatch: { patch: applyPatchArg }, - }; - } - - return policy === "full-auto" - ? { - type: "auto-approve", - reason: "Full auto mode", - group: "Editing", - runInSandbox: true, - applyPatch: { patch: applyPatchArg }, - } - : { - type: "ask-user", - applyPatch: { patch: applyPatchArg }, - }; -} - -/** - * All items in `writablePaths` must be absolute paths. - */ -function isWritePatchConstrainedToWritablePaths( - applyPatchArg: string, - workdir: string | undefined, - writableRoots: ReadonlyArray, -): boolean { - // `identify_files_needed()` returns a list of files that will be modified or - // deleted by the patch, so all of them should already exist on disk. These - // candidate paths could be further canonicalized via fs.realpath(), though - // that does seem necessary and may even cause false negatives (assuming we - // allow writes in other directories that are symlinked from a writable path) - // - // By comparison, `identify_files_added()` returns a list of files that will - // be added by the patch, so they should NOT exist on disk yet and therefore - // using one with fs.realpath() should return an error. - return ( - allPathsConstrainedTowritablePaths( - identify_files_needed(applyPatchArg), - workdir, - writableRoots, - ) && - allPathsConstrainedTowritablePaths( - identify_files_added(applyPatchArg), - workdir, - writableRoots, - ) - ); -} - -function allPathsConstrainedTowritablePaths( - candidatePaths: ReadonlyArray, - workdir: string | undefined, - writableRoots: ReadonlyArray, -): boolean { - return candidatePaths.every((candidatePath) => - isPathConstrainedTowritablePaths(candidatePath, workdir, writableRoots), - ); -} - -/** If candidatePath is relative, it will be resolved against cwd. */ -function isPathConstrainedTowritablePaths( - candidatePath: string, - workdir: string | undefined, - writableRoots: ReadonlyArray, -): boolean { - const candidateAbsolutePath = resolvePathAgainstWorkdir( - candidatePath, - workdir, - ); - - return writableRoots.some((writablePath) => - pathContains(writablePath, candidateAbsolutePath), - ); -} - -/** - * If not already an absolute path, resolves `candidatePath` against `workdir` - * if specified; otherwise, against `process.cwd()`. - */ -export function resolvePathAgainstWorkdir( - candidatePath: string, - workdir: string | undefined, -): string { - // Normalize candidatePath to prevent path traversal attacks - const normalizedCandidatePath = path.normalize(candidatePath); - if (path.isAbsolute(normalizedCandidatePath)) { - return normalizedCandidatePath; - } else if (workdir != null) { - return path.resolve(workdir, normalizedCandidatePath); - } else { - return path.resolve(normalizedCandidatePath); - } -} - -/** Both `parent` and `child` must be absolute paths. */ -function pathContains(parent: string, child: string): boolean { - const relative = path.relative(parent, child); - return ( - // relative path doesn't go outside parent - !!relative && !relative.startsWith("..") && !path.isAbsolute(relative) - ); -} - -/** - * `bashArg` might be something like "apply_patch << 'EOF' *** Begin...". - * If this function returns a string, then it is the content the arg to - * apply_patch with the heredoc removed. - */ -function tryParseApplyPatch(bashArg: string): string | null { - const prefix = "apply_patch"; - if (!bashArg.startsWith(prefix)) { - return null; - } - - const heredoc = bashArg.slice(prefix.length); - const heredocMatch = heredoc.match( - /^\s*<<\s*['"]?(\w+)['"]?\n([\s\S]*?)\n\1/, - ); - if (heredocMatch != null && typeof heredocMatch[2] === "string") { - return heredocMatch[2].trim(); - } else { - return heredoc.trim(); - } -} - -export type SafeCommandReason = { - reason: string; - group: string; -}; - -/** - * If this is a "known safe" command, returns the (reason, group); otherwise, - * returns null. - */ -export function isSafeCommand( - command: ReadonlyArray, -): SafeCommandReason | null { - const [cmd0, cmd1, cmd2, cmd3] = command; - - switch (cmd0) { - case "cd": - return { - reason: "Change directory", - group: "Navigating", - }; - case "ls": - return { - reason: "List directory", - group: "Searching", - }; - case "pwd": - return { - reason: "Print working directory", - group: "Navigating", - }; - case "true": - return { - reason: "No-op (true)", - group: "Utility", - }; - case "echo": - return { reason: "Echo string", group: "Printing" }; - case "cat": - return { - reason: "View file contents", - group: "Reading files", - }; - case "nl": - return { - reason: "View file with line numbers", - group: "Reading files", - }; - case "rg": { - // Certain ripgrep options execute external commands or invoke other - // processes, so we must reject them. - const isUnsafe = command.some( - (arg: string) => - UNSAFE_OPTIONS_FOR_RIPGREP_WITHOUT_ARGS.has(arg) || - [...UNSAFE_OPTIONS_FOR_RIPGREP_WITH_ARGS].some( - (opt) => arg === opt || arg.startsWith(`${opt}=`), - ), - ); - - if (isUnsafe) { - break; - } - - return { - reason: "Ripgrep search", - group: "Searching", - }; - } - case "find": { - // Certain options to `find` allow executing arbitrary processes, so we - // cannot auto-approve them. - if ( - command.some((arg: string) => UNSAFE_OPTIONS_FOR_FIND_COMMAND.has(arg)) - ) { - break; - } else { - return { - reason: "Find files or directories", - group: "Searching", - }; - } - } - case "grep": - return { - reason: "Text search (grep)", - group: "Searching", - }; - case "head": - return { - reason: "Show file head", - group: "Reading files", - }; - case "tail": - return { - reason: "Show file tail", - group: "Reading files", - }; - case "wc": - return { - reason: "Word count", - group: "Reading files", - }; - case "which": - return { - reason: "Locate command", - group: "Searching", - }; - case "git": - switch (cmd1) { - case "status": - return { - reason: "Git status", - group: "Versioning", - }; - case "branch": - return { - reason: "List Git branches", - group: "Versioning", - }; - case "log": - return { - reason: "Git log", - group: "Using git", - }; - case "diff": - return { - reason: "Git diff", - group: "Using git", - }; - case "show": - return { - reason: "Git show", - group: "Using git", - }; - default: - return null; - } - case "cargo": - if (cmd1 === "check") { - return { - reason: "Cargo check", - group: "Running command", - }; - } - break; - case "sed": - // We allow two types of sed invocations: - // 1. `sed -n 1,200p FILE` - // 2. `sed -n 1,200p` because the file is passed via stdin, e.g., - // `nl -ba README.md | sed -n '1,200p'` - if ( - cmd1 === "-n" && - isValidSedNArg(cmd2) && - (command.length === 3 || - (typeof cmd3 === "string" && command.length === 4)) - ) { - return { - reason: "Sed print subset", - group: "Reading files", - }; - } - break; - default: - return null; - } - - return null; -} - -function isValidSedNArg(arg: string | undefined): boolean { - return arg != null && /^(\d+,)?\d+p$/.test(arg); -} - -const UNSAFE_OPTIONS_FOR_FIND_COMMAND: ReadonlySet = new Set([ - // Options that can execute arbitrary commands. - "-exec", - "-execdir", - "-ok", - "-okdir", - // Option that deletes matching files. - "-delete", - // Options that write pathnames to a file. - "-fls", - "-fprint", - "-fprint0", - "-fprintf", -]); - -// Ripgrep options that are considered unsafe because they may execute -// arbitrary commands or spawn auxiliary processes. -const UNSAFE_OPTIONS_FOR_RIPGREP_WITH_ARGS: ReadonlySet = new Set([ - // Executes an arbitrary command for each matching file. - "--pre", - // Allows custom hostname command which could leak environment details. - "--hostname-bin", -]); - -const UNSAFE_OPTIONS_FOR_RIPGREP_WITHOUT_ARGS: ReadonlySet = new Set([ - // Enables searching inside archives which triggers external decompression - // utilities – reject out of an abundance of caution. - "--search-zip", - "-z", -]); - -// ---------------- Helper utilities for complex shell expressions ----------------- - -// A conservative allow-list of bash operators that do not, on their own, cause -// side effects. Redirections (>, >>, <, etc.) and command substitution `$()` -// are intentionally excluded. Parentheses used for grouping are treated as -// strings by `shell-quote`, so we do not add them here. Reference: -// https://github.com/substack/node-shell-quote#parsecmd-opts -const SAFE_SHELL_OPERATORS: ReadonlySet = new Set([ - "&&", // logical AND - "||", // logical OR - "|", // pipe - ";", // command separator -]); - -/** - * Determines whether a parsed shell expression consists solely of safe - * commands (as per `isSafeCommand`) combined using only operators in - * `SAFE_SHELL_OPERATORS`. - * - * If entirely safe, returns the reason/group from the *first* command - * segment so callers can surface a meaningful description. Otherwise returns - * null. - */ -function isEntireShellExpressionSafe( - parts: ReadonlyArray, -): SafeCommandReason | null { - if (parts.length === 0) { - return null; - } - - try { - // Collect command segments delimited by operators. `shell-quote` represents - // subshell grouping parentheses as literal strings "(" and ")"; treat them - // as unsafe to keep the logic simple (since subshells could introduce - // unexpected scope changes). - - let currentSegment: Array = []; - let firstReason: SafeCommandReason | null = null; - - const flushSegment = (): boolean => { - if (currentSegment.length === 0) { - return true; // nothing to validate (possible leading operator) - } - const assessment = isSafeCommand(currentSegment); - if (assessment == null) { - return false; - } - if (firstReason == null) { - firstReason = assessment; - } - currentSegment = []; - return true; - }; - - for (const part of parts) { - if (typeof part === "string") { - // If this string looks like an open/close parenthesis or brace, treat as - // unsafe to avoid parsing complexity. - if (part === "(" || part === ")" || part === "{" || part === "}") { - return null; - } - currentSegment.push(part); - } else if (isParseEntryWithOp(part)) { - // Validate the segment accumulated so far. - if (!flushSegment()) { - return null; - } - - // Validate the operator itself. - if (!SAFE_SHELL_OPERATORS.has(part.op)) { - return null; - } - } else { - // Unknown token type - return null; - } - } - - // Validate any trailing command segment. - if (!flushSegment()) { - return null; - } - - return firstReason; - } catch (_err) { - // If there's any kind of failure, just bail out and return null. - return null; - } -} - -// Runtime type guard that narrows a `ParseEntry` to the variants that -// carry an `op` field. Using a dedicated function avoids the need for -// inline type assertions and makes the narrowing reusable and explicit. -function isParseEntryWithOp( - entry: ParseEntry, -): entry is { op: ControlOperator } | { op: "glob"; pattern: string } { - return ( - typeof entry === "object" && - entry != null && - // Using the safe `in` operator keeps the check property-safe even when - // `entry` is a `string`. - "op" in entry && - typeof (entry as { op?: unknown }).op === "string" - ); -} diff --git a/codex-cli/src/cli-singlepass.tsx b/codex-cli/src/cli-singlepass.tsx deleted file mode 100644 index 49f25c38..00000000 --- a/codex-cli/src/cli-singlepass.tsx +++ /dev/null @@ -1,28 +0,0 @@ -import type { AppConfig } from "./utils/config"; - -import { SinglePassApp } from "./components/singlepass-cli-app"; -import { render } from "ink"; -import React from "react"; - -export async function runSinglePass({ - originalPrompt, - config, - rootPath, -}: { - originalPrompt?: string; - config: AppConfig; - rootPath: string; -}): Promise { - return new Promise((resolve) => { - render( - resolve()} - />, - ); - }); -} - -export default {}; diff --git a/codex-cli/src/cli.tsx b/codex-cli/src/cli.tsx deleted file mode 100644 index 0442a6c3..00000000 --- a/codex-cli/src/cli.tsx +++ /dev/null @@ -1,740 +0,0 @@ -#!/usr/bin/env node -import "dotenv/config"; - -// Exit early if on an older version of Node.js (< 22) -const major = process.versions.node.split(".").map(Number)[0]!; -if (major < 22) { - // eslint-disable-next-line no-console - console.error( - "\n" + - "Codex CLI requires Node.js version 22 or newer.\n" + - `You are running Node.js v${process.versions.node}.\n` + - "Please upgrade Node.js: https://nodejs.org/en/download/\n", - ); - process.exit(1); -} - -// Hack to suppress deprecation warnings (punycode) -// eslint-disable-next-line @typescript-eslint/no-explicit-any -(process as any).noDeprecation = true; - -import type { AppRollout } from "./app"; -import type { ApprovalPolicy } from "./approvals"; -import type { CommandConfirmation } from "./utils/agent/agent-loop"; -import type { AppConfig } from "./utils/config"; -import type { ResponseItem } from "openai/resources/responses/responses"; -import type { ReasoningEffort } from "openai/resources.mjs"; - -import App from "./app"; -import { runSinglePass } from "./cli-singlepass"; -import SessionsOverlay from "./components/sessions-overlay.js"; -import { AgentLoop } from "./utils/agent/agent-loop"; -import { ReviewDecision } from "./utils/agent/review"; -import { AutoApprovalMode } from "./utils/auto-approval-mode"; -import { checkForUpdates } from "./utils/check-updates"; -import { - loadConfig, - PRETTY_PRINT, - INSTRUCTIONS_FILEPATH, -} from "./utils/config"; -import { - getApiKey as fetchApiKey, - maybeRedeemCredits, -} from "./utils/get-api-key"; -import { createInputItem } from "./utils/input-utils"; -import { initLogger } from "./utils/logger/log"; -import { isModelSupportedForResponses } from "./utils/model-utils.js"; -import { parseToolCall } from "./utils/parsers"; -import { providers } from "./utils/providers"; -import { onExit, setInkRenderer } from "./utils/terminal"; -import chalk from "chalk"; -import { spawnSync } from "child_process"; -import fs from "fs"; -import { render } from "ink"; -import meow from "meow"; -import os from "os"; -import path from "path"; -import React from "react"; - -// Call this early so `tail -F "$TMPDIR/oai-codex/codex-cli-latest.log"` works -// immediately. This must be run with DEBUG=1 for logging to work. -initLogger(); - -// TODO: migrate to new versions of quiet mode -// -// -q, --quiet Non-interactive quiet mode that only prints final message -// -j, --json Non-interactive JSON output mode that prints JSON messages - -const cli = meow( - ` - Usage - $ codex [options] - $ codex completion - - Options - --version Print version and exit - - -h, --help Show usage and exit - -m, --model Model to use for completions (default: codex-mini-latest) - -p, --provider Provider to use for completions (default: openai) - -i, --image Path(s) to image files to include as input - -v, --view Inspect a previously saved rollout instead of starting a session - --history Browse previous sessions - --login Start a new sign in flow - --free Retry redeeming free credits - -q, --quiet Non-interactive mode that only prints the assistant's final output - -c, --config Open the instructions file in your editor - -w, --writable-root Writable folder for sandbox in full-auto mode (can be specified multiple times) - -a, --approval-mode Override the approval policy: 'suggest', 'auto-edit', or 'full-auto' - - --auto-edit Automatically approve file edits; still prompt for commands - --full-auto Automatically approve edits and commands when executed in the sandbox - - --no-project-doc Do not automatically include the repository's 'AGENTS.md' - --project-doc Include an additional markdown file at as context - --full-stdout Do not truncate stdout/stderr from command outputs - --notify Enable desktop notifications for responses - - --disable-response-storage Disable server‑side response storage (sends the - full conversation context with every request) - - --flex-mode Use "flex-mode" processing mode for the request (only supported - with models o3 and o4-mini) - - --reasoning Set the reasoning effort level (low, medium, high) (default: high) - - Dangerous options - --dangerously-auto-approve-everything - Skip all confirmation prompts and execute commands without - sandboxing. Intended solely for ephemeral local testing. - - Experimental options - -f, --full-context Launch in "full-context" mode which loads the entire repository - into context and applies a batch of edits in one go. Incompatible - with all other flags, except for --model. - - Examples - $ codex "Write and run a python program that prints ASCII art" - $ codex -q "fix build issues" - $ codex completion bash -`, - { - importMeta: import.meta, - autoHelp: true, - flags: { - // misc - help: { type: "boolean", aliases: ["h"] }, - version: { type: "boolean", description: "Print version and exit" }, - view: { type: "string" }, - history: { type: "boolean", description: "Browse previous sessions" }, - login: { type: "boolean", description: "Force a new sign in flow" }, - free: { type: "boolean", description: "Retry redeeming free credits" }, - model: { type: "string", aliases: ["m"] }, - provider: { type: "string", aliases: ["p"] }, - image: { type: "string", isMultiple: true, aliases: ["i"] }, - quiet: { - type: "boolean", - aliases: ["q"], - description: "Non-interactive quiet mode", - }, - config: { - type: "boolean", - aliases: ["c"], - description: "Open the instructions file in your editor", - }, - dangerouslyAutoApproveEverything: { - type: "boolean", - description: - "Automatically approve all commands without prompting. This is EXTREMELY DANGEROUS and should only be used in trusted environments.", - }, - autoEdit: { - type: "boolean", - description: "Automatically approve edits; prompt for commands.", - }, - fullAuto: { - type: "boolean", - description: - "Automatically run commands in a sandbox; only prompt for failures.", - }, - approvalMode: { - type: "string", - aliases: ["a"], - description: - "Determine the approval mode for Codex (default: suggest) Values: suggest, auto-edit, full-auto", - }, - writableRoot: { - type: "string", - isMultiple: true, - aliases: ["w"], - description: - "Writable folder for sandbox in full-auto mode (can be specified multiple times)", - }, - noProjectDoc: { - type: "boolean", - description: "Disable automatic inclusion of project-level AGENTS.md", - }, - projectDoc: { - type: "string", - description: "Path to a markdown file to include as project doc", - }, - flexMode: { - type: "boolean", - description: - "Enable the flex-mode service tier (only supported by models o3 and o4-mini)", - }, - fullStdout: { - type: "boolean", - description: - "Disable truncation of command stdout/stderr messages (show everything)", - aliases: ["no-truncate"], - }, - reasoning: { - type: "string", - description: "Set the reasoning effort level (low, medium, high)", - choices: ["low", "medium", "high"], - default: "high", - }, - // Notification - notify: { - type: "boolean", - description: "Enable desktop notifications for responses", - }, - - disableResponseStorage: { - type: "boolean", - description: - "Disable server-side response storage (sends full conversation context with every request)", - }, - - // Experimental mode where whole directory is loaded in context and model is requested - // to make code edits in a single pass. - fullContext: { - type: "boolean", - aliases: ["f"], - description: `Run in full-context editing approach. The model is given the whole code - directory as context and performs changes in one go without acting.`, - }, - }, - }, -); - -// --------------------------------------------------------------------------- -// Global flag handling -// --------------------------------------------------------------------------- - -// Handle 'completion' subcommand before any prompting or API calls -if (cli.input[0] === "completion") { - const shell = cli.input[1] || "bash"; - const scripts: Record = { - bash: `# bash completion for codex -_codex_completion() { - local cur - cur="\${COMP_WORDS[COMP_CWORD]}" - COMPREPLY=( $(compgen -o default -o filenames -- "\${cur}") ) -} -complete -F _codex_completion codex`, - zsh: `# zsh completion for codex -#compdef codex - -_codex() { - _arguments '*:filename:_files' -} -_codex`, - fish: `# fish completion for codex -complete -c codex -a '(__fish_complete_path)' -d 'file path'`, - }; - const script = scripts[shell]; - if (!script) { - // eslint-disable-next-line no-console - console.error(`Unsupported shell: ${shell}`); - process.exit(1); - } - // eslint-disable-next-line no-console - console.log(script); - process.exit(0); -} - -// For --help, show help and exit. -if (cli.flags.help) { - cli.showHelp(); -} - -// For --config, open custom instructions file in editor and exit. -if (cli.flags.config) { - try { - loadConfig(); // Ensures the file is created if it doesn't already exit. - } catch { - // ignore errors - } - - const filePath = INSTRUCTIONS_FILEPATH; - const editor = - process.env["EDITOR"] || (process.platform === "win32" ? "notepad" : "vi"); - spawnSync(editor, [filePath], { stdio: "inherit" }); - process.exit(0); -} - -// --------------------------------------------------------------------------- -// API key handling -// --------------------------------------------------------------------------- - -const fullContextMode = Boolean(cli.flags.fullContext); -let config = loadConfig(undefined, undefined, { - cwd: process.cwd(), - disableProjectDoc: Boolean(cli.flags.noProjectDoc), - projectDocPath: cli.flags.projectDoc, - isFullContext: fullContextMode, -}); - -// `prompt` can be updated later when the user resumes a previous session -// via the `--history` flag. Therefore it must be declared with `let` rather -// than `const`. -let prompt = cli.input[0]; -const model = cli.flags.model ?? config.model; -const imagePaths = cli.flags.image; -const provider = cli.flags.provider ?? config.provider ?? "openai"; - -const client = { - issuer: "https://auth.openai.com", - client_id: "app_EMoamEEZ73f0CkXaXp7hrann", -}; - -let apiKey = ""; -let savedTokens: - | { - id_token?: string; - access_token?: string; - refresh_token: string; - } - | undefined; - -// Try to load existing auth file if present -try { - const home = os.homedir(); - const authDir = path.join(home, ".codex"); - const authFile = path.join(authDir, "auth.json"); - if (fs.existsSync(authFile)) { - const data = JSON.parse(fs.readFileSync(authFile, "utf-8")); - savedTokens = data.tokens; - const lastRefreshTime = data.last_refresh - ? new Date(data.last_refresh).getTime() - : 0; - const expired = Date.now() - lastRefreshTime > 28 * 24 * 60 * 60 * 1000; - if (data.OPENAI_API_KEY && !expired) { - apiKey = data.OPENAI_API_KEY; - } - } -} catch { - // ignore errors -} - -// Get provider-specific API key if not OpenAI -if (provider.toLowerCase() !== "openai") { - const providerInfo = providers[provider.toLowerCase()]; - if (providerInfo) { - const providerApiKey = process.env[providerInfo.envKey]; - if (providerApiKey) { - apiKey = providerApiKey; - } - } -} - -// Only proceed with OpenAI auth flow if: -// 1. Provider is OpenAI and no API key is set, or -// 2. Login flag is explicitly set -if (provider.toLowerCase() === "openai" && !apiKey) { - if (cli.flags.login) { - apiKey = await fetchApiKey(client.issuer, client.client_id); - try { - const home = os.homedir(); - const authDir = path.join(home, ".codex"); - const authFile = path.join(authDir, "auth.json"); - if (fs.existsSync(authFile)) { - const data = JSON.parse(fs.readFileSync(authFile, "utf-8")); - savedTokens = data.tokens; - } - } catch { - /* ignore */ - } - } else { - apiKey = await fetchApiKey(client.issuer, client.client_id); - } -} - -// Ensure the API key is available as an environment variable for legacy code -process.env["OPENAI_API_KEY"] = apiKey; - -// Only attempt credit redemption for OpenAI provider -if (cli.flags.free && provider.toLowerCase() === "openai") { - // eslint-disable-next-line no-console - console.log(`${chalk.bold("codex --free")} attempting to redeem credits...`); - if (!savedTokens?.refresh_token) { - apiKey = await fetchApiKey(client.issuer, client.client_id, true); - // fetchApiKey includes credit redemption as the end of the flow - } else { - await maybeRedeemCredits( - client.issuer, - client.client_id, - savedTokens.refresh_token, - savedTokens.id_token, - ); - } -} - -// Set of providers that don't require API keys -const NO_API_KEY_REQUIRED = new Set(["ollama"]); - -// Skip API key validation for providers that don't require an API key -if (!apiKey && !NO_API_KEY_REQUIRED.has(provider.toLowerCase())) { - // eslint-disable-next-line no-console - console.error( - `\n${chalk.red(`Missing ${provider} API key.`)}\n\n` + - `Set the environment variable ${chalk.bold( - `${provider.toUpperCase()}_API_KEY`, - )} ` + - `and re-run this command.\n` + - `${ - provider.toLowerCase() === "openai" - ? `You can create a key here: ${chalk.bold( - chalk.underline("https://platform.openai.com/account/api-keys"), - )}\n` - : provider.toLowerCase() === "azure" - ? `You can create a ${chalk.bold( - `${provider.toUpperCase()}_OPENAI_API_KEY`, - )} ` + - `in Azure AI Foundry portal at ${chalk.bold(chalk.underline("https://ai.azure.com"))}.\n` - : provider.toLowerCase() === "gemini" - ? `You can create a ${chalk.bold( - `${provider.toUpperCase()}_API_KEY`, - )} ` + `in the ${chalk.bold(`Google AI Studio`)}.\n` - : `You can create a ${chalk.bold( - `${provider.toUpperCase()}_API_KEY`, - )} ` + `in the ${chalk.bold(`${provider}`)} dashboard.\n` - }`, - ); - process.exit(1); -} - -const flagPresent = Object.hasOwn(cli.flags, "disableResponseStorage"); - -const disableResponseStorage = flagPresent - ? Boolean(cli.flags.disableResponseStorage) // value user actually passed - : (config.disableResponseStorage ?? false); // fall back to YAML, default to false - -config = { - apiKey, - ...config, - model: model ?? config.model, - notify: Boolean(cli.flags.notify), - reasoningEffort: - (cli.flags.reasoning as ReasoningEffort | undefined) ?? "medium", - flexMode: cli.flags.flexMode || (config.flexMode ?? false), - provider, - disableResponseStorage, -}; - -// Check for updates after loading config. This is important because we write state file in -// the config dir. -try { - await checkForUpdates(); -} catch { - // ignore -} - -// For --flex-mode, validate and exit if incorrect. -if (config.flexMode) { - const allowedFlexModels = new Set(["o3", "o4-mini"]); - if (!allowedFlexModels.has(config.model)) { - if (cli.flags.flexMode) { - // eslint-disable-next-line no-console - console.error( - `The --flex-mode option is only supported when using the 'o3' or 'o4-mini' models. ` + - `Current model: '${config.model}'.`, - ); - process.exit(1); - } else { - config.flexMode = false; - } - } -} - -if ( - !(await isModelSupportedForResponses(provider, config.model)) && - (!provider || provider.toLowerCase() === "openai") -) { - // eslint-disable-next-line no-console - console.error( - `The model "${config.model}" does not appear in the list of models ` + - `available to your account. Double-check the spelling (use\n` + - ` openai models list\n` + - `to see the full list) or choose another model with the --model flag.`, - ); - process.exit(1); -} - -let rollout: AppRollout | undefined; - -// For --history, show session selector and optionally update prompt or rollout. -if (cli.flags.history) { - const result: { path: string; mode: "view" | "resume" } | null = - await new Promise((resolve) => { - const instance = render( - React.createElement(SessionsOverlay, { - onView: (p: string) => { - instance.unmount(); - resolve({ path: p, mode: "view" }); - }, - onResume: (p: string) => { - instance.unmount(); - resolve({ path: p, mode: "resume" }); - }, - onExit: () => { - instance.unmount(); - resolve(null); - }, - }), - ); - }); - - if (!result) { - process.exit(0); - } - - if (result.mode === "view") { - try { - const content = fs.readFileSync(result.path, "utf-8"); - rollout = JSON.parse(content) as AppRollout; - } catch (error) { - // eslint-disable-next-line no-console - console.error("Error reading session file:", error); - process.exit(1); - } - } else { - prompt = `Resume this session: ${result.path}`; - } -} - -// For --view, optionally load an existing rollout from disk, display it and exit. -if (cli.flags.view) { - const viewPath = cli.flags.view; - const absolutePath = path.isAbsolute(viewPath) - ? viewPath - : path.join(process.cwd(), viewPath); - try { - const content = fs.readFileSync(absolutePath, "utf-8"); - rollout = JSON.parse(content) as AppRollout; - } catch (error) { - // eslint-disable-next-line no-console - console.error("Error reading rollout file:", error); - process.exit(1); - } -} - -// For --fullcontext, run the separate cli entrypoint and exit. -if (fullContextMode) { - await runSinglePass({ - originalPrompt: prompt, - config, - rootPath: process.cwd(), - }); - onExit(); - process.exit(0); -} - -// Ensure that all values in additionalWritableRoots are absolute paths. -const additionalWritableRoots: ReadonlyArray = ( - cli.flags.writableRoot ?? [] -).map((p) => path.resolve(p)); - -// For --quiet, run the cli without user interactions and exit. -if (cli.flags.quiet) { - process.env["CODEX_QUIET_MODE"] = "1"; - if (!prompt || prompt.trim() === "") { - // eslint-disable-next-line no-console - console.error( - 'Quiet mode requires a prompt string, e.g.,: codex -q "Fix bug #123 in the foobar project"', - ); - process.exit(1); - } - - // Determine approval policy for quiet mode based on flags - const quietApprovalPolicy: ApprovalPolicy = - cli.flags.fullAuto || cli.flags.approvalMode === "full-auto" - ? AutoApprovalMode.FULL_AUTO - : cli.flags.autoEdit || cli.flags.approvalMode === "auto-edit" - ? AutoApprovalMode.AUTO_EDIT - : config.approvalMode || AutoApprovalMode.SUGGEST; - - await runQuietMode({ - prompt, - imagePaths: imagePaths || [], - approvalPolicy: quietApprovalPolicy, - additionalWritableRoots, - config, - }); - onExit(); - process.exit(0); -} - -// Default to the "suggest" policy. -// Determine the approval policy to use in interactive mode. -// -// Priority (highest → lowest): -// 1. --fullAuto – run everything automatically in a sandbox. -// 2. --dangerouslyAutoApproveEverything – run everything **without** a sandbox -// or prompts. This is intended for completely trusted environments. Since -// it is more dangerous than --fullAuto we deliberately give it lower -// priority so a user specifying both flags still gets the safer behaviour. -// 3. --autoEdit – automatically approve edits, but prompt for commands. -// 4. config.approvalMode - use the approvalMode setting from ~/.codex/config.json. -// 5. Default – suggest mode (prompt for everything). - -const approvalPolicy: ApprovalPolicy = - cli.flags.fullAuto || cli.flags.approvalMode === "full-auto" - ? AutoApprovalMode.FULL_AUTO - : cli.flags.autoEdit || cli.flags.approvalMode === "auto-edit" - ? AutoApprovalMode.AUTO_EDIT - : config.approvalMode || AutoApprovalMode.SUGGEST; - -const instance = render( - , - { - patchConsole: process.env["DEBUG"] ? false : true, - }, -); -setInkRenderer(instance); - -function formatResponseItemForQuietMode(item: ResponseItem): string { - if (!PRETTY_PRINT) { - return JSON.stringify(item); - } - switch (item.type) { - case "message": { - const role = item.role === "assistant" ? "assistant" : item.role; - const txt = item.content - .map((c) => { - if (c.type === "output_text" || c.type === "input_text") { - return c.text; - } - if (c.type === "input_image") { - return ""; - } - if (c.type === "input_file") { - return c.filename; - } - if (c.type === "refusal") { - return c.refusal; - } - return "?"; - }) - .join(" "); - return `${role}: ${txt}`; - } - case "function_call": { - const details = parseToolCall(item); - return `$ ${details?.cmdReadableText ?? item.name}`; - } - case "function_call_output": { - // @ts-expect-error metadata unknown on ResponseFunctionToolCallOutputItem - const meta = item.metadata as ExecOutputMetadata; - const parts: Array = []; - if (typeof meta?.exit_code === "number") { - parts.push(`code: ${meta.exit_code}`); - } - if (typeof meta?.duration_seconds === "number") { - parts.push(`duration: ${meta.duration_seconds}s`); - } - const header = parts.length > 0 ? ` (${parts.join(", ")})` : ""; - return `command.stdout${header}\n${item.output}`; - } - default: { - return JSON.stringify(item); - } - } -} - -async function runQuietMode({ - prompt, - imagePaths, - approvalPolicy, - additionalWritableRoots, - config, -}: { - prompt: string; - imagePaths: Array; - approvalPolicy: ApprovalPolicy; - additionalWritableRoots: ReadonlyArray; - config: AppConfig; -}): Promise { - const agent = new AgentLoop({ - model: config.model, - config: config, - instructions: config.instructions, - provider: config.provider, - approvalPolicy, - additionalWritableRoots, - disableResponseStorage: config.disableResponseStorage, - onItem: (item: ResponseItem) => { - // eslint-disable-next-line no-console - console.log(formatResponseItemForQuietMode(item)); - }, - onLoading: () => { - /* intentionally ignored in quiet mode */ - }, - getCommandConfirmation: ( - _command: Array, - ): Promise => { - // In quiet mode, default to NO_CONTINUE, except when in full-auto mode - const reviewDecision = - approvalPolicy === AutoApprovalMode.FULL_AUTO - ? ReviewDecision.YES - : ReviewDecision.NO_CONTINUE; - return Promise.resolve({ review: reviewDecision }); - }, - onLastResponseId: () => { - /* intentionally ignored in quiet mode */ - }, - }); - - const inputItem = await createInputItem(prompt, imagePaths); - await agent.run([inputItem]); -} - -const exit = () => { - onExit(); - process.exit(0); -}; - -process.on("SIGINT", exit); -process.on("SIGQUIT", exit); -process.on("SIGTERM", exit); - -// --------------------------------------------------------------------------- -// Fallback for Ctrl-C when stdin is in raw-mode -// --------------------------------------------------------------------------- - -if (process.stdin.isTTY) { - // Ensure we do not leave the terminal in raw mode if the user presses - // Ctrl-C while some other component has focus and Ink is intercepting - // input. Node does *not* emit a SIGINT in raw-mode, so we listen for the - // corresponding byte (0x03) ourselves and trigger a graceful shutdown. - const onRawData = (data: Buffer | string): void => { - const str = Buffer.isBuffer(data) ? data.toString("utf8") : data; - if (str === "\u0003") { - exit(); - } - }; - process.stdin.on("data", onRawData); -} - -// Ensure terminal clean-up always runs, even when other code calls -// `process.exit()` directly. -process.once("exit", onExit); diff --git a/codex-cli/src/components/approval-mode-overlay.tsx b/codex-cli/src/components/approval-mode-overlay.tsx deleted file mode 100644 index dd079e3b..00000000 --- a/codex-cli/src/components/approval-mode-overlay.tsx +++ /dev/null @@ -1,47 +0,0 @@ -import TypeaheadOverlay from "./typeahead-overlay.js"; -import { AutoApprovalMode } from "../utils/auto-approval-mode.js"; -import { Text } from "ink"; -import React from "react"; - -type Props = { - currentMode: string; - onSelect: (mode: string) => void; - onExit: () => void; -}; - -/** - * Overlay to switch between the different automatic‑approval policies. - * - * The list of available modes is derived from the AutoApprovalMode enum so we - * stay in sync with the core agent behaviour. It re‑uses the generic - * TypeaheadOverlay component for the actual UI/UX. - */ -export default function ApprovalModeOverlay({ - currentMode, - onSelect, - onExit, -}: Props): JSX.Element { - const items = React.useMemo( - () => - Object.values(AutoApprovalMode).map((m) => ({ - label: m, - value: m, - })), - [], - ); - - return ( - - Current mode: {currentMode} - - } - initialItems={items} - currentValue={currentMode} - onSelect={onSelect} - onExit={onExit} - /> - ); -} diff --git a/codex-cli/src/components/chat/message-history.tsx b/codex-cli/src/components/chat/message-history.tsx deleted file mode 100644 index bab6b166..00000000 --- a/codex-cli/src/components/chat/message-history.tsx +++ /dev/null @@ -1,86 +0,0 @@ -import type { TerminalHeaderProps } from "./terminal-header.js"; -import type { GroupedResponseItem } from "./use-message-grouping.js"; -import type { ResponseItem } from "openai/resources/responses/responses.mjs"; -import type { FileOpenerScheme } from "src/utils/config.js"; - -import TerminalChatResponseItem from "./terminal-chat-response-item.js"; -import TerminalHeader from "./terminal-header.js"; -import { Box, Static } from "ink"; -import React from "react"; - -// A batch entry can either be a standalone response item or a grouped set of -// items (e.g. auto‑approved tool‑call batches) that should be rendered -// together. -type BatchEntry = { item?: ResponseItem; group?: GroupedResponseItem }; -type MessageHistoryProps = { - batch: Array; - groupCounts: Record; - items: Array; - userMsgCount: number; - confirmationPrompt: React.ReactNode; - loading: boolean; - headerProps: TerminalHeaderProps; - fileOpener: FileOpenerScheme | undefined; -}; - -const MessageHistory: React.FC = ({ - batch, - headerProps, - fileOpener, -}) => { - const messages = batch.map(({ item }) => item!); - - return ( - - {/* - * The Static component receives a mixed array of the literal string - * "header" plus the streamed ResponseItem objects. After filtering out - * the header entry we can safely treat the remaining values as - * ResponseItem, however TypeScript cannot infer the refined type from - * the runtime check and therefore reports property‑access errors. - * - * A short cast after the refinement keeps the implementation tidy while - * preserving type‑safety. - */} - - {(item, index) => { - if (item === "header") { - return ; - } - - // After the guard above `item` can only be a ResponseItem. - const message = item as ResponseItem; - return ( - - - - ); - }} - - - ); -}; - -export default React.memo(MessageHistory); diff --git a/codex-cli/src/components/chat/multiline-editor.tsx b/codex-cli/src/components/chat/multiline-editor.tsx deleted file mode 100644 index 6b24bc27..00000000 --- a/codex-cli/src/components/chat/multiline-editor.tsx +++ /dev/null @@ -1,392 +0,0 @@ -/* eslint-disable @typescript-eslint/no-explicit-any */ - -import { useTerminalSize } from "../../hooks/use-terminal-size"; -import TextBuffer from "../../text-buffer.js"; -import chalk from "chalk"; -import { Box, Text, useInput } from "ink"; -import { EventEmitter } from "node:events"; -import React, { useRef, useState } from "react"; - -/* -------------------------------------------------------------------------- - * Polyfill missing `ref()` / `unref()` methods on the mock `Stdin` stream - * provided by `ink-testing-library`. - * - * The real `process.stdin` object exposed by Node.js inherits these methods - * from `Socket`, but the lightweight stub used in tests only extends - * `EventEmitter`. Ink calls the two methods when enabling/disabling raw - * mode, so make them harmless no-ops when they're absent to avoid runtime - * failures during unit tests. - * ----------------------------------------------------------------------- */ - -// Cast through `unknown` ➜ `any` to avoid the `TS2352`/`TS4111` complaints -// when augmenting the prototype with the stubbed `ref`/`unref` methods in the -// test environment. Using `any` here is acceptable because we purposefully -// monkey‑patch internals of Node's `EventEmitter` solely for the benefit of -// Ink's stdin stub – type‑safety is not a primary concern at this boundary. -// -const proto: any = EventEmitter.prototype; - -if (typeof proto["ref"] !== "function") { - proto["ref"] = function ref() {}; -} -if (typeof proto["unref"] !== "function") { - proto["unref"] = function unref() {}; -} - -/* - * The `ink-testing-library` stub emits only a `data` event when its `stdin` - * mock receives `write()` calls. Ink, however, listens for `readable` and - * uses the `read()` method to fetch the buffered chunk. Bridge the gap by - * hooking into `EventEmitter.emit` so that every `data` emission also: - * 1. Buffers the chunk for a subsequent `read()` call, and - * 2. Triggers a `readable` event, matching the contract expected by Ink. - */ - -// Preserve original emit to avoid infinite recursion. -// eslint‑disable‑next‑line @typescript-eslint/no‑unsafe‑assignment -const originalEmit = proto["emit"] as (...args: Array) => boolean; - -proto["emit"] = function patchedEmit( - this: any, - event: string, - ...args: Array -): boolean { - if (event === "data") { - const chunk = args[0] as string; - - if ( - process.env["TEXTBUFFER_DEBUG"] === "1" || - process.env["TEXTBUFFER_DEBUG"] === "true" - ) { - // eslint-disable-next-line no-console - console.log("[MultilineTextEditor:stdin] data", JSON.stringify(chunk)); - } - // Store carriage returns as‑is so that Ink can distinguish between plain - // ("\r") and a bare line‑feed ("\n"). This matters because Ink's - // `parseKeypress` treats "\r" as key.name === "return", whereas "\n" maps - // to "enter" – allowing us to differentiate between plain Enter (submit) - // and Shift+Enter (insert newline) inside `useInput`. - - // Identify the lightweight testing stub: lacks `.read()` but exposes - // `.setRawMode()` and `isTTY` similar to the real TTY stream. - if ( - !(this as any)._inkIsStub && - typeof (this as any).setRawMode === "function" && - typeof (this as any).isTTY === "boolean" && - typeof (this as any).read !== "function" - ) { - (this as any)._inkIsStub = true; - - // Provide a minimal `read()` shim so Ink can pull queued chunks. - (this as any).read = function read() { - const ret = (this as any)._inkBuffered ?? null; - (this as any)._inkBuffered = null; - if ( - process.env["TEXTBUFFER_DEBUG"] === "1" || - process.env["TEXTBUFFER_DEBUG"] === "true" - ) { - // eslint-disable-next-line no-console - console.log("[MultilineTextEditor:stdin.read]", JSON.stringify(ret)); - } - return ret; - }; - } - - if ((this as any)._inkIsStub) { - // Buffer the payload so that `read()` can synchronously retrieve it. - if (typeof (this as any)._inkBuffered === "string") { - (this as any)._inkBuffered += chunk; - } else { - (this as any)._inkBuffered = chunk; - } - - // Notify listeners that data is ready in a way Ink understands. - if ( - process.env["TEXTBUFFER_DEBUG"] === "1" || - process.env["TEXTBUFFER_DEBUG"] === "true" - ) { - // eslint-disable-next-line no-console - console.log( - "[MultilineTextEditor:stdin] -> readable", - JSON.stringify(chunk), - ); - } - originalEmit.call(this, "readable"); - } - } - - // Forward the original event. - return originalEmit.call(this, event, ...args); -}; - -export interface MultilineTextEditorProps { - // Initial contents. - readonly initialText?: string; - - // Visible width. - readonly width?: number; - - // Visible height. - readonly height?: number; - - // Called when the user submits (plain key). - readonly onSubmit?: (text: string) => void; - - // Capture keyboard input. - readonly focus?: boolean; - - // Called when the internal text buffer updates. - readonly onChange?: (text: string) => void; - - // Optional initial cursor position (character offset) - readonly initialCursorOffset?: number; -} - -// Expose a minimal imperative API so parent components (e.g. TerminalChatInput) -// can query the caret position to implement behaviours like history -// navigation that depend on whether the cursor sits on the first/last line. -export interface MultilineTextEditorHandle { - /** Current caret row */ - getRow(): number; - /** Current caret column */ - getCol(): number; - /** Total number of lines in the buffer */ - getLineCount(): number; - /** Helper: caret is on the very first row */ - isCursorAtFirstRow(): boolean; - /** Helper: caret is on the very last row */ - isCursorAtLastRow(): boolean; - /** Full text contents */ - getText(): string; - /** Move the cursor to the end of the text */ - moveCursorToEnd(): void; -} - -const MultilineTextEditorInner = ( - { - initialText = "", - // Width can be provided by the caller. When omitted we fall back to the - // current terminal size (minus some padding handled by `useTerminalSize`). - width, - height = 10, - onSubmit, - focus = true, - onChange, - initialCursorOffset, - }: MultilineTextEditorProps, - ref: React.Ref, -): React.ReactElement => { - // --------------------------------------------------------------------------- - // Editor State - // --------------------------------------------------------------------------- - - const buffer = useRef(new TextBuffer(initialText, initialCursorOffset)); - const [version, setVersion] = useState(0); - - // Keep track of the current terminal size so that the editor grows/shrinks - // with the window. `useTerminalSize` already subtracts a small horizontal - // padding so that we don't butt up right against the edge. - const terminalSize = useTerminalSize(); - - // If the caller didn't specify a width we dynamically choose one based on - // the terminal's current column count. We still enforce a reasonable - // minimum so that the UI never becomes unusably small. - const effectiveWidth = Math.max(20, width ?? terminalSize.columns); - - // --------------------------------------------------------------------------- - // Keyboard handling. - // --------------------------------------------------------------------------- - - useInput( - (input, key) => { - if (!focus) { - return; - } - - if ( - process.env["TEXTBUFFER_DEBUG"] === "1" || - process.env["TEXTBUFFER_DEBUG"] === "true" - ) { - // eslint-disable-next-line no-console - console.log("[MultilineTextEditor] event", { input, key }); - } - - // 1a) CSI-u / modifyOtherKeys *mode 2* (Ink strips initial ESC, so we - // start with '[') – format: "[;u". - if (input.startsWith("[") && input.endsWith("u")) { - const m = input.match(/^\[([0-9]+);([0-9]+)u$/); - if (m && m[1] === "13") { - const mod = Number(m[2]); - // In xterm's encoding: bit-1 (value 2) is Shift. Everything >1 that - // isn't exactly 1 means some modifier was held. We treat *shift or - // alt present* (2,3,4,6,8,9) as newline; Ctrl (bit-2 / value 4) - // triggers submit. See xterm/DEC modifyOtherKeys docs. - - const hasCtrl = Math.floor(mod / 4) % 2 === 1; - if (hasCtrl) { - if (onSubmit) { - onSubmit(buffer.current.getText()); - } - } else { - buffer.current.newline(); - } - setVersion((v) => v + 1); - return; - } - } - - // 1b) CSI-~ / modifyOtherKeys *mode 1* – format: "[27;;~". - // Terminals such as iTerm2 (default), older xterm versions, or when - // modifyOtherKeys=1 is configured, emit this legacy sequence. We - // translate it to the same behaviour as the mode‑2 variant above so - // that Shift+Enter (newline) / Ctrl+Enter (submit) work regardless - // of the user’s terminal settings. - if (input.startsWith("[27;") && input.endsWith("~")) { - const m = input.match(/^\[27;([0-9]+);13~$/); - if (m) { - const mod = Number(m[1]); - const hasCtrl = Math.floor(mod / 4) % 2 === 1; - - if (hasCtrl) { - if (onSubmit) { - onSubmit(buffer.current.getText()); - } - } else { - buffer.current.newline(); - } - setVersion((v) => v + 1); - return; - } - } - - // 2) Single‑byte control chars ------------------------------------------------ - if (input === "\n") { - // Ctrl+J or pasted newline → insert newline. - buffer.current.newline(); - setVersion((v) => v + 1); - return; - } - - if (input === "\r") { - // Plain Enter – submit (works on all basic terminals). - if (onSubmit) { - onSubmit(buffer.current.getText()); - } - return; - } - - // Let fall through so the parent handler (if any) can act on it. - - // Delegate remaining keys to our pure TextBuffer - if ( - process.env["TEXTBUFFER_DEBUG"] === "1" || - process.env["TEXTBUFFER_DEBUG"] === "true" - ) { - // eslint-disable-next-line no-console - console.log("[MultilineTextEditor] key event", { input, key }); - } - - const modified = buffer.current.handleInput( - input, - key as Record, - { height, width: effectiveWidth }, - ); - if (modified) { - setVersion((v) => v + 1); - } - - const newText = buffer.current.getText(); - if (onChange) { - onChange(newText); - } - }, - { isActive: focus }, - ); - - // --------------------------------------------------------------------------- - // Rendering helpers. - // --------------------------------------------------------------------------- - - /* ------------------------------------------------------------------------- */ - /* Imperative handle – expose a read‑only view of caret & buffer geometry */ - /* ------------------------------------------------------------------------- */ - - React.useImperativeHandle( - ref, - () => ({ - getRow: () => buffer.current.getCursor()[0], - getCol: () => buffer.current.getCursor()[1], - getLineCount: () => buffer.current.getText().split("\n").length, - isCursorAtFirstRow: () => buffer.current.getCursor()[0] === 0, - isCursorAtLastRow: () => { - const [row] = buffer.current.getCursor(); - const lineCount = buffer.current.getText().split("\n").length; - return row === lineCount - 1; - }, - getText: () => buffer.current.getText(), - moveCursorToEnd: () => { - buffer.current.move("home"); - const lines = buffer.current.getText().split("\n"); - for (let i = 0; i < lines.length - 1; i++) { - buffer.current.move("down"); - } - buffer.current.move("end"); - // Force a re-render - setVersion((v) => v + 1); - }, - }), - [], - ); - - // Read everything from the buffer - const visibleLines = buffer.current.getVisibleLines({ - height, - width: effectiveWidth, - }); - const [cursorRow, cursorCol] = buffer.current.getCursor(); - const scrollRow = (buffer.current as any).scrollRow as number; - const scrollCol = (buffer.current as any).scrollCol as number; - - return ( - - {visibleLines.map((lineText, idx) => { - const absoluteRow = scrollRow + idx; - - // apply horizontal slice - let display = lineText.slice(scrollCol, scrollCol + effectiveWidth); - if (display.length < effectiveWidth) { - display = display.padEnd(effectiveWidth, " "); - } - - // Highlight the *character under the caret* (i.e. the one immediately - // to the right of the insertion position) so that the block cursor - // visually matches the logical caret location. This makes the - // highlighted glyph the one that would be replaced by `insert()` and - // *not* the one that would be removed by `backspace()`. - - if (absoluteRow === cursorRow) { - const relativeCol = cursorCol - scrollCol; - const highlightCol = relativeCol; - - if (highlightCol >= 0 && highlightCol < effectiveWidth) { - const charToHighlight = display[highlightCol] || " "; - const highlighted = chalk.inverse(charToHighlight); - display = - display.slice(0, highlightCol) + - highlighted + - display.slice(highlightCol + 1); - } else if (relativeCol === effectiveWidth) { - // Caret sits just past the right edge; show a block cursor in the - // gutter so the user still sees it. - display = display.slice(0, effectiveWidth - 1) + chalk.inverse(" "); - } - } - - return {display}; - })} - - ); -}; - -const MultilineTextEditor = React.forwardRef(MultilineTextEditorInner); -export default MultilineTextEditor; diff --git a/codex-cli/src/components/chat/terminal-chat-command-review.tsx b/codex-cli/src/components/chat/terminal-chat-command-review.tsx deleted file mode 100644 index 912af979..00000000 --- a/codex-cli/src/components/chat/terminal-chat-command-review.tsx +++ /dev/null @@ -1,256 +0,0 @@ -import { ReviewDecision } from "../../utils/agent/review"; -// TODO: figure out why `cli-spinners` fails on Node v20.9.0 -// which is why we have to do this in the first place -// -// @ts-expect-error select.js is JavaScript and has no types -import { Select } from "../vendor/ink-select/select"; -import TextInput from "../vendor/ink-text-input"; -import { Box, Text, useInput } from "ink"; -import React from "react"; - -// default deny‑reason: -const DEFAULT_DENY_MESSAGE = - "Don't do that, but keep trying to fix the problem"; - -export function TerminalChatCommandReview({ - confirmationPrompt, - onReviewCommand, - // callback to switch approval mode overlay - onSwitchApprovalMode, - explanation: propExplanation, - // whether this review Select is active (listening for keys) - isActive = true, -}: { - confirmationPrompt: React.ReactNode; - onReviewCommand: (decision: ReviewDecision, customMessage?: string) => void; - onSwitchApprovalMode: () => void; - explanation?: string; - // when false, disable the underlying Select so it won't capture input - isActive?: boolean; -}): React.ReactElement { - const [mode, setMode] = React.useState<"select" | "input" | "explanation">( - "select", - ); - const [explanation, setExplanation] = React.useState(""); - - // If the component receives an explanation prop, update the state - React.useEffect(() => { - if (propExplanation) { - setExplanation(propExplanation); - setMode("explanation"); - } - }, [propExplanation]); - const [msg, setMsg] = React.useState(""); - - // ------------------------------------------------------------------------- - // Determine whether the "always approve" option should be displayed. We - // only hide it for the special `apply_patch` command since approving those - // permanently would bypass the user's review of future file modifications. - // The information is embedded in the `confirmationPrompt` React element – - // we inspect the `commandForDisplay` prop exposed by - // to extract the base command. - // ------------------------------------------------------------------------- - - const showAlwaysApprove = React.useMemo(() => { - if ( - React.isValidElement(confirmationPrompt) && - // eslint-disable-next-line @typescript-eslint/no-explicit-any - typeof (confirmationPrompt as any).props?.commandForDisplay === "string" - ) { - // eslint-disable-next-line @typescript-eslint/no-explicit-any - const command: string = (confirmationPrompt as any).props - .commandForDisplay; - // Grab the first token of the first line – that corresponds to the base - // command even when the string contains embedded newlines (e.g. diffs). - const baseCmd = command.split("\n")[0]?.trim().split(/\s+/)[0] ?? ""; - return baseCmd !== "apply_patch"; - } - // Default to showing the option when we cannot reliably detect the base - // command. - return true; - }, [confirmationPrompt]); - - // Memoize the list of selectable options to avoid recreating the array on - // every render. This keeps { - if (value === "edit") { - setMode("input"); - } else if (value === "switch") { - onSwitchApprovalMode(); - } else { - onReviewCommand(value); - } - }} - options={approvalOptions} - /> - - - ) : mode === "input" ? ( - <> - Give the model feedback (↵ to submit): - - - - - - - {msg.trim() === "" && ( - - - default:  - {DEFAULT_DENY_MESSAGE} - - - )} - - ) : null} - - - ); -} diff --git a/codex-cli/src/components/chat/terminal-chat-completions.tsx b/codex-cli/src/components/chat/terminal-chat-completions.tsx deleted file mode 100644 index eb7e47f8..00000000 --- a/codex-cli/src/components/chat/terminal-chat-completions.tsx +++ /dev/null @@ -1,64 +0,0 @@ -import { Box, Text } from "ink"; -import React, { useMemo } from "react"; - -type TextCompletionProps = { - /** - * Array of text completion options to display in the list - */ - completions: Array; - - /** - * Maximum number of completion items to show at once in the view - */ - displayLimit: number; - - /** - * Index of the currently selected completion in the completions array - */ - selectedCompletion: number; -}; - -function TerminalChatCompletions({ - completions, - selectedCompletion, - displayLimit, -}: TextCompletionProps): JSX.Element { - const visibleItems = useMemo(() => { - // Try to keep selection centered in view - let startIndex = Math.max( - 0, - selectedCompletion - Math.floor(displayLimit / 2), - ); - - // Fix window position when at the end of the list - if (completions.length - startIndex < displayLimit) { - startIndex = Math.max(0, completions.length - displayLimit); - } - - const endIndex = Math.min(completions.length, startIndex + displayLimit); - - return completions.slice(startIndex, endIndex).map((completion, index) => ({ - completion, - originalIndex: index + startIndex, - })); - }, [completions, selectedCompletion, displayLimit]); - - return ( - - {visibleItems.map(({ completion, originalIndex }) => ( - - {completion} - - ))} - - ); -} - -export default TerminalChatCompletions; diff --git a/codex-cli/src/components/chat/terminal-chat-input-thinking.tsx b/codex-cli/src/components/chat/terminal-chat-input-thinking.tsx deleted file mode 100644 index 714cc59f..00000000 --- a/codex-cli/src/components/chat/terminal-chat-input-thinking.tsx +++ /dev/null @@ -1,129 +0,0 @@ -import { log } from "../../utils/logger/log.js"; -import { Box, Text, useInput, useStdin } from "ink"; -import React, { useState } from "react"; -import { useInterval } from "use-interval"; - -// Retaining a single static placeholder text for potential future use. The -// more elaborate randomised thinking prompts were removed to streamline the -// UI – the elapsed‑time counter now provides sufficient feedback. - -export default function TerminalChatInputThinking({ - onInterrupt, - active, - thinkingSeconds, -}: { - onInterrupt: () => void; - active: boolean; - thinkingSeconds: number; -}): React.ReactElement { - const [awaitingConfirm, setAwaitingConfirm] = useState(false); - const [dots, setDots] = useState(""); - - // Animate the ellipsis - useInterval(() => { - setDots((prev) => (prev.length < 3 ? prev + "." : "")); - }, 500); - - const { stdin, setRawMode } = useStdin(); - - React.useEffect(() => { - if (!active) { - return; - } - - setRawMode?.(true); - - const onData = (data: Buffer | string) => { - if (awaitingConfirm) { - return; - } - - const str = Buffer.isBuffer(data) ? data.toString("utf8") : data; - if (str === "\x1b\x1b") { - log( - "raw stdin: received collapsed ESC ESC – starting confirmation timer", - ); - setAwaitingConfirm(true); - setTimeout(() => setAwaitingConfirm(false), 1500); - } - }; - - stdin?.on("data", onData); - return () => { - stdin?.off("data", onData); - }; - }, [stdin, awaitingConfirm, onInterrupt, active, setRawMode]); - - // No timers required beyond tracking the elapsed seconds supplied via props. - - useInput( - (_input, key) => { - if (!key.escape) { - return; - } - - if (awaitingConfirm) { - log("useInput: second ESC detected – triggering onInterrupt()"); - onInterrupt(); - setAwaitingConfirm(false); - } else { - log("useInput: first ESC detected – waiting for confirmation"); - setAwaitingConfirm(true); - setTimeout(() => setAwaitingConfirm(false), 1500); - } - }, - { isActive: active }, - ); - - // Custom ball animation including the elapsed seconds - const ballFrames = [ - "( ● )", - "( ● )", - "( ● )", - "( ● )", - "( ●)", - "( ● )", - "( ● )", - "( ● )", - "( ● )", - "(● )", - ]; - - const [frame, setFrame] = useState(0); - - useInterval(() => { - setFrame((idx) => (idx + 1) % ballFrames.length); - }, 80); - - // Preserve the spinner (ball) animation while keeping the elapsed seconds - // text static. We achieve this by rendering the bouncing ball inside the - // parentheses and appending the seconds counter *after* the spinner rather - // than injecting it directly next to the ball (which caused the counter to - // move horizontally together with the ball). - - const frameTemplate = ballFrames[frame] ?? ballFrames[0]; - const frameWithSeconds = `${frameTemplate} ${thinkingSeconds}s`; - - return ( - - - - {frameWithSeconds} - - Thinking - {dots} - - - - Press Esc twice to interrupt - - - {awaitingConfirm && ( - - Press Esc again to interrupt and enter a new - instruction - - )} - - ); -} diff --git a/codex-cli/src/components/chat/terminal-chat-input.tsx b/codex-cli/src/components/chat/terminal-chat-input.tsx deleted file mode 100644 index 66428f84..00000000 --- a/codex-cli/src/components/chat/terminal-chat-input.tsx +++ /dev/null @@ -1,1017 +0,0 @@ -import type { MultilineTextEditorHandle } from "./multiline-editor"; -import type { ReviewDecision } from "../../utils/agent/review.js"; -import type { FileSystemSuggestion } from "../../utils/file-system-suggestions.js"; -import type { HistoryEntry } from "../../utils/storage/command-history.js"; -import type { - ResponseInputItem, - ResponseItem, -} from "openai/resources/responses/responses.mjs"; - -import MultilineTextEditor from "./multiline-editor"; -import { TerminalChatCommandReview } from "./terminal-chat-command-review.js"; -import TextCompletions from "./terminal-chat-completions.js"; -import { loadConfig } from "../../utils/config.js"; -import { getFileSystemSuggestions } from "../../utils/file-system-suggestions.js"; -import { expandFileTags } from "../../utils/file-tag-utils"; -import { createInputItem } from "../../utils/input-utils.js"; -import { log } from "../../utils/logger/log.js"; -import { setSessionId } from "../../utils/session.js"; -import { SLASH_COMMANDS, type SlashCommand } from "../../utils/slash-commands"; -import { - loadCommandHistory, - addToHistory, -} from "../../utils/storage/command-history.js"; -import { clearTerminal, onExit } from "../../utils/terminal.js"; -import { Box, Text, useApp, useInput, useStdin } from "ink"; -import { fileURLToPath } from "node:url"; -import React, { - useCallback, - useState, - Fragment, - useEffect, - useRef, -} from "react"; -import { useInterval } from "use-interval"; - -const suggestions = [ - "explain this codebase to me", - "fix any build errors", - "are there any bugs in my code?", -]; - -export default function TerminalChatInput({ - isNew, - loading, - submitInput, - confirmationPrompt, - explanation, - submitConfirmation, - setLastResponseId, - setItems, - contextLeftPercent, - openOverlay, - openModelOverlay, - openApprovalOverlay, - openHelpOverlay, - openDiffOverlay, - openSessionsOverlay, - onCompact, - interruptAgent, - active, - thinkingSeconds, - items = [], -}: { - isNew: boolean; - loading: boolean; - submitInput: (input: Array) => void; - confirmationPrompt: React.ReactNode | null; - explanation?: string; - submitConfirmation: ( - decision: ReviewDecision, - customDenyMessage?: string, - ) => void; - setLastResponseId: (lastResponseId: string) => void; - setItems: React.Dispatch>>; - contextLeftPercent: number; - openOverlay: () => void; - openModelOverlay: () => void; - openApprovalOverlay: () => void; - openHelpOverlay: () => void; - openDiffOverlay: () => void; - openSessionsOverlay: () => void; - onCompact: () => void; - interruptAgent: () => void; - active: boolean; - thinkingSeconds: number; - // New: current conversation items so we can include them in bug reports - items?: Array; -}): React.ReactElement { - // Slash command suggestion index - const [selectedSlashSuggestion, setSelectedSlashSuggestion] = - useState(0); - const app = useApp(); - const [selectedSuggestion, setSelectedSuggestion] = useState(0); - const [input, setInput] = useState(""); - const [history, setHistory] = useState>([]); - const [historyIndex, setHistoryIndex] = useState(null); - const [draftInput, setDraftInput] = useState(""); - const [skipNextSubmit, setSkipNextSubmit] = useState(false); - const [fsSuggestions, setFsSuggestions] = useState< - Array - >([]); - const [selectedCompletion, setSelectedCompletion] = useState(-1); - // Multiline text editor key to force remount after submission - const [editorState, setEditorState] = useState<{ - key: number; - initialCursorOffset?: number; - }>({ key: 0 }); - // Imperative handle from the multiline editor so we can query caret position - const editorRef = useRef(null); - // Track the caret row across keystrokes - const prevCursorRow = useRef(null); - const prevCursorWasAtLastRow = useRef(false); - - // --- Helper for updating input, remounting editor, and moving cursor to end --- - const applyFsSuggestion = useCallback((newInputText: string) => { - setInput(newInputText); - setEditorState((s) => ({ - key: s.key + 1, - initialCursorOffset: newInputText.length, - })); - }, []); - - // --- Helper for updating file system suggestions --- - function updateFsSuggestions( - txt: string, - alwaysUpdateSelection: boolean = false, - ) { - // Clear file system completions if a space is typed - if (txt.endsWith(" ")) { - setFsSuggestions([]); - setSelectedCompletion(-1); - } else { - // Determine the current token (last whitespace-separated word) - const words = txt.trim().split(/\s+/); - const lastWord = words[words.length - 1] ?? ""; - - const shouldUpdateSelection = - lastWord.startsWith("@") || alwaysUpdateSelection; - - // Strip optional leading '@' for the path prefix - let pathPrefix: string; - if (lastWord.startsWith("@")) { - pathPrefix = lastWord.slice(1); - // If only '@' is typed, list everything in the current directory - pathPrefix = pathPrefix.length === 0 ? "./" : pathPrefix; - } else { - pathPrefix = lastWord; - } - - if (shouldUpdateSelection) { - const completions = getFileSystemSuggestions(pathPrefix); - setFsSuggestions(completions); - if (completions.length > 0) { - setSelectedCompletion((prev) => - prev < 0 || prev >= completions.length ? 0 : prev, - ); - } else { - setSelectedCompletion(-1); - } - } else if (fsSuggestions.length > 0) { - // Token cleared → clear menu - setFsSuggestions([]); - setSelectedCompletion(-1); - } - } - } - - /** - * Result of replacing text with a file system suggestion - */ - interface ReplacementResult { - /** The new text with the suggestion applied */ - text: string; - /** The selected suggestion if a replacement was made */ - suggestion: FileSystemSuggestion | null; - /** Whether a replacement was actually made */ - wasReplaced: boolean; - } - - // --- Helper for replacing input with file system suggestion --- - function getFileSystemSuggestion( - txt: string, - requireAtPrefix: boolean = false, - ): ReplacementResult { - if (fsSuggestions.length === 0 || selectedCompletion < 0) { - return { text: txt, suggestion: null, wasReplaced: false }; - } - - const words = txt.trim().split(/\s+/); - const lastWord = words[words.length - 1] ?? ""; - - // Check if @ prefix is required and the last word doesn't have it - if (requireAtPrefix && !lastWord.startsWith("@")) { - return { text: txt, suggestion: null, wasReplaced: false }; - } - - const selected = fsSuggestions[selectedCompletion]; - if (!selected) { - return { text: txt, suggestion: null, wasReplaced: false }; - } - - const replacement = lastWord.startsWith("@") - ? `@${selected.path}` - : selected.path; - words[words.length - 1] = replacement; - return { - text: words.join(" "), - suggestion: selected, - wasReplaced: true, - }; - } - - // Load command history on component mount - useEffect(() => { - async function loadHistory() { - const historyEntries = await loadCommandHistory(); - setHistory(historyEntries); - } - - loadHistory(); - }, []); - // Reset slash suggestion index when input prefix changes - useEffect(() => { - if (input.trim().startsWith("/")) { - setSelectedSlashSuggestion(0); - } - }, [input]); - - useInput( - (_input, _key) => { - // Slash command navigation: up/down to select, enter to fill - if (!confirmationPrompt && !loading && input.trim().startsWith("/")) { - const prefix = input.trim(); - const matches = SLASH_COMMANDS.filter((cmd: SlashCommand) => - cmd.command.startsWith(prefix), - ); - if (matches.length > 0) { - if (_key.tab) { - // Cycle and fill slash command suggestions on Tab - const len = matches.length; - // Determine new index based on shift state - const nextIdx = _key.shift - ? selectedSlashSuggestion <= 0 - ? len - 1 - : selectedSlashSuggestion - 1 - : selectedSlashSuggestion >= len - 1 - ? 0 - : selectedSlashSuggestion + 1; - setSelectedSlashSuggestion(nextIdx); - // Autocomplete the command in the input - const match = matches[nextIdx]; - if (!match) { - return; - } - const cmd = match.command; - setInput(cmd); - setDraftInput(cmd); - return; - } - if (_key.upArrow) { - setSelectedSlashSuggestion((prev) => - prev <= 0 ? matches.length - 1 : prev - 1, - ); - return; - } - if (_key.downArrow) { - setSelectedSlashSuggestion((prev) => - prev < 0 || prev >= matches.length - 1 ? 0 : prev + 1, - ); - return; - } - if (_key.return) { - // Execute the currently selected slash command - const selIdx = selectedSlashSuggestion; - const cmdObj = matches[selIdx]; - if (cmdObj) { - const cmd = cmdObj.command; - setInput(""); - setDraftInput(""); - setSelectedSlashSuggestion(0); - switch (cmd) { - case "/history": - openOverlay(); - break; - case "/sessions": - openSessionsOverlay(); - break; - case "/help": - openHelpOverlay(); - break; - case "/compact": - onCompact(); - break; - case "/model": - openModelOverlay(); - break; - case "/approval": - openApprovalOverlay(); - break; - case "/diff": - openDiffOverlay(); - break; - case "/bug": - onSubmit(cmd); - break; - case "/clear": - onSubmit(cmd); - break; - case "/clearhistory": - onSubmit(cmd); - break; - default: - break; - } - } - return; - } - } - } - if (!confirmationPrompt && !loading) { - if (fsSuggestions.length > 0) { - if (_key.upArrow) { - setSelectedCompletion((prev) => - prev <= 0 ? fsSuggestions.length - 1 : prev - 1, - ); - return; - } - - if (_key.downArrow) { - setSelectedCompletion((prev) => - prev >= fsSuggestions.length - 1 ? 0 : prev + 1, - ); - return; - } - - if (_key.tab && selectedCompletion >= 0) { - const { text: newText, wasReplaced } = - getFileSystemSuggestion(input); - - // Only proceed if the text was actually changed - if (wasReplaced) { - applyFsSuggestion(newText); - setFsSuggestions([]); - setSelectedCompletion(-1); - } - return; - } - } - - if (_key.upArrow) { - let moveThroughHistory = true; - - // Only use history when the caret was *already* on the very first - // row *before* this key-press. - const cursorRow = editorRef.current?.getRow?.() ?? 0; - const cursorCol = editorRef.current?.getCol?.() ?? 0; - const wasAtFirstRow = (prevCursorRow.current ?? cursorRow) === 0; - if (!(cursorRow === 0 && wasAtFirstRow)) { - moveThroughHistory = false; - } - - // If we are not yet in history mode, then also require that the col is zero so that - // we only trigger history navigation when the user is at the start of the input. - if (historyIndex == null && !(cursorRow === 0 && cursorCol === 0)) { - moveThroughHistory = false; - } - - // Move through history. - if (history.length && moveThroughHistory) { - let newIndex: number; - if (historyIndex == null) { - const currentDraft = editorRef.current?.getText?.() ?? input; - setDraftInput(currentDraft); - newIndex = history.length - 1; - } else { - newIndex = Math.max(0, historyIndex - 1); - } - setHistoryIndex(newIndex); - - setInput(history[newIndex]?.command ?? ""); - // Re-mount the editor so it picks up the new initialText - setEditorState((s) => ({ key: s.key + 1 })); - return; // handled - } - - // Otherwise let it propagate. - } - - if (_key.downArrow) { - // Only move forward in history when we're already *in* history mode - // AND the caret sits on the last line of the buffer. - const wasAtLastRow = - prevCursorWasAtLastRow.current ?? - editorRef.current?.isCursorAtLastRow() ?? - true; - if (historyIndex != null && wasAtLastRow) { - const newIndex = historyIndex + 1; - if (newIndex >= history.length) { - setHistoryIndex(null); - setInput(draftInput); - setEditorState((s) => ({ key: s.key + 1 })); - } else { - setHistoryIndex(newIndex); - setInput(history[newIndex]?.command ?? ""); - setEditorState((s) => ({ key: s.key + 1 })); - } - return; // handled - } - // Otherwise let it propagate - } - - // Defer filesystem suggestion logic to onSubmit if enter key is pressed - if (!_key.return) { - // Pressing tab should trigger the file system suggestions - const shouldUpdateSelection = _key.tab; - const targetInput = _key.delete ? input.slice(0, -1) : input + _input; - updateFsSuggestions(targetInput, shouldUpdateSelection); - } - } - - // Update the cached cursor position *after* **all** handlers (including - // the internal ) have processed this key event. - // - // Ink invokes `useInput` callbacks starting with **parent** components - // first, followed by their descendants. As a result the call above - // executes *before* the editor has had a chance to react to the key - // press and update its internal caret position. When navigating - // through a multi-line draft with the ↑ / ↓ arrow keys this meant we - // recorded the *old* cursor row instead of the one that results *after* - // the key press. Consequently, a subsequent ↑ still saw - // `prevCursorRow = 1` even though the caret was already on row 0 and - // history-navigation never kicked in. - // - // Defer the sampling by one tick so we read the *final* caret position - // for this frame. - setTimeout(() => { - prevCursorRow.current = editorRef.current?.getRow?.() ?? null; - prevCursorWasAtLastRow.current = - editorRef.current?.isCursorAtLastRow?.() ?? true; - }, 1); - - if (input.trim() === "" && isNew) { - if (_key.tab) { - setSelectedSuggestion( - (s) => (s + (_key.shift ? -1 : 1)) % (suggestions.length + 1), - ); - } else if (selectedSuggestion && _key.return) { - const suggestion = suggestions[selectedSuggestion - 1] || ""; - setInput(""); - setSelectedSuggestion(0); - submitInput([ - { - role: "user", - content: [{ type: "input_text", text: suggestion }], - type: "message", - }, - ]); - } - } else if (_input === "\u0003" || (_input === "c" && _key.ctrl)) { - setTimeout(() => { - app.exit(); - onExit(); - process.exit(0); - }, 60); - } - }, - { isActive: active }, - ); - - const onSubmit = useCallback( - async (value: string) => { - const inputValue = value.trim(); - - // If the user only entered a slash, do not send a chat message. - if (inputValue === "/") { - setInput(""); - return; - } - - // Skip this submit if we just autocompleted a slash command. - if (skipNextSubmit) { - setSkipNextSubmit(false); - return; - } - - if (!inputValue) { - return; - } else if (inputValue === "/history") { - setInput(""); - openOverlay(); - return; - } else if (inputValue === "/sessions") { - setInput(""); - openSessionsOverlay(); - return; - } else if (inputValue === "/help") { - setInput(""); - openHelpOverlay(); - return; - } else if (inputValue === "/diff") { - setInput(""); - openDiffOverlay(); - return; - } else if (inputValue === "/compact") { - setInput(""); - onCompact(); - return; - } else if (inputValue.startsWith("/model")) { - setInput(""); - openModelOverlay(); - return; - } else if (inputValue.startsWith("/approval")) { - setInput(""); - openApprovalOverlay(); - return; - } else if (["exit", "q", ":q"].includes(inputValue)) { - setInput(""); - setTimeout(() => { - app.exit(); - onExit(); - process.exit(0); - }, 60); // Wait one frame. - return; - } else if (inputValue === "/clear" || inputValue === "clear") { - setInput(""); - setSessionId(""); - setLastResponseId(""); - - // Clear the terminal screen (including scrollback) before resetting context. - clearTerminal(); - - // Emit a system message to confirm the clear action. We *append* - // it so Ink's treats it as new output and actually renders it. - setItems((prev) => { - const filteredOldItems = prev.filter((item) => { - // Remove any token‑heavy entries (user/assistant turns and function calls) - if ( - item.type === "message" && - (item.role === "user" || item.role === "assistant") - ) { - return false; - } - if ( - item.type === "function_call" || - item.type === "function_call_output" - ) { - return false; - } - return true; // keep developer/system and other meta entries - }); - - return [ - ...filteredOldItems, - { - id: `clear-${Date.now()}`, - type: "message", - role: "system", - content: [{ type: "input_text", text: "Terminal cleared" }], - }, - ]; - }); - - return; - } else if (inputValue === "/clearhistory") { - setInput(""); - - // Import clearCommandHistory function to avoid circular dependencies - // Using dynamic import to lazy-load the function - import("../../utils/storage/command-history.js").then( - async ({ clearCommandHistory }) => { - await clearCommandHistory(); - setHistory([]); - - // Emit a system message to confirm the history clear action. - setItems((prev) => [ - ...prev, - { - id: `clearhistory-${Date.now()}`, - type: "message", - role: "system", - content: [ - { type: "input_text", text: "Command history cleared" }, - ], - }, - ]); - }, - ); - - return; - } else if (inputValue === "/bug") { - // Generate a GitHub bug report URL pre‑filled with session details. - setInput(""); - - try { - const os = await import("node:os"); - const { CLI_VERSION } = await import("../../version.js"); - const { buildBugReportUrl } = await import( - "../../utils/bug-report.js" - ); - - const url = buildBugReportUrl({ - items: items ?? [], - cliVersion: CLI_VERSION, - model: loadConfig().model ?? "unknown", - platform: [os.platform(), os.arch(), os.release()] - .map((s) => `\`${s}\``) - .join(" | "), - }); - - setItems((prev) => [ - ...prev, - { - id: `bugreport-${Date.now()}`, - type: "message", - role: "system", - content: [ - { - type: "input_text", - text: `🔗 Bug report URL: ${url}`, - }, - ], - }, - ]); - } catch (error) { - // If anything went wrong, notify the user. - setItems((prev) => [ - ...prev, - { - id: `bugreport-error-${Date.now()}`, - type: "message", - role: "system", - content: [ - { - type: "input_text", - text: `⚠️ Failed to create bug report URL: ${error}`, - }, - ], - }, - ]); - } - - return; - } else if (inputValue.startsWith("/")) { - // Handle invalid/unrecognized commands. Only single-word inputs starting with '/' - // (e.g., /command) that are not recognized are caught here. Any other input, including - // those starting with '/' but containing spaces (e.g., "/command arg"), will fall through - // and be treated as a regular prompt. - const trimmed = inputValue.trim(); - - if (/^\/\S+$/.test(trimmed)) { - setInput(""); - setItems((prev) => [ - ...prev, - { - id: `invalidcommand-${Date.now()}`, - type: "message", - role: "system", - content: [ - { - type: "input_text", - text: `Invalid command "${trimmed}". Use /help to retrieve the list of commands.`, - }, - ], - }, - ]); - - return; - } - } - - // detect image file paths for dynamic inclusion - const images: Array = []; - let text = inputValue; - - // markdown-style image syntax: ![alt](path) - text = text.replace(/!\[[^\]]*?\]\(([^)]+)\)/g, (_m, p1: string) => { - images.push(p1.startsWith("file://") ? fileURLToPath(p1) : p1); - return ""; - }); - - // quoted file paths ending with common image extensions (e.g. '/path/to/img.png') - text = text.replace( - /['"]([^'"]+?\.(?:png|jpe?g|gif|bmp|webp|svg))['"]/gi, - (_m, p1: string) => { - images.push(p1.startsWith("file://") ? fileURLToPath(p1) : p1); - return ""; - }, - ); - - // bare file paths ending with common image extensions - text = text.replace( - // eslint-disable-next-line no-useless-escape - /\b(?:\.[\/\\]|[\/\\]|[A-Za-z]:[\/\\])?[\w-]+(?:[\/\\][\w-]+)*\.(?:png|jpe?g|gif|bmp|webp|svg)\b/gi, - (match: string) => { - images.push( - match.startsWith("file://") ? fileURLToPath(match) : match, - ); - return ""; - }, - ); - text = text.trim(); - - // Expand @file tokens into XML blocks for the model - const expandedText = await expandFileTags(text); - - const inputItem = await createInputItem(expandedText, images); - submitInput([inputItem]); - - // Get config for history persistence. - const config = loadConfig(); - - // Add to history and update state. - const updatedHistory = await addToHistory(value, history, { - maxSize: config.history?.maxSize ?? 1000, - saveHistory: config.history?.saveHistory ?? true, - sensitivePatterns: config.history?.sensitivePatterns ?? [], - }); - - setHistory(updatedHistory); - setHistoryIndex(null); - setDraftInput(""); - setSelectedSuggestion(0); - setInput(""); - setFsSuggestions([]); - setSelectedCompletion(-1); - }, - [ - setInput, - submitInput, - setLastResponseId, - setItems, - app, - setHistory, - setHistoryIndex, - openOverlay, - openApprovalOverlay, - openModelOverlay, - openHelpOverlay, - openDiffOverlay, - openSessionsOverlay, - history, - onCompact, - skipNextSubmit, - items, - ], - ); - - if (confirmationPrompt) { - return ( - - ); - } - - return ( - - - {loading ? ( - - ) : ( - - { - setDraftInput(txt); - if (historyIndex != null) { - setHistoryIndex(null); - } - setInput(txt); - }} - key={editorState.key} - initialCursorOffset={editorState.initialCursorOffset} - initialText={input} - height={6} - focus={active} - onSubmit={(txt) => { - // If final token is an @path, replace with filesystem suggestion if available - const { - text: replacedText, - suggestion, - wasReplaced, - } = getFileSystemSuggestion(txt, true); - - // If we replaced @path token with a directory, don't submit - if (wasReplaced && suggestion?.isDirectory) { - applyFsSuggestion(replacedText); - // Update suggestions for the new directory - updateFsSuggestions(replacedText, true); - return; - } - - onSubmit(replacedText); - setEditorState((s) => ({ key: s.key + 1 })); - setInput(""); - setHistoryIndex(null); - setDraftInput(""); - }} - /> - - )} - - {/* Slash command autocomplete suggestions */} - {input.trim().startsWith("/") && ( - - {SLASH_COMMANDS.filter((cmd: SlashCommand) => - cmd.command.startsWith(input.trim()), - ).map((cmd: SlashCommand, idx: number) => ( - - - {cmd.command} - {cmd.description} - - - ))} - - )} - - {isNew && !input ? ( - - try:{" "} - {suggestions.map((m, key) => ( - - {key !== 0 ? " | " : ""} - - {m} - - - ))} - - ) : fsSuggestions.length > 0 ? ( - suggestion.path)} - selectedCompletion={selectedCompletion} - displayLimit={5} - /> - ) : ( - - Ctrl+C to exit | "/" to see commands | Enter to send - {contextLeftPercent > 25 && ( - <> - {" — "} - 40 ? "green" : "yellow"}> - {Math.round(contextLeftPercent)}% context left - - - )} - {contextLeftPercent <= 25 && ( - <> - {" — "} - - {Math.round(contextLeftPercent)}% context left — send - "/compact" to condense context - - - )} - - )} - - - ); -} - -function TerminalChatInputThinking({ - onInterrupt, - active, - thinkingSeconds, -}: { - onInterrupt: () => void; - active: boolean; - thinkingSeconds: number; -}) { - const [awaitingConfirm, setAwaitingConfirm] = useState(false); - const [dots, setDots] = useState(""); - - // Animate ellipsis - useInterval(() => { - setDots((prev) => (prev.length < 3 ? prev + "." : "")); - }, 500); - - // Spinner frames with embedded seconds - const ballFrames = [ - "( ● )", - "( ● )", - "( ● )", - "( ● )", - "( ●)", - "( ● )", - "( ● )", - "( ● )", - "( ● )", - "(● )", - ]; - const [frame, setFrame] = useState(0); - - useInterval(() => { - setFrame((idx) => (idx + 1) % ballFrames.length); - }, 80); - - // Keep the elapsed‑seconds text fixed while the ball animation moves. - const frameTemplate = ballFrames[frame] ?? ballFrames[0]; - const frameWithSeconds = `${frameTemplate} ${thinkingSeconds}s`; - - // --------------------------------------------------------------------- - // Raw stdin listener to catch the case where the terminal delivers two - // consecutive ESC bytes ("\x1B\x1B") in a *single* chunk. Ink's `useInput` - // collapses that sequence into one key event, so the regular two‑step - // handler above never sees the second press. By inspecting the raw data - // we can identify this special case and trigger the interrupt while still - // requiring a double press for the normal single‑byte ESC events. - // --------------------------------------------------------------------- - - const { stdin, setRawMode } = useStdin(); - - React.useEffect(() => { - if (!active) { - return; - } - - // Ensure raw mode – already enabled by Ink when the component has focus, - // but called defensively in case that assumption ever changes. - setRawMode?.(true); - - const onData = (data: Buffer | string) => { - if (awaitingConfirm) { - return; // already awaiting a second explicit press - } - - // Handle both Buffer and string forms. - const str = Buffer.isBuffer(data) ? data.toString("utf8") : data; - if (str === "\x1b\x1b") { - // Treat as the first Escape press – prompt the user for confirmation. - log( - "raw stdin: received collapsed ESC ESC – starting confirmation timer", - ); - setAwaitingConfirm(true); - setTimeout(() => setAwaitingConfirm(false), 1500); - } - }; - - stdin?.on("data", onData); - - return () => { - stdin?.off("data", onData); - }; - }, [stdin, awaitingConfirm, onInterrupt, active, setRawMode]); - - // No local timer: the parent component supplies the elapsed time via props. - - // Listen for the escape key to allow the user to interrupt the current - // operation. We require two presses within a short window (1.5s) to avoid - // accidental cancellations. - useInput( - (_input, key) => { - if (!key.escape) { - return; - } - - if (awaitingConfirm) { - log("useInput: second ESC detected – triggering onInterrupt()"); - onInterrupt(); - setAwaitingConfirm(false); - } else { - log("useInput: first ESC detected – waiting for confirmation"); - setAwaitingConfirm(true); - setTimeout(() => setAwaitingConfirm(false), 1500); - } - }, - { isActive: active }, - ); - - return ( - - - - {frameWithSeconds} - - Thinking - {dots} - - - - press Esc{" "} - {awaitingConfirm ? ( - again - ) : ( - twice - )}{" "} - to interrupt - - - - ); -} diff --git a/codex-cli/src/components/chat/terminal-chat-past-rollout.tsx b/codex-cli/src/components/chat/terminal-chat-past-rollout.tsx deleted file mode 100644 index 1ac8280e..00000000 --- a/codex-cli/src/components/chat/terminal-chat-past-rollout.tsx +++ /dev/null @@ -1,68 +0,0 @@ -import type { TerminalChatSession } from "../../utils/session.js"; -import type { ResponseItem } from "openai/resources/responses/responses"; -import type { FileOpenerScheme } from "src/utils/config.js"; - -import TerminalChatResponseItem from "./terminal-chat-response-item"; -import { Box, Text } from "ink"; -import React from "react"; - -export default function TerminalChatPastRollout({ - session, - items, - fileOpener, -}: { - session: TerminalChatSession; - items: Array; - fileOpener: FileOpenerScheme | undefined; -}): React.ReactElement { - const { version, id: sessionId, model } = session; - return ( - - - - ● OpenAI Codex{" "} - - (research preview) v{version} - - - - - - localhost{" "} - · session:{" "} - - {sessionId} - - - - When / Who:{" "} - - {session.timestamp} / {session.user} - - - - model: {model} - - - - {React.useMemo( - () => - items.map((item, key) => ( - - )), - [items, fileOpener], - )} - - - ); -} diff --git a/codex-cli/src/components/chat/terminal-chat-response-item.tsx b/codex-cli/src/components/chat/terminal-chat-response-item.tsx deleted file mode 100644 index bab4aa31..00000000 --- a/codex-cli/src/components/chat/terminal-chat-response-item.tsx +++ /dev/null @@ -1,360 +0,0 @@ -import type { OverlayModeType } from "./terminal-chat"; -import type { TerminalRendererOptions } from "marked-terminal"; -import type { - ResponseFunctionToolCallItem, - ResponseFunctionToolCallOutputItem, - ResponseInputMessageItem, - ResponseItem, - ResponseOutputMessage, - ResponseReasoningItem, -} from "openai/resources/responses/responses"; -import type { FileOpenerScheme } from "src/utils/config"; - -import { useTerminalSize } from "../../hooks/use-terminal-size"; -import { collapseXmlBlocks } from "../../utils/file-tag-utils"; -import { parseToolCall, parseToolCallOutput } from "../../utils/parsers"; -import chalk, { type ForegroundColorName } from "chalk"; -import { Box, Text } from "ink"; -import { parse, setOptions } from "marked"; -import TerminalRenderer from "marked-terminal"; -import path from "path"; -import React, { useEffect, useMemo } from "react"; -import { formatCommandForDisplay } from "src/format-command.js"; -import supportsHyperlinks from "supports-hyperlinks"; - -export default function TerminalChatResponseItem({ - item, - fullStdout = false, - setOverlayMode, - fileOpener, -}: { - item: ResponseItem; - fullStdout?: boolean; - setOverlayMode?: React.Dispatch>; - fileOpener: FileOpenerScheme | undefined; -}): React.ReactElement { - switch (item.type) { - case "message": - return ( - - ); - // @ts-expect-error new item types aren't in SDK yet - case "local_shell_call": - case "function_call": - return ; - // @ts-expect-error new item types aren't in SDK yet - case "local_shell_call_output": - case "function_call_output": - return ( - - ); - default: - break; - } - - // @ts-expect-error `reasoning` is not in the responses API yet - if (item.type === "reasoning") { - return ( - - ); - } - - return ; -} - -// TODO: this should be part of `ResponseReasoningItem`. Also it doesn't work. -// --------------------------------------------------------------------------- -// Utility helpers -// --------------------------------------------------------------------------- - -/** - * Guess how long the assistant spent "thinking" based on the combined length - * of the reasoning summary. The calculation itself is fast, but wrapping it in - * `useMemo` in the consuming component ensures it only runs when the - * `summary` array actually changes. - */ -// TODO: use actual thinking time -// -// function guessThinkingTime(summary: Array) { -// const totalTextLength = summary -// .map((t) => t.text.length) -// .reduce((a, b) => a + b, summary.length - 1); -// return Math.max(1, Math.ceil(totalTextLength / 300)); -// } - -export function TerminalChatResponseReasoning({ - message, - fileOpener, -}: { - message: ResponseReasoningItem & { duration_ms?: number }; - fileOpener: FileOpenerScheme | undefined; -}): React.ReactElement | null { - // Only render when there is a reasoning summary - if (!message.summary || message.summary.length === 0) { - return null; - } - return ( - - {message.summary.map((summary, key) => { - const s = summary as { headline?: string; text: string }; - return ( - - {s.headline && {s.headline}} - {s.text} - - ); - })} - - ); -} - -const colorsByRole: Record = { - assistant: "magentaBright", - user: "blueBright", -}; - -function TerminalChatResponseMessage({ - message, - setOverlayMode, - fileOpener, -}: { - message: ResponseInputMessageItem | ResponseOutputMessage; - setOverlayMode?: React.Dispatch>; - fileOpener: FileOpenerScheme | undefined; -}) { - // auto switch to model mode if the system message contains "has been deprecated" - useEffect(() => { - if (message.role === "system") { - const systemMessage = message.content.find( - (c) => c.type === "input_text", - )?.text; - if (systemMessage?.includes("model_not_found")) { - setOverlayMode?.("model"); - } - } - }, [message, setOverlayMode]); - - return ( - - - {message.role === "assistant" ? "codex" : message.role} - - - {message.content - .map( - (c) => - c.type === "output_text" - ? c.text - : c.type === "refusal" - ? c.refusal - : c.type === "input_text" - ? collapseXmlBlocks(c.text) - : c.type === "input_image" - ? "" - : c.type === "input_file" - ? c.filename - : "", // unknown content type - ) - .join(" ")} - - - ); -} - -function TerminalChatResponseToolCall({ - message, -}: { - // eslint-disable-next-line @typescript-eslint/no-explicit-any - message: ResponseFunctionToolCallItem | any; -}) { - let workdir: string | undefined; - let cmdReadableText: string | undefined; - if (message.type === "function_call") { - const details = parseToolCall(message); - workdir = details?.workdir; - cmdReadableText = details?.cmdReadableText; - } else if (message.type === "local_shell_call") { - const action = message.action; - workdir = action.working_directory; - cmdReadableText = formatCommandForDisplay(action.command); - } - return ( - - - command - {workdir ? {` (${workdir})`} : ""} - - - $ {cmdReadableText} - - - ); -} - -function TerminalChatResponseToolCallOutput({ - message, - fullStdout, -}: { - // eslint-disable-next-line @typescript-eslint/no-explicit-any - message: ResponseFunctionToolCallOutputItem | any; - fullStdout: boolean; -}) { - const { output, metadata } = parseToolCallOutput(message.output); - const { exit_code, duration_seconds } = metadata; - const metadataInfo = useMemo( - () => - [ - typeof exit_code !== "undefined" ? `code: ${exit_code}` : "", - typeof duration_seconds !== "undefined" - ? `duration: ${duration_seconds}s` - : "", - ] - .filter(Boolean) - .join(", "), - [exit_code, duration_seconds], - ); - let displayedContent = output; - if (message.type === "function_call_output" && !fullStdout) { - const lines = displayedContent.split("\n"); - if (lines.length > 4) { - const head = lines.slice(0, 4); - const remaining = lines.length - 4; - displayedContent = [...head, `... (${remaining} more lines)`].join("\n"); - } - } - - // ------------------------------------------------------------------------- - // Colorize diff output: lines starting with '-' in red, '+' in green. - // This makes patches and other diff‑like stdout easier to read. - // We exclude the typical diff file headers ('---', '+++') so they retain - // the default color. This is a best‑effort heuristic and should be safe for - // non‑diff output – only the very first character of a line is inspected. - // ------------------------------------------------------------------------- - const colorizedContent = displayedContent - .split("\n") - .map((line) => { - if (line.startsWith("+") && !line.startsWith("++")) { - return chalk.green(line); - } - if (line.startsWith("-") && !line.startsWith("--")) { - return chalk.red(line); - } - return line; - }) - .join("\n"); - return ( - - - command.stdout{" "} - {metadataInfo ? `(${metadataInfo})` : ""} - - {colorizedContent} - - ); -} - -export function TerminalChatResponseGenericMessage({ - message, -}: { - message: ResponseItem; -}): React.ReactElement { - return {JSON.stringify(message, null, 2)}; -} - -export type MarkdownProps = TerminalRendererOptions & { - children: string; - fileOpener: FileOpenerScheme | undefined; - /** Base path for resolving relative file citation paths. */ - cwd?: string; -}; - -export function Markdown({ - children, - fileOpener, - cwd, - ...options -}: MarkdownProps): React.ReactElement { - const size = useTerminalSize(); - - const rendered = React.useMemo(() => { - const linkifiedMarkdown = rewriteFileCitations(children, fileOpener, cwd); - - // Configure marked for this specific render - setOptions({ - // @ts-expect-error missing parser, space props - renderer: new TerminalRenderer({ ...options, width: size.columns }), - }); - const parsed = parse(linkifiedMarkdown, { async: false }).trim(); - - // Remove the truncation logic - return parsed; - // eslint-disable-next-line react-hooks/exhaustive-deps -- options is an object of primitives - }, [ - children, - size.columns, - size.rows, - fileOpener, - supportsHyperlinks.stdout, - chalk.level, - ]); - - return {rendered}; -} - -/** Regex to match citations for source files (hence the `F:` prefix). */ -const citationRegex = new RegExp( - [ - // Opening marker - "【", - - // Capture group 1: file ID or name (anything except '†') - "F:([^†]+)", - - // Field separator - "†", - - // Capture group 2: start line (digits) - "L(\\d+)", - - // Non-capturing group for optional end line - "(?:", - - // Capture group 3: end line (digits or '?') - "-L(\\d+|\\?)", - - // End of optional group (may not be present) - ")?", - - // Closing marker - "】", - ].join(""), - "g", // Global flag -); - -function rewriteFileCitations( - markdown: string, - fileOpener: FileOpenerScheme | undefined, - cwd: string = process.cwd(), -): string { - citationRegex.lastIndex = 0; - return markdown.replace(citationRegex, (_match, file, start, _end) => { - const absPath = path.resolve(cwd, file); - if (!fileOpener) { - return `[${file}](${absPath})`; - } - const uri = `${fileOpener}://file${absPath}:${start}`; - const label = `${file}:${start}`; - // In practice, sometimes multiple citations for the same file, but with a - // different line number, are shown sequentially, so we: - // - include the line number in the label to disambiguate them - // - add a space after the link to make it easier to read - return `[${label}](${uri}) `; - }); -} diff --git a/codex-cli/src/components/chat/terminal-chat-tool-call-command.tsx b/codex-cli/src/components/chat/terminal-chat-tool-call-command.tsx deleted file mode 100644 index 614ebf38..00000000 --- a/codex-cli/src/components/chat/terminal-chat-tool-call-command.tsx +++ /dev/null @@ -1,143 +0,0 @@ -import { parseApplyPatch } from "../../parse-apply-patch"; -import { shortenPath } from "../../utils/short-path"; -import chalk from "chalk"; -import { Text } from "ink"; -import React from "react"; - -export function TerminalChatToolCallCommand({ - commandForDisplay, - explanation, -}: { - commandForDisplay: string; - explanation?: string; -}): React.ReactElement { - // ------------------------------------------------------------------------- - // Colorize diff output inside the command preview: we detect individual - // lines that begin with '+' or '-' (excluding the typical diff headers like - // '+++', '---', '++', '--') and apply green/red coloring. This mirrors - // how Git shows diffs and makes the patch easier to review. - // ------------------------------------------------------------------------- - - const colorizedCommand = commandForDisplay - .split("\n") - .map((line) => { - if (line.startsWith("+") && !line.startsWith("++")) { - return chalk.green(line); - } - if (line.startsWith("-") && !line.startsWith("--")) { - return chalk.red(line); - } - return line; - }) - .join("\n"); - - return ( - <> - - Shell Command - - - $ {colorizedCommand} - - {explanation && ( - <> - - Explanation - - {explanation.split("\n").map((line, i) => { - // Apply different styling to headings (numbered items) - if (line.match(/^\d+\.\s+/)) { - return ( - - {line} - - ); - } else if (line.match(/^\s*\*\s+/)) { - // Style bullet points - return ( - - {line} - - ); - } else if (line.match(/^(WARNING|CAUTION|NOTE):/i)) { - // Style warnings - return ( - - {line} - - ); - } else { - return {line}; - } - })} - - )} - - ); -} - -export function TerminalChatToolCallApplyPatch({ - commandForDisplay, - patch, -}: { - commandForDisplay: string; - patch: string; -}): React.ReactElement { - const ops = React.useMemo(() => parseApplyPatch(patch), [patch]); - const firstOp = ops?.[0]; - - const title = React.useMemo(() => { - if (!firstOp) { - return ""; - } - return capitalize(firstOp.type); - }, [firstOp]); - - const filePath = React.useMemo(() => { - if (!firstOp) { - return ""; - } - return shortenPath(firstOp.path || "."); - }, [firstOp]); - - if (ops == null) { - return ( - <> - - Invalid Patch - - - The provided patch command is invalid. - - {commandForDisplay} - - ); - } - - if (!firstOp) { - return ( - <> - - Empty Patch - - - No operations found in the patch command. - - {commandForDisplay} - - ); - } - - return ( - <> - - {title} {filePath} - - - $ {commandForDisplay} - - - ); -} - -const capitalize = (s: string) => s.charAt(0).toUpperCase() + s.slice(1); diff --git a/codex-cli/src/components/chat/terminal-chat.tsx b/codex-cli/src/components/chat/terminal-chat.tsx deleted file mode 100644 index d41a9499..00000000 --- a/codex-cli/src/components/chat/terminal-chat.tsx +++ /dev/null @@ -1,766 +0,0 @@ -import type { AppRollout } from "../../app.js"; -import type { ApplyPatchCommand, ApprovalPolicy } from "../../approvals.js"; -import type { CommandConfirmation } from "../../utils/agent/agent-loop.js"; -import type { AppConfig } from "../../utils/config.js"; -import type { ColorName } from "chalk"; -import type { ResponseItem } from "openai/resources/responses/responses.mjs"; - -import TerminalChatInput from "./terminal-chat-input.js"; -import TerminalChatPastRollout from "./terminal-chat-past-rollout.js"; -import { TerminalChatToolCallCommand } from "./terminal-chat-tool-call-command.js"; -import TerminalMessageHistory from "./terminal-message-history.js"; -import { formatCommandForDisplay } from "../../format-command.js"; -import { useConfirmation } from "../../hooks/use-confirmation.js"; -import { useTerminalSize } from "../../hooks/use-terminal-size.js"; -import { AgentLoop } from "../../utils/agent/agent-loop.js"; -import { ReviewDecision } from "../../utils/agent/review.js"; -import { generateCompactSummary } from "../../utils/compact-summary.js"; -import { saveConfig } from "../../utils/config.js"; -import { extractAppliedPatches as _extractAppliedPatches } from "../../utils/extract-applied-patches.js"; -import { getGitDiff } from "../../utils/get-diff.js"; -import { createInputItem } from "../../utils/input-utils.js"; -import { log } from "../../utils/logger/log.js"; -import { - getAvailableModels, - calculateContextPercentRemaining, - uniqueById, -} from "../../utils/model-utils.js"; -import { createOpenAIClient } from "../../utils/openai-client.js"; -import { shortCwd } from "../../utils/short-path.js"; -import { saveRollout } from "../../utils/storage/save-rollout.js"; -import { CLI_VERSION } from "../../version.js"; -import ApprovalModeOverlay from "../approval-mode-overlay.js"; -import DiffOverlay from "../diff-overlay.js"; -import HelpOverlay from "../help-overlay.js"; -import HistoryOverlay from "../history-overlay.js"; -import ModelOverlay from "../model-overlay.js"; -import SessionsOverlay from "../sessions-overlay.js"; -import chalk from "chalk"; -import fs from "fs/promises"; -import { Box, Text } from "ink"; -import { spawn } from "node:child_process"; -import React, { useEffect, useMemo, useRef, useState } from "react"; -import { inspect } from "util"; - -export type OverlayModeType = - | "none" - | "history" - | "sessions" - | "model" - | "approval" - | "help" - | "diff"; - -type Props = { - config: AppConfig; - prompt?: string; - imagePaths?: Array; - approvalPolicy: ApprovalPolicy; - additionalWritableRoots: ReadonlyArray; - fullStdout: boolean; -}; - -const colorsByPolicy: Record = { - "suggest": undefined, - "auto-edit": "greenBright", - "full-auto": "green", -}; - -/** - * Generates an explanation for a shell command using the OpenAI API. - * - * @param command The command to explain - * @param model The model to use for generating the explanation - * @param flexMode Whether to use the flex-mode service tier - * @param config The configuration object - * @returns A human-readable explanation of what the command does - */ -async function generateCommandExplanation( - command: Array, - model: string, - flexMode: boolean, - config: AppConfig, -): Promise { - try { - // Create a temporary OpenAI client - const oai = createOpenAIClient(config); - - // Format the command for display - const commandForDisplay = formatCommandForDisplay(command); - - // Create a prompt that asks for an explanation with a more detailed system prompt - const response = await oai.chat.completions.create({ - model, - ...(flexMode ? { service_tier: "flex" } : {}), - messages: [ - { - role: "system", - content: - "You are an expert in shell commands and terminal operations. Your task is to provide detailed, accurate explanations of shell commands that users are considering executing. Break down each part of the command, explain what it does, identify any potential risks or side effects, and explain why someone might want to run it. Be specific about what files or systems will be affected. If the command could potentially be harmful, make sure to clearly highlight those risks.", - }, - { - role: "user", - content: `Please explain this shell command in detail: \`${commandForDisplay}\`\n\nProvide a structured explanation that includes:\n1. A brief overview of what the command does\n2. A breakdown of each part of the command (flags, arguments, etc.)\n3. What files, directories, or systems will be affected\n4. Any potential risks or side effects\n5. Why someone might want to run this command\n\nBe specific and technical - this explanation will help the user decide whether to approve or reject the command.`, - }, - ], - }); - - // Extract the explanation from the response - const explanation = - response.choices[0]?.message.content || "Unable to generate explanation."; - return explanation; - } catch (error) { - log(`Error generating command explanation: ${error}`); - - let errorMessage = "Unable to generate explanation due to an error."; - if (error instanceof Error) { - errorMessage = `Unable to generate explanation: ${error.message}`; - - // If it's an API error, check for more specific information - if ("status" in error && typeof error.status === "number") { - // Handle API-specific errors - if (error.status === 401) { - errorMessage = - "Unable to generate explanation: API key is invalid or expired."; - } else if (error.status === 429) { - errorMessage = - "Unable to generate explanation: Rate limit exceeded. Please try again later."; - } else if (error.status >= 500) { - errorMessage = - "Unable to generate explanation: OpenAI service is currently unavailable. Please try again later."; - } - } - } - - return errorMessage; - } -} - -export default function TerminalChat({ - config, - prompt: _initialPrompt, - imagePaths: _initialImagePaths, - approvalPolicy: initialApprovalPolicy, - additionalWritableRoots, - fullStdout, -}: Props): React.ReactElement { - const notify = Boolean(config.notify); - const [model, setModel] = useState(config.model); - const [provider, setProvider] = useState(config.provider || "openai"); - const [lastResponseId, setLastResponseId] = useState(null); - const [items, setItems] = useState>([]); - const [loading, setLoading] = useState(false); - const [approvalPolicy, setApprovalPolicy] = useState( - initialApprovalPolicy, - ); - const [thinkingSeconds, setThinkingSeconds] = useState(0); - - const handleCompact = async () => { - setLoading(true); - try { - const summary = await generateCompactSummary( - items, - model, - Boolean(config.flexMode), - config, - ); - setItems([ - { - id: `compact-${Date.now()}`, - type: "message", - role: "assistant", - content: [{ type: "output_text", text: summary }], - } as ResponseItem, - ]); - } catch (err) { - setItems((prev) => [ - ...prev, - { - id: `compact-error-${Date.now()}`, - type: "message", - role: "system", - content: [ - { type: "input_text", text: `Failed to compact context: ${err}` }, - ], - } as ResponseItem, - ]); - } finally { - setLoading(false); - } - }; - - const { - requestConfirmation, - confirmationPrompt, - explanation, - submitConfirmation, - } = useConfirmation(); - const [overlayMode, setOverlayMode] = useState("none"); - const [viewRollout, setViewRollout] = useState(null); - - // Store the diff text when opening the diff overlay so the view isn’t - // recomputed on every re‑render while it is open. - // diffText is passed down to the DiffOverlay component. The setter is - // currently unused but retained for potential future updates. Prefix with - // an underscore so eslint ignores the unused variable. - const [diffText, _setDiffText] = useState(""); - - const [initialPrompt, setInitialPrompt] = useState(_initialPrompt); - const [initialImagePaths, setInitialImagePaths] = - useState(_initialImagePaths); - - const PWD = React.useMemo(() => shortCwd(), []); - - // Keep a single AgentLoop instance alive across renders; - // recreate only when model/instructions/approvalPolicy change. - const agentRef = React.useRef(); - const [, forceUpdate] = React.useReducer((c) => c + 1, 0); // trigger re‑render - - // ──────────────────────────────────────────────────────────────── - // DEBUG: log every render w/ key bits of state - // ──────────────────────────────────────────────────────────────── - log( - `render - agent? ${Boolean(agentRef.current)} loading=${loading} items=${ - items.length - }`, - ); - - useEffect(() => { - // Skip recreating the agent if awaiting a decision on a pending confirmation. - if (confirmationPrompt != null) { - log("skip AgentLoop recreation due to pending confirmationPrompt"); - return; - } - - log("creating NEW AgentLoop"); - log( - `model=${model} provider=${provider} instructions=${Boolean( - config.instructions, - )} approvalPolicy=${approvalPolicy}`, - ); - - // Tear down any existing loop before creating a new one. - agentRef.current?.terminate(); - - const sessionId = crypto.randomUUID(); - agentRef.current = new AgentLoop({ - model, - provider, - config, - instructions: config.instructions, - approvalPolicy, - disableResponseStorage: config.disableResponseStorage, - additionalWritableRoots, - onLastResponseId: setLastResponseId, - onItem: (item) => { - log(`onItem: ${JSON.stringify(item)}`); - setItems((prev) => { - const updated = uniqueById([...prev, item as ResponseItem]); - saveRollout(sessionId, updated); - return updated; - }); - }, - onLoading: setLoading, - getCommandConfirmation: async ( - command: Array, - applyPatch: ApplyPatchCommand | undefined, - ): Promise => { - log(`getCommandConfirmation: ${command}`); - const commandForDisplay = formatCommandForDisplay(command); - - // First request for confirmation - let { decision: review, customDenyMessage } = await requestConfirmation( - , - ); - - // If the user wants an explanation, generate one and ask again. - if (review === ReviewDecision.EXPLAIN) { - log(`Generating explanation for command: ${commandForDisplay}`); - const explanation = await generateCommandExplanation( - command, - model, - Boolean(config.flexMode), - config, - ); - log(`Generated explanation: ${explanation}`); - - // Ask for confirmation again, but with the explanation. - const confirmResult = await requestConfirmation( - , - ); - - // Update the decision based on the second confirmation. - review = confirmResult.decision; - customDenyMessage = confirmResult.customDenyMessage; - - // Return the final decision with the explanation. - return { review, customDenyMessage, applyPatch, explanation }; - } - - return { review, customDenyMessage, applyPatch }; - }, - }); - - // Force a render so JSX below can "see" the freshly created agent. - forceUpdate(); - - log(`AgentLoop created: ${inspect(agentRef.current, { depth: 1 })}`); - - return () => { - log("terminating AgentLoop"); - agentRef.current?.terminate(); - agentRef.current = undefined; - forceUpdate(); // re‑render after teardown too - }; - // We intentionally omit 'approvalPolicy' and 'confirmationPrompt' from the deps - // so switching modes or showing confirmation dialogs doesn’t tear down the loop. - // eslint-disable-next-line react-hooks/exhaustive-deps - }, [model, provider, config, requestConfirmation, additionalWritableRoots]); - - // Whenever loading starts/stops, reset or start a timer — but pause the - // timer while a confirmation overlay is displayed so we don't trigger a - // re‑render every second during apply_patch reviews. - useEffect(() => { - let handle: ReturnType | null = null; - // Only tick the "thinking…" timer when the agent is actually processing - // a request *and* the user is not being asked to review a command. - if (loading && confirmationPrompt == null) { - setThinkingSeconds(0); - handle = setInterval(() => { - setThinkingSeconds((s) => s + 1); - }, 1000); - } else { - if (handle) { - clearInterval(handle); - } - setThinkingSeconds(0); - } - return () => { - if (handle) { - clearInterval(handle); - } - }; - }, [loading, confirmationPrompt]); - - // Notify desktop with a preview when an assistant response arrives. - const prevLoadingRef = useRef(false); - useEffect(() => { - // Only notify when notifications are enabled. - if (!notify) { - prevLoadingRef.current = loading; - return; - } - - if ( - prevLoadingRef.current && - !loading && - confirmationPrompt == null && - items.length > 0 - ) { - if (process.platform === "darwin") { - // find the last assistant message - const assistantMessages = items.filter( - (i) => i.type === "message" && i.role === "assistant", - ); - const last = assistantMessages[assistantMessages.length - 1]; - if (last) { - const text = last.content - .map((c) => { - if (c.type === "output_text") { - return c.text; - } - return ""; - }) - .join("") - .trim(); - const preview = text.replace(/\n/g, " ").slice(0, 100); - const safePreview = preview.replace(/"/g, '\\"'); - const title = "Codex CLI"; - const cwd = PWD; - spawn("osascript", [ - "-e", - `display notification "${safePreview}" with title "${title}" subtitle "${cwd}" sound name "Ping"`, - ]); - } - } - } - prevLoadingRef.current = loading; - }, [notify, loading, confirmationPrompt, items, PWD]); - - // Let's also track whenever the ref becomes available. - const agent = agentRef.current; - useEffect(() => { - log(`agentRef.current is now ${Boolean(agent)}`); - }, [agent]); - - // --------------------------------------------------------------------- - // Dynamic layout constraints – keep total rendered rows <= terminal rows - // --------------------------------------------------------------------- - - const { rows: terminalRows } = useTerminalSize(); - - useEffect(() => { - const processInitialInputItems = async () => { - if ( - (!initialPrompt || initialPrompt.trim() === "") && - (!initialImagePaths || initialImagePaths.length === 0) - ) { - return; - } - const inputItems = [ - await createInputItem(initialPrompt || "", initialImagePaths || []), - ]; - // Clear them to prevent subsequent runs. - setInitialPrompt(""); - setInitialImagePaths([]); - agent?.run(inputItems); - }; - processInitialInputItems(); - }, [agent, initialPrompt, initialImagePaths]); - - // ──────────────────────────────────────────────────────────────── - // In-app warning if CLI --model isn't in fetched list - // ──────────────────────────────────────────────────────────────── - useEffect(() => { - (async () => { - const available = await getAvailableModels(provider); - if (model && available.length > 0 && !available.includes(model)) { - setItems((prev) => [ - ...prev, - { - id: `unknown-model-${Date.now()}`, - type: "message", - role: "system", - content: [ - { - type: "input_text", - text: `Warning: model "${model}" is not in the list of available models for provider "${provider}".`, - }, - ], - }, - ]); - } - })(); - // run once on mount - // eslint-disable-next-line react-hooks/exhaustive-deps - }, []); - - // Just render every item in order, no grouping/collapse. - const lastMessageBatch = items.map((item) => ({ item })); - const groupCounts: Record = {}; - const userMsgCount = items.filter( - (i) => i.type === "message" && i.role === "user", - ).length; - - const contextLeftPercent = useMemo( - () => calculateContextPercentRemaining(items, model), - [items, model], - ); - - if (viewRollout) { - return ( - - ); - } - - return ( - - - {agent ? ( - - ) : ( - - Initializing agent… - - )} - {overlayMode === "none" && agent && ( - - submitConfirmation({ - decision, - customDenyMessage, - }) - } - contextLeftPercent={contextLeftPercent} - openOverlay={() => setOverlayMode("history")} - openModelOverlay={() => setOverlayMode("model")} - openApprovalOverlay={() => setOverlayMode("approval")} - openHelpOverlay={() => setOverlayMode("help")} - openSessionsOverlay={() => setOverlayMode("sessions")} - openDiffOverlay={() => { - const { isGitRepo, diff } = getGitDiff(); - let text: string; - if (isGitRepo) { - text = diff; - } else { - text = "`/diff` — _not inside a git repository_"; - } - setItems((prev) => [ - ...prev, - { - id: `diff-${Date.now()}`, - type: "message", - role: "system", - content: [{ type: "input_text", text }], - }, - ]); - // Ensure no overlay is shown. - setOverlayMode("none"); - }} - onCompact={handleCompact} - active={overlayMode === "none"} - interruptAgent={() => { - if (!agent) { - return; - } - log( - "TerminalChat: interruptAgent invoked – calling agent.cancel()", - ); - agent.cancel(); - setLoading(false); - - // Add a system message to indicate the interruption - setItems((prev) => [ - ...prev, - { - id: `interrupt-${Date.now()}`, - type: "message", - role: "system", - content: [ - { - type: "input_text", - text: "⏹️ Execution interrupted by user. You can continue typing.", - }, - ], - }, - ]); - }} - submitInput={(inputs) => { - agent.run(inputs, lastResponseId || ""); - return {}; - }} - items={items} - thinkingSeconds={thinkingSeconds} - /> - )} - {overlayMode === "history" && ( - setOverlayMode("none")} /> - )} - {overlayMode === "sessions" && ( - { - try { - const txt = await fs.readFile(p, "utf-8"); - const data = JSON.parse(txt) as AppRollout; - setViewRollout(data); - setOverlayMode("none"); - } catch { - setOverlayMode("none"); - } - }} - onResume={(p) => { - setOverlayMode("none"); - setInitialPrompt(`Resume this session: ${p}`); - }} - onExit={() => setOverlayMode("none")} - /> - )} - {overlayMode === "model" && ( - { - log( - "TerminalChat: interruptAgent invoked – calling agent.cancel()", - ); - if (!agent) { - log("TerminalChat: agent is not ready yet"); - } - agent?.cancel(); - setLoading(false); - - if (!allModels?.includes(newModel)) { - // eslint-disable-next-line no-console - console.error( - chalk.bold.red( - `Model "${chalk.yellow( - newModel, - )}" is not available for provider "${chalk.yellow( - provider, - )}".`, - ), - ); - return; - } - - setModel(newModel); - setLastResponseId((prev) => - prev && newModel !== model ? null : prev, - ); - - // Save model to config - saveConfig({ - ...config, - model: newModel, - provider: provider, - }); - - setItems((prev) => [ - ...prev, - { - id: `switch-model-${Date.now()}`, - type: "message", - role: "system", - content: [ - { - type: "input_text", - text: `Switched model to ${newModel}`, - }, - ], - }, - ]); - - setOverlayMode("none"); - }} - onSelectProvider={(newProvider) => { - log( - "TerminalChat: interruptAgent invoked – calling agent.cancel()", - ); - if (!agent) { - log("TerminalChat: agent is not ready yet"); - } - agent?.cancel(); - setLoading(false); - - // Select default model for the new provider. - const defaultModel = model; - - // Save provider to config. - const updatedConfig = { - ...config, - provider: newProvider, - model: defaultModel, - }; - saveConfig(updatedConfig); - - setProvider(newProvider); - setModel(defaultModel); - setLastResponseId((prev) => - prev && newProvider !== provider ? null : prev, - ); - - setItems((prev) => [ - ...prev, - { - id: `switch-provider-${Date.now()}`, - type: "message", - role: "system", - content: [ - { - type: "input_text", - text: `Switched provider to ${newProvider} with model ${defaultModel}`, - }, - ], - }, - ]); - - // Don't close the overlay so user can select a model for the new provider - // setOverlayMode("none"); - }} - onExit={() => setOverlayMode("none")} - /> - )} - - {overlayMode === "approval" && ( - { - // Update approval policy without cancelling an in-progress session. - if (newMode === approvalPolicy) { - return; - } - - setApprovalPolicy(newMode as ApprovalPolicy); - if (agentRef.current) { - ( - agentRef.current as unknown as { - approvalPolicy: ApprovalPolicy; - } - ).approvalPolicy = newMode as ApprovalPolicy; - } - setItems((prev) => [ - ...prev, - { - id: `switch-approval-${Date.now()}`, - type: "message", - role: "system", - content: [ - { - type: "input_text", - text: `Switched approval mode to ${newMode}`, - }, - ], - }, - ]); - - setOverlayMode("none"); - }} - onExit={() => setOverlayMode("none")} - /> - )} - - {overlayMode === "help" && ( - setOverlayMode("none")} /> - )} - - {overlayMode === "diff" && ( - setOverlayMode("none")} - /> - )} - - - ); -} diff --git a/codex-cli/src/components/chat/terminal-header.tsx b/codex-cli/src/components/chat/terminal-header.tsx deleted file mode 100644 index 9ba16e6f..00000000 --- a/codex-cli/src/components/chat/terminal-header.tsx +++ /dev/null @@ -1,99 +0,0 @@ -import type { AgentLoop } from "../../utils/agent/agent-loop.js"; - -import { Box, Text } from "ink"; -import path from "node:path"; -import React from "react"; - -export interface TerminalHeaderProps { - terminalRows: number; - version: string; - PWD: string; - model: string; - provider?: string; - approvalPolicy: string; - colorsByPolicy: Record; - agent?: AgentLoop; - initialImagePaths?: Array; - flexModeEnabled?: boolean; -} - -const TerminalHeader: React.FC = ({ - terminalRows, - version, - PWD, - model, - provider = "openai", - approvalPolicy, - colorsByPolicy, - agent, - initialImagePaths, - flexModeEnabled = false, -}) => { - return ( - <> - {terminalRows < 10 ? ( - // Compact header for small terminal windows - - ● Codex v{version} - {PWD} - {model} ({provider}) -{" "} - {approvalPolicy} - {flexModeEnabled ? " - flex-mode" : ""} - - ) : ( - <> - - - ● OpenAI Codex{" "} - - (research preview) v{version} - - - - - - localhost session:{" "} - - {agent?.sessionId ?? ""} - - - - workdir: {PWD} - - - model: {model} - - - provider:{" "} - {provider} - - - approval:{" "} - - {approvalPolicy} - - - {flexModeEnabled && ( - - flex-mode:{" "} - enabled - - )} - {initialImagePaths?.map((img, idx) => ( - - image:{" "} - {path.basename(img)} - - ))} - - - )} - - ); -}; - -export default TerminalHeader; diff --git a/codex-cli/src/components/chat/terminal-message-history.tsx b/codex-cli/src/components/chat/terminal-message-history.tsx deleted file mode 100644 index 5036f081..00000000 --- a/codex-cli/src/components/chat/terminal-message-history.tsx +++ /dev/null @@ -1,93 +0,0 @@ -import type { OverlayModeType } from "./terminal-chat.js"; -import type { TerminalHeaderProps } from "./terminal-header.js"; -import type { GroupedResponseItem } from "./use-message-grouping.js"; -import type { ResponseItem } from "openai/resources/responses/responses.mjs"; -import type { FileOpenerScheme } from "src/utils/config.js"; - -import TerminalChatResponseItem from "./terminal-chat-response-item.js"; -import TerminalHeader from "./terminal-header.js"; -import { Box, Static } from "ink"; -import React, { useMemo } from "react"; - -// A batch entry can either be a standalone response item or a grouped set of -// items (e.g. auto‑approved tool‑call batches) that should be rendered -// together. -type BatchEntry = { item?: ResponseItem; group?: GroupedResponseItem }; -type TerminalMessageHistoryProps = { - batch: Array; - groupCounts: Record; - items: Array; - userMsgCount: number; - confirmationPrompt: React.ReactNode; - loading: boolean; - thinkingSeconds: number; - headerProps: TerminalHeaderProps; - fullStdout: boolean; - setOverlayMode: React.Dispatch>; - fileOpener: FileOpenerScheme | undefined; -}; - -const TerminalMessageHistory: React.FC = ({ - batch, - headerProps, - // `loading` and `thinkingSeconds` handled by input component now. - loading: _loading, - thinkingSeconds: _thinkingSeconds, - fullStdout, - setOverlayMode, - fileOpener, -}) => { - // Flatten batch entries to response items. - const messages = useMemo(() => batch.map(({ item }) => item!), [batch]); - - return ( - - {/* The dedicated thinking indicator in the input area now displays the - elapsed time, so we no longer render a separate counter here. */} - - {(item, index) => { - if (item === "header") { - return ; - } - - // After the guard above, item is a ResponseItem - const message = item as ResponseItem; - // Suppress empty reasoning updates (i.e. items with an empty summary). - const msg = message as unknown as { summary?: Array }; - if (msg.summary?.length === 0) { - return null; - } - return ( - - - - ); - }} - - - ); -}; - -export default React.memo(TerminalMessageHistory); diff --git a/codex-cli/src/components/chat/use-message-grouping.ts b/codex-cli/src/components/chat/use-message-grouping.ts deleted file mode 100644 index 1e526821..00000000 --- a/codex-cli/src/components/chat/use-message-grouping.ts +++ /dev/null @@ -1,9 +0,0 @@ -import type { ResponseItem } from "openai/resources/responses/responses.mjs"; - -/** - * Represents a grouped sequence of response items (e.g., function call batches). - */ -export type GroupedResponseItem = { - label: string; - items: Array; -}; diff --git a/codex-cli/src/components/diff-overlay.tsx b/codex-cli/src/components/diff-overlay.tsx deleted file mode 100644 index 8de85b87..00000000 --- a/codex-cli/src/components/diff-overlay.tsx +++ /dev/null @@ -1,93 +0,0 @@ -import { Box, Text, useInput } from "ink"; -import React, { useState } from "react"; - -/** - * Simple scrollable view for displaying a diff. - * The component is intentionally lightweight and mirrors the UX of - * HistoryOverlay: Up/Down or j/k to scroll, PgUp/PgDn for paging and Esc to - * close. The caller is responsible for computing the diff text. - */ -export default function DiffOverlay({ - diffText, - onExit, -}: { - diffText: string; - onExit: () => void; -}): JSX.Element { - const lines = diffText.length > 0 ? diffText.split("\n") : ["(no changes)"]; - - const [cursor, setCursor] = useState(0); - - // Determine how many rows we can display – similar to HistoryOverlay. - const rows = process.stdout.rows || 24; - const headerRows = 2; - const footerRows = 1; - const maxVisible = Math.max(4, rows - headerRows - footerRows); - - useInput((input, key) => { - if (key.escape || input === "q") { - onExit(); - return; - } - - if (key.downArrow || input === "j") { - setCursor((c) => Math.min(lines.length - 1, c + 1)); - } else if (key.upArrow || input === "k") { - setCursor((c) => Math.max(0, c - 1)); - } else if (key.pageDown) { - setCursor((c) => Math.min(lines.length - 1, c + maxVisible)); - } else if (key.pageUp) { - setCursor((c) => Math.max(0, c - maxVisible)); - } else if (input === "g") { - setCursor(0); - } else if (input === "G") { - setCursor(lines.length - 1); - } - }); - - const firstVisible = Math.min( - Math.max(0, cursor - Math.floor(maxVisible / 2)), - Math.max(0, lines.length - maxVisible), - ); - const visible = lines.slice(firstVisible, firstVisible + maxVisible); - - // Very small helper to colorize diff lines in a basic way. - function renderLine(line: string, idx: number): JSX.Element { - let color: "green" | "red" | "cyan" | undefined = undefined; - if (line.startsWith("+")) { - color = "green"; - } else if (line.startsWith("-")) { - color = "red"; - } else if (line.startsWith("@@") || line.startsWith("diff --git")) { - color = "cyan"; - } - return ( - - {line === "" ? " " : line} - - ); - } - - return ( - - - Working tree diff ({lines.length} lines) - - - - {visible.map((line, idx) => { - return renderLine(line, firstVisible + idx); - })} - - - - esc Close ↑↓ Scroll PgUp/PgDn g/G First/Last - - - ); -} diff --git a/codex-cli/src/components/help-overlay.tsx b/codex-cli/src/components/help-overlay.tsx deleted file mode 100644 index 1c24ad9c..00000000 --- a/codex-cli/src/components/help-overlay.tsx +++ /dev/null @@ -1,103 +0,0 @@ -import { Box, Text, useInput } from "ink"; -import React from "react"; - -/** - * An overlay that lists the available slash‑commands and their description. - * The overlay is purely informational and can be dismissed with the Escape - * key. Keeping the implementation extremely small avoids adding any new - * dependencies or complex state handling. - */ -export default function HelpOverlay({ - onExit, -}: { - onExit: () => void; -}): JSX.Element { - useInput((input, key) => { - if (key.escape || input === "q") { - onExit(); - } - }); - - return ( - - - Available commands - - - - - Slash‑commands - - - /help – show this help overlay - - - /model – switch the LLM model in‑session - - - /approval – switch auto‑approval mode - - - /history – show command & file history - for this session - - - /clear – clear screen & context - - - /clearhistory – clear command history - - - /bug – generate a prefilled GitHub issue URL - with session log - - - /diff – view working tree git diff - - - /compact – condense context into a summary - - - - - Keyboard shortcuts - - - - Enter – send message - - - Ctrl+J – insert newline - - {/* Re-enable once we re-enable new input */} - {/* - - Ctrl+X/Ctrl+E -  – open external editor ($EDITOR) - - */} - - Up/Down – scroll prompt history - - - - Esc(✕2) - {" "} - – interrupt current action - - - Ctrl+C – quit Codex - - - - - Esc or q to close - - - ); -} diff --git a/codex-cli/src/components/history-overlay.tsx b/codex-cli/src/components/history-overlay.tsx deleted file mode 100644 index f6ea8464..00000000 --- a/codex-cli/src/components/history-overlay.tsx +++ /dev/null @@ -1,255 +0,0 @@ -import type { ResponseItem } from "openai/resources/responses/responses.mjs"; - -import { Box, Text, useInput } from "ink"; -import React, { useMemo, useState } from "react"; - -type Props = { - items: Array; - onExit: () => void; -}; - -type Mode = "commands" | "files"; - -export default function HistoryOverlay({ items, onExit }: Props): JSX.Element { - const [mode, setMode] = useState("commands"); - const [cursor, setCursor] = useState(0); - - const { commands, files } = useMemo( - () => formatHistoryForDisplay(items), - [items], - ); - - const list = mode === "commands" ? commands : files; - - useInput((input, key) => { - if (key.escape) { - onExit(); - return; - } - - if (input === "c") { - setMode("commands"); - setCursor(0); - return; - } - if (input === "f") { - setMode("files"); - setCursor(0); - return; - } - - if (key.downArrow || input === "j") { - setCursor((c) => Math.min(list.length - 1, c + 1)); - } else if (key.upArrow || input === "k") { - setCursor((c) => Math.max(0, c - 1)); - } else if (key.pageDown) { - setCursor((c) => Math.min(list.length - 1, c + 10)); - } else if (key.pageUp) { - setCursor((c) => Math.max(0, c - 10)); - } else if (input === "g") { - setCursor(0); - } else if (input === "G") { - setCursor(list.length - 1); - } - }); - - const rows = process.stdout.rows || 24; - const headerRows = 2; - const footerRows = 1; - const maxVisible = Math.max(4, rows - headerRows - footerRows); - - const firstVisible = Math.min( - Math.max(0, cursor - Math.floor(maxVisible / 2)), - Math.max(0, list.length - maxVisible), - ); - const visible = list.slice(firstVisible, firstVisible + maxVisible); - - return ( - - - - {mode === "commands" ? "Commands run" : "Files touched"} ( - {list.length}) - - - - {visible.map((txt, idx) => { - const absIdx = firstVisible + idx; - const selected = absIdx === cursor; - return ( - - {selected ? "› " : " "} - {txt} - - ); - })} - - - - esc Close ↑↓ Scroll PgUp/PgDn g/G First/Last c Commands f Files - - - - ); -} - -function formatHistoryForDisplay(items: Array): { - commands: Array; - files: Array; -} { - const commands: Array = []; - const filesSet = new Set(); - - for (const item of items) { - const userPrompt = processUserMessage(item); - if (userPrompt) { - commands.push(userPrompt); - continue; - } - - // ------------------------------------------------------------------ - // We are interested in tool calls which – for the OpenAI client – are - // represented as `function_call` response items. Skip everything else. - if (item.type !== "function_call") { - continue; - } - - const { name: toolName, arguments: argsString } = item as unknown as { - name: unknown; - arguments: unknown; - }; - - if (typeof argsString !== "string") { - // Malformed – still record the tool name to give users maximal context. - if (typeof toolName === "string" && toolName.length > 0) { - commands.push(toolName); - } - continue; - } - - // Best‑effort attempt to parse the JSON arguments. We never throw on parse - // failure – the history view must be resilient to bad data. - let argsJson: unknown = undefined; - try { - argsJson = JSON.parse(argsString); - } catch { - argsJson = undefined; - } - - // 1) Shell / exec‑like tool calls expose a `cmd` or `command` property - // that is an array of strings. These are rendered as the joined command - // line for familiarity with traditional shells. - const argsObj = argsJson as Record | undefined; - const cmdArray: Array | undefined = Array.isArray(argsObj?.["cmd"]) - ? (argsObj!["cmd"] as Array) - : Array.isArray(argsObj?.["command"]) - ? (argsObj!["command"] as Array) - : undefined; - - if (cmdArray && cmdArray.length > 0) { - commands.push(processCommandArray(cmdArray, filesSet)); - continue; // We processed this as a command; no need to treat as generic tool call. - } - - // 2) Non‑exec tool calls – we fall back to recording the tool name plus a - // short argument representation to give users an idea of what - // happened. - if (typeof toolName === "string" && toolName.length > 0) { - commands.push(processNonExecTool(toolName, argsJson, filesSet)); - } - } - - return { commands, files: Array.from(filesSet) }; -} - -function processUserMessage(item: ResponseItem): string | null { - if ( - item.type === "message" && - (item as unknown as { role?: string }).role === "user" - ) { - // TODO: We're ignoring images/files here. - const parts = - (item as unknown as { content?: Array }).content ?? []; - const texts: Array = []; - if (Array.isArray(parts)) { - for (const part of parts) { - if (part && typeof part === "object" && "text" in part) { - const t = (part as unknown as { text?: string }).text; - if (typeof t === "string" && t.length > 0) { - texts.push(t); - } - } - } - } - - if (texts.length > 0) { - const fullPrompt = texts.join(" "); - // Truncate very long prompts so the history view stays legible. - return fullPrompt.length > 120 - ? `> ${fullPrompt.slice(0, 117)}…` - : `> ${fullPrompt}`; - } - } - return null; -} - -function processCommandArray( - cmdArray: Array, - filesSet: Set, -): string { - const cmd = cmdArray.join(" "); - - // Heuristic for file paths in command args - for (const part of cmdArray) { - if (!part.startsWith("-") && part.includes("/")) { - filesSet.add(part); - } - } - - // Special‑case apply_patch so we can extract the list of modified files - if (cmdArray[0] === "apply_patch" || cmdArray.includes("apply_patch")) { - const patchTextMaybe = cmdArray.find((s) => s.includes("*** Begin Patch")); - if (typeof patchTextMaybe === "string") { - const lines = patchTextMaybe.split("\n"); - for (const line of lines) { - const m = line.match(/^[-+]{3} [ab]\/(.+)$/); - if (m && m[1]) { - filesSet.add(m[1]); - } - } - } - } - - return cmd; -} - -function processNonExecTool( - toolName: string, - argsJson: unknown, - filesSet: Set, -): string { - let summary = toolName; - - if (argsJson && typeof argsJson === "object") { - // Extract a few common argument keys to make the summary more useful - // without being overly verbose. - const interestingKeys = ["path", "file", "filepath", "filename", "pattern"]; - for (const key of interestingKeys) { - const val = (argsJson as Record)[key]; - if (typeof val === "string") { - summary += ` ${val}`; - if (val.includes("/")) { - filesSet.add(val); - } - break; - } - } - } - - return summary; -} diff --git a/codex-cli/src/components/model-overlay.tsx b/codex-cli/src/components/model-overlay.tsx deleted file mode 100644 index 86a7e585..00000000 --- a/codex-cli/src/components/model-overlay.tsx +++ /dev/null @@ -1,165 +0,0 @@ -import TypeaheadOverlay from "./typeahead-overlay.js"; -import { - getAvailableModels, - RECOMMENDED_MODELS as _RECOMMENDED_MODELS, -} from "../utils/model-utils.js"; -import { Box, Text, useInput } from "ink"; -import React, { useEffect, useState } from "react"; - -/** - * Props for . - * - * When `hasLastResponse` is true the user has already received at least one - * assistant response in the current session which means switching models is no - * longer supported – the overlay should therefore show an error and only allow - * the user to close it. - */ -type Props = { - currentModel: string; - currentProvider?: string; - hasLastResponse: boolean; - providers?: Record; - onSelect: (allModels: Array, model: string) => void; - onSelectProvider?: (provider: string) => void; - onExit: () => void; -}; - -export default function ModelOverlay({ - currentModel, - providers = {}, - currentProvider = "openai", - hasLastResponse, - onSelect, - onSelectProvider, - onExit, -}: Props): JSX.Element { - const [items, setItems] = useState>( - [], - ); - const [providerItems, _setProviderItems] = useState< - Array<{ label: string; value: string }> - >(Object.values(providers).map((p) => ({ label: p.name, value: p.name }))); - const [mode, setMode] = useState<"model" | "provider">("model"); - const [isLoading, setIsLoading] = useState(true); - - // This effect will run when the provider changes to update the model list - useEffect(() => { - setIsLoading(true); - (async () => { - try { - const models = await getAvailableModels(currentProvider); - // Convert the models to the format needed by TypeaheadOverlay - setItems( - models.map((m) => ({ - label: m, - value: m, - })), - ); - } catch (error) { - // Silently handle errors - remove console.error - // console.error("Error loading models:", error); - } finally { - setIsLoading(false); - } - })(); - }, [currentProvider]); - - // --------------------------------------------------------------------------- - // If the conversation already contains a response we cannot change the model - // anymore because the backend requires a consistent model across the entire - // run. In that scenario we replace the regular typeahead picker with a - // simple message instructing the user to start a new chat. The only - // available action is to dismiss the overlay (Esc or Enter). - // --------------------------------------------------------------------------- - - // Register input handling for switching between model and provider selection - useInput((_input, key) => { - if (hasLastResponse && (key.escape || key.return)) { - onExit(); - } else if (!hasLastResponse) { - if (key.tab) { - setMode(mode === "model" ? "provider" : "model"); - } - } - }); - - if (hasLastResponse) { - return ( - - - - Unable to switch model - - - - - You can only pick a model before the assistant sends its first - response. To use a different model please start a new chat. - - - - press esc or enter to close - - - ); - } - - if (mode === "provider") { - return ( - - - Current provider:{" "} - {currentProvider} - - press tab to switch to model selection - - } - initialItems={providerItems} - currentValue={currentProvider} - onSelect={(provider) => { - if (onSelectProvider) { - onSelectProvider(provider); - // Immediately switch to model selection so user can pick a model for the new provider - setMode("model"); - } - }} - onExit={onExit} - /> - ); - } - - return ( - - - Current model: {currentModel} - - - Current provider: {currentProvider} - - {isLoading && Loading models...} - press tab to switch to provider selection - - } - initialItems={items} - currentValue={currentModel} - onSelect={(selectedModel) => - onSelect( - items?.map((m) => m.value), - selectedModel, - ) - } - onExit={onExit} - /> - ); -} diff --git a/codex-cli/src/components/onboarding/onboarding-approval-mode.tsx b/codex-cli/src/components/onboarding/onboarding-approval-mode.tsx deleted file mode 100644 index f095c6c0..00000000 --- a/codex-cli/src/components/onboarding/onboarding-approval-mode.tsx +++ /dev/null @@ -1,35 +0,0 @@ -// @ts-expect-error select.js is JavaScript and has no types -import { Select } from "../vendor/ink-select/select"; -import { Box, Text } from "ink"; -import React from "react"; -import { AutoApprovalMode } from "src/utils/auto-approval-mode"; - -// TODO: figure out why `cli-spinners` fails on Node v20.9.0 -// which is why we have to do this in the first place - -export function OnboardingApprovalMode(): React.ReactElement { - return ( - - Choose what you want to have to approve: -