@openparachute/agent 0.1.2 → 0.2.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.parachute/module.json +124 -8
- package/LICENSE +2 -16
- package/README.md +118 -166
- package/package.json +32 -43
- package/scripts/spawn-agent.ts +371 -0
- package/src/_parked/interactive-spawn.test.ts +324 -0
- package/src/_parked/interactive-spawn.ts +701 -0
- package/src/agent-defs.test.ts +1504 -0
- package/src/agent-defs.ts +1702 -0
- package/src/agent-mcp-config.test.ts +115 -0
- package/src/agent-mcp-config.ts +115 -0
- package/src/agents.test.ts +360 -0
- package/src/agents.ts +379 -0
- package/src/auth.test.ts +46 -0
- package/src/auth.ts +140 -0
- package/src/backends/attached-queue.test.ts +376 -0
- package/src/backends/attached-queue.ts +372 -0
- package/src/backends/programmatic.test.ts +1715 -0
- package/src/backends/programmatic.ts +927 -0
- package/src/backends/registry.test.ts +1494 -0
- package/src/backends/registry.ts +1202 -0
- package/src/backends/stream-json.test.ts +570 -0
- package/src/backends/stream-json.ts +392 -0
- package/src/backends/types.ts +223 -0
- package/src/bridge.ts +417 -0
- package/src/channel-backend-wiring.test.ts +237 -0
- package/src/credentials.test.ts +274 -0
- package/src/credentials.ts +380 -0
- package/src/cron.test.ts +342 -0
- package/src/cron.ts +380 -0
- package/src/daemon-agent-def-api.test.ts +166 -0
- package/src/daemon-agent-defs-api.test.ts +953 -0
- package/src/daemon-agent-env-api.test.ts +338 -0
- package/src/daemon-attached-queue-store.test.ts +65 -0
- package/src/daemon-config-api.test.ts +962 -0
- package/src/daemon-jobs-api.test.ts +271 -0
- package/src/daemon-vault-chat.test.ts +250 -0
- package/src/daemon.test.ts +746 -0
- package/src/daemon.ts +3314 -0
- package/src/def-vaults.test.ts +136 -0
- package/src/def-vaults.ts +165 -0
- package/src/delivery-state.test.ts +110 -0
- package/src/delivery-state.ts +154 -0
- package/src/effective-env.test.ts +114 -0
- package/src/effective-env.ts +184 -0
- package/src/env-compat.ts +39 -0
- package/src/grants.test.ts +638 -0
- package/src/grants.ts +675 -0
- package/src/hub-jwt.test.ts +161 -0
- package/src/hub-jwt.ts +182 -0
- package/src/jobs.test.ts +245 -0
- package/src/jobs.ts +266 -0
- package/src/mcp-http.test.ts +265 -0
- package/src/mcp-http.ts +771 -0
- package/src/mint-token.test.ts +152 -0
- package/src/mint-token.ts +139 -0
- package/src/module-manifest.test.ts +158 -0
- package/src/oauth-discovery.ts +134 -0
- package/src/programmatic-wiring.test.ts +838 -0
- package/src/registry.test.ts +227 -0
- package/src/registry.ts +228 -0
- package/src/resolve-port.test.ts +64 -0
- package/src/routing.test.ts +184 -0
- package/src/routing.ts +76 -0
- package/src/runner.test.ts +506 -0
- package/src/runner.ts +255 -0
- package/src/sandbox/config.test.ts +150 -0
- package/src/sandbox/config.ts +102 -0
- package/src/sandbox/egress.test.ts +113 -0
- package/src/sandbox/egress.ts +123 -0
- package/src/sandbox/index.ts +180 -0
- package/src/sandbox/live-seatbelt.test.ts +277 -0
- package/src/sandbox/mounts.test.ts +154 -0
- package/src/sandbox/mounts.ts +133 -0
- package/src/sandbox/sandbox.test.ts +168 -0
- package/src/sandbox/types.ts +382 -0
- package/src/services-manifest.test.ts +106 -0
- package/src/services-manifest.ts +95 -0
- package/src/spa-serve.test.ts +116 -0
- package/src/spa-serve.ts +116 -0
- package/src/spawn-agent-cli.test.ts +172 -0
- package/src/spawn-agent.test.ts +1218 -0
- package/src/spawn-agent.ts +569 -0
- package/src/spawn-deps.test.ts +54 -0
- package/src/spawn-deps.ts +166 -0
- package/src/telegram/api.ts +153 -0
- package/src/terminal-assets.test.ts +50 -0
- package/src/terminal-assets.ts +79 -0
- package/src/terminal-ui.ts +305 -0
- package/src/terminal.test.ts +530 -0
- package/src/terminal.ts +458 -0
- package/src/transport.ts +270 -0
- package/src/transports/http-ui.test.ts +455 -0
- package/src/transports/http-ui.ts +201 -0
- package/src/transports/telegram.test.ts +174 -0
- package/src/transports/telegram.ts +426 -0
- package/src/transports/vault.test.ts +2011 -0
- package/src/transports/vault.ts +1790 -0
- package/src/ui-kit.test.ts +178 -0
- package/src/ui-kit.ts +402 -0
- package/tsconfig.json +8 -14
- package/web/ui/tsconfig.json +2 -1
- package/.claude/scheduled_tasks.lock +0 -1
- package/.claude/settings.json +0 -5
- package/.claude/skills/add-atomic-chat-tool/SKILL.md +0 -243
- package/.claude/skills/add-atomic-chat-tool/atomic-chat-mcp-stdio.ts +0 -229
- package/.claude/skills/add-codex/SKILL.md +0 -161
- package/.claude/skills/add-dashboard/SKILL.md +0 -138
- package/.claude/skills/add-dashboard/resources/dashboard-pusher.ts +0 -495
- package/.claude/skills/add-emacs/SKILL.md +0 -296
- package/.claude/skills/add-gcal-tool/SKILL.md +0 -210
- package/.claude/skills/add-gchat/REMOVE.md +0 -6
- package/.claude/skills/add-gchat/SKILL.md +0 -92
- package/.claude/skills/add-gchat/VERIFY.md +0 -3
- package/.claude/skills/add-github/REMOVE.md +0 -6
- package/.claude/skills/add-github/SKILL.md +0 -148
- package/.claude/skills/add-github/VERIFY.md +0 -3
- package/.claude/skills/add-gmail-tool/SKILL.md +0 -229
- package/.claude/skills/add-imessage/REMOVE.md +0 -6
- package/.claude/skills/add-imessage/SKILL.md +0 -113
- package/.claude/skills/add-imessage/VERIFY.md +0 -3
- package/.claude/skills/add-karpathy-llm-wiki/SKILL.md +0 -110
- package/.claude/skills/add-karpathy-llm-wiki/llm-wiki.md +0 -75
- package/.claude/skills/add-linear/REMOVE.md +0 -6
- package/.claude/skills/add-linear/SKILL.md +0 -168
- package/.claude/skills/add-linear/VERIFY.md +0 -3
- package/.claude/skills/add-macos-statusbar/SKILL.md +0 -133
- package/.claude/skills/add-macos-statusbar/add/src/statusbar.swift +0 -147
- package/.claude/skills/add-matrix/REMOVE.md +0 -6
- package/.claude/skills/add-matrix/SKILL.md +0 -148
- package/.claude/skills/add-matrix/VERIFY.md +0 -3
- package/.claude/skills/add-ollama-provider/SKILL.md +0 -179
- package/.claude/skills/add-ollama-tool/SKILL.md +0 -193
- package/.claude/skills/add-opencode/SKILL.md +0 -229
- package/.claude/skills/add-parallel/SKILL.md +0 -290
- package/.claude/skills/add-resend/REMOVE.md +0 -6
- package/.claude/skills/add-resend/SKILL.md +0 -93
- package/.claude/skills/add-resend/VERIFY.md +0 -3
- package/.claude/skills/add-signal/REMOVE.md +0 -13
- package/.claude/skills/add-signal/SKILL.md +0 -318
- package/.claude/skills/add-signal/VERIFY.md +0 -5
- package/.claude/skills/add-slack/REMOVE.md +0 -6
- package/.claude/skills/add-slack/SKILL.md +0 -112
- package/.claude/skills/add-slack/VERIFY.md +0 -3
- package/.claude/skills/add-teams/REMOVE.md +0 -6
- package/.claude/skills/add-teams/SKILL.md +0 -207
- package/.claude/skills/add-teams/VERIFY.md +0 -3
- package/.claude/skills/add-vercel/SKILL.md +0 -147
- package/.claude/skills/add-vercel/container-skills/vercel-cli/SKILL.md +0 -103
- package/.claude/skills/add-webex/REMOVE.md +0 -6
- package/.claude/skills/add-webex/SKILL.md +0 -88
- package/.claude/skills/add-webex/VERIFY.md +0 -3
- package/.claude/skills/add-wechat/REMOVE.md +0 -49
- package/.claude/skills/add-wechat/SKILL.md +0 -170
- package/.claude/skills/add-wechat/scripts/wire-dm.ts +0 -172
- package/.claude/skills/add-whatsapp/SKILL.md +0 -264
- package/.claude/skills/add-whatsapp-cloud/REMOVE.md +0 -6
- package/.claude/skills/add-whatsapp-cloud/SKILL.md +0 -95
- package/.claude/skills/add-whatsapp-cloud/VERIFY.md +0 -3
- package/.claude/skills/claw/SKILL.md +0 -131
- package/.claude/skills/claw/scripts/claw +0 -374
- package/.claude/skills/convert-to-apple-container/SKILL.md +0 -212
- package/.claude/skills/customize/SKILL.md +0 -110
- package/.claude/skills/debug/SKILL.md +0 -349
- package/.claude/skills/get-qodo-rules/SKILL.md +0 -122
- package/.claude/skills/get-qodo-rules/references/output-format.md +0 -41
- package/.claude/skills/get-qodo-rules/references/pagination.md +0 -33
- package/.claude/skills/get-qodo-rules/references/repository-scope.md +0 -26
- package/.claude/skills/init-first-agent/SKILL.md +0 -120
- package/.claude/skills/init-onecli/SKILL.md +0 -270
- package/.claude/skills/manage-channels/SKILL.md +0 -87
- package/.claude/skills/manage-mounts/SKILL.md +0 -47
- package/.claude/skills/migrate-from-openclaw/MIGRATE_CRONS.md +0 -100
- package/.claude/skills/migrate-from-openclaw/SKILL.md +0 -447
- package/.claude/skills/migrate-from-openclaw/scripts/discover-openclaw.ts +0 -734
- package/.claude/skills/migrate-from-openclaw/scripts/extract-channel-credentials.ts +0 -476
- package/.claude/skills/migrate-nanoclaw/SKILL.md +0 -484
- package/.claude/skills/migrate-nanoclaw/diagnostics.md +0 -51
- package/.claude/skills/qodo-pr-resolver/SKILL.md +0 -326
- package/.claude/skills/qodo-pr-resolver/resources/providers.md +0 -329
- package/.claude/skills/update-nanoclaw/SKILL.md +0 -243
- package/.claude/skills/update-nanoclaw/diagnostics.md +0 -48
- package/.claude/skills/update-skills/SKILL.md +0 -130
- package/.claude/skills/use-native-credential-proxy/SKILL.md +0 -167
- package/.claude/skills/x-integration/SKILL.md +0 -417
- package/.claude/skills/x-integration/agent.ts +0 -243
- package/.claude/skills/x-integration/host.ts +0 -155
- package/.claude/skills/x-integration/lib/browser.ts +0 -148
- package/.claude/skills/x-integration/lib/config.ts +0 -62
- package/.claude/skills/x-integration/scripts/like.ts +0 -56
- package/.claude/skills/x-integration/scripts/post.ts +0 -66
- package/.claude/skills/x-integration/scripts/quote.ts +0 -80
- package/.claude/skills/x-integration/scripts/reply.ts +0 -74
- package/.claude/skills/x-integration/scripts/retweet.ts +0 -62
- package/.claude/skills/x-integration/scripts/setup.ts +0 -87
- package/.github/CODEOWNERS +0 -10
- package/.github/PULL_REQUEST_TEMPLATE.md +0 -18
- package/.github/workflows/bump-version.yml +0 -35
- package/.github/workflows/ci.yml +0 -39
- package/.github/workflows/label-pr.yml +0 -40
- package/.github/workflows/update-tokens.yml +0 -43
- package/.husky/pre-commit +0 -1
- package/.mcp.json +0 -3
- package/.nvmrc +0 -1
- package/.prettierrc +0 -4
- package/CHANGELOG.md +0 -263
- package/CLAUDE.md +0 -307
- package/CODE_OF_CONDUCT.md +0 -128
- package/CONTRIBUTING.md +0 -159
- package/CONTRIBUTORS.md +0 -26
- package/LICENSE-NANOCLAW-MIT +0 -21
- package/README_ja.md +0 -194
- package/README_zh.md +0 -194
- package/assets/nanoclaw-favicon.png +0 -0
- package/assets/nanoclaw-icon.png +0 -0
- package/assets/nanoclaw-logo-dark.png +0 -0
- package/assets/nanoclaw-logo.png +0 -0
- package/assets/nanoclaw-profile.jpeg +0 -0
- package/assets/nanoclaw-sales.png +0 -0
- package/assets/social-preview.jpg +0 -0
- package/config-examples/mount-allowlist.json +0 -25
- package/container/.dockerignore +0 -2
- package/container/CLAUDE.md +0 -21
- package/container/Dockerfile +0 -121
- package/container/agent-runner/bun.lock +0 -243
- package/container/agent-runner/package.json +0 -22
- package/container/agent-runner/scripts/sdk-signal-probe.ts +0 -169
- package/container/agent-runner/src/config.ts +0 -55
- package/container/agent-runner/src/db/connection.ts +0 -267
- package/container/agent-runner/src/db/index.ts +0 -20
- package/container/agent-runner/src/db/messages-in.ts +0 -138
- package/container/agent-runner/src/db/messages-out.ts +0 -143
- package/container/agent-runner/src/db/session-routing.ts +0 -30
- package/container/agent-runner/src/db/session-state.test.ts +0 -100
- package/container/agent-runner/src/db/session-state.ts +0 -79
- package/container/agent-runner/src/destinations.ts +0 -135
- package/container/agent-runner/src/formatter.test.ts +0 -167
- package/container/agent-runner/src/formatter.ts +0 -260
- package/container/agent-runner/src/index.ts +0 -110
- package/container/agent-runner/src/integration.test.ts +0 -121
- package/container/agent-runner/src/mcp-tools/agents.instructions.md +0 -26
- package/container/agent-runner/src/mcp-tools/agents.ts +0 -66
- package/container/agent-runner/src/mcp-tools/core.instructions.md +0 -27
- package/container/agent-runner/src/mcp-tools/core.ts +0 -262
- package/container/agent-runner/src/mcp-tools/index.ts +0 -22
- package/container/agent-runner/src/mcp-tools/interactive.instructions.md +0 -22
- package/container/agent-runner/src/mcp-tools/interactive.ts +0 -169
- package/container/agent-runner/src/mcp-tools/scheduling.instructions.md +0 -40
- package/container/agent-runner/src/mcp-tools/scheduling.ts +0 -299
- package/container/agent-runner/src/mcp-tools/self-mod.instructions.md +0 -25
- package/container/agent-runner/src/mcp-tools/self-mod.ts +0 -120
- package/container/agent-runner/src/mcp-tools/server.ts +0 -54
- package/container/agent-runner/src/mcp-tools/types.ts +0 -6
- package/container/agent-runner/src/poll-loop.test.ts +0 -248
- package/container/agent-runner/src/poll-loop.ts +0 -437
- package/container/agent-runner/src/providers/claude.ts +0 -379
- package/container/agent-runner/src/providers/factory.test.ts +0 -19
- package/container/agent-runner/src/providers/factory.ts +0 -13
- package/container/agent-runner/src/providers/index.ts +0 -6
- package/container/agent-runner/src/providers/mock.ts +0 -77
- package/container/agent-runner/src/providers/provider-registry.ts +0 -33
- package/container/agent-runner/src/providers/types.ts +0 -82
- package/container/agent-runner/src/scheduling/task-script.ts +0 -121
- package/container/agent-runner/src/timezone.test.ts +0 -93
- package/container/agent-runner/src/timezone.ts +0 -107
- package/container/agent-runner/tsconfig.json +0 -14
- package/container/build.sh +0 -48
- package/container/entrypoint.sh +0 -16
- package/container/skills/agent-browser/SKILL.md +0 -159
- package/container/skills/frontend-engineer/SKILL.md +0 -157
- package/container/skills/self-customize/SKILL.md +0 -87
- package/container/skills/slack-formatting/SKILL.md +0 -94
- package/container/skills/vercel-cli/SKILL.md +0 -111
- package/container/skills/welcome/SKILL.md +0 -85
- package/docs/APPLE-CONTAINER-NETWORKING.md +0 -90
- package/docs/BRANCH-FORK-MAINTENANCE.md +0 -81
- package/docs/README.md +0 -25
- package/docs/SDK_DEEP_DIVE.md +0 -643
- package/docs/SECURITY.md +0 -162
- package/docs/agent-runner-details.md +0 -749
- package/docs/api-details.md +0 -365
- package/docs/architecture-diagram.html +0 -422
- package/docs/architecture-diagram.md +0 -215
- package/docs/architecture.md +0 -751
- package/docs/audit/2026-04-30-channel-endpoint-audit.md +0 -36
- package/docs/build-and-runtime.md +0 -80
- package/docs/cross-mount-stress/README.md +0 -112
- package/docs/cross-mount-stress/container-writer-retry.mjs +0 -55
- package/docs/cross-mount-stress/container-writer-slow.mjs +0 -42
- package/docs/cross-mount-stress/container-writer.mjs +0 -47
- package/docs/cross-mount-stress/host-writer-retry.mjs +0 -55
- package/docs/cross-mount-stress/host-writer-slow.mjs +0 -43
- package/docs/cross-mount-stress/host-writer.mjs +0 -47
- package/docs/db-central.md +0 -316
- package/docs/db-session.md +0 -183
- package/docs/db.md +0 -119
- package/docs/design/2026-04-29-vault-management-ui.md +0 -231
- package/docs/design/2026-04-30-channel-wiring-rework.md +0 -234
- package/docs/design/2026-05-01-channel-wiring-approvals-deep-dive.md +0 -272
- package/docs/design/2026-05-02-channel-policy-and-approval-routing.md +0 -250
- package/docs/docker-sandboxes.md +0 -359
- package/docs/isolation-model.md +0 -88
- package/docs/ollama.md +0 -79
- package/docs/parachute-integration.md +0 -109
- package/docs/post-night-rebirth-reflections.md +0 -151
- package/eslint.config.js +0 -32
- package/pnpm-workspace.yaml +0 -8
- package/repo-tokens/README.md +0 -113
- package/repo-tokens/action.yml +0 -186
- package/repo-tokens/badge.svg +0 -23
- package/repo-tokens/examples/green.svg +0 -14
- package/repo-tokens/examples/red.svg +0 -14
- package/repo-tokens/examples/yellow-green.svg +0 -14
- package/repo-tokens/examples/yellow.svg +0 -14
- package/scripts/chat.ts +0 -101
- package/scripts/cleanup-sessions.sh +0 -150
- package/scripts/init-cli-agent.ts +0 -172
- package/scripts/init-first-agent.ts +0 -378
- package/scripts/parachute.ts +0 -158
- package/scripts/run-migrations.ts +0 -105
- package/scripts/sanity-live-poll.ts +0 -95
- package/scripts/seed-discord.ts +0 -80
- package/scripts/test-v2-agent.ts +0 -106
- package/scripts/test-v2-channel-e2e.ts +0 -265
- package/scripts/test-v2-host.ts +0 -184
- package/src/channels/adapter.ts +0 -214
- package/src/channels/api-translator.test.ts +0 -306
- package/src/channels/api-translator.ts +0 -214
- package/src/channels/ask-question.ts +0 -46
- package/src/channels/channel-registry.test.ts +0 -421
- package/src/channels/channel-registry.ts +0 -313
- package/src/channels/chat-sdk-bridge.test.ts +0 -84
- package/src/channels/chat-sdk-bridge.ts +0 -652
- package/src/channels/cli.ts +0 -276
- package/src/channels/discord.ts +0 -90
- package/src/channels/index.ts +0 -17
- package/src/channels/telegram-markdown-sanitize.test.ts +0 -78
- package/src/channels/telegram-markdown-sanitize.ts +0 -55
- package/src/channels/telegram-pairing.test.ts +0 -254
- package/src/channels/telegram-pairing.ts +0 -339
- package/src/channels/telegram.ts +0 -279
- package/src/channels/trust-hint.test.ts +0 -48
- package/src/channels/trust-hint.ts +0 -75
- package/src/claude-md-compose.migrate.test.ts +0 -64
- package/src/claude-md-compose.ts +0 -205
- package/src/command-gate.ts +0 -63
- package/src/config.test.ts +0 -93
- package/src/config.ts +0 -128
- package/src/container-config.ts +0 -167
- package/src/container-runner.test.ts +0 -32
- package/src/container-runner.ts +0 -576
- package/src/container-runtime.test.ts +0 -269
- package/src/container-runtime.ts +0 -167
- package/src/db/_bun-sqlite-shim.ts +0 -88
- package/src/db/agent-activity.test.ts +0 -155
- package/src/db/agent-activity.ts +0 -121
- package/src/db/agent-groups.ts +0 -77
- package/src/db/connection.migrate.test.ts +0 -176
- package/src/db/connection.ts +0 -259
- package/src/db/db-v2.test.ts +0 -440
- package/src/db/dropped-messages.ts +0 -44
- package/src/db/index.ts +0 -40
- package/src/db/messaging-groups.ts +0 -252
- package/src/db/migrations/001-initial.ts +0 -112
- package/src/db/migrations/002-chat-sdk-state.ts +0 -36
- package/src/db/migrations/008-dropped-messages.ts +0 -27
- package/src/db/migrations/009-drop-pending-credentials.ts +0 -13
- package/src/db/migrations/010-engage-modes.ts +0 -103
- package/src/db/migrations/011-pending-sender-approvals.ts +0 -40
- package/src/db/migrations/012-channel-registration.ts +0 -48
- package/src/db/migrations/013-approval-render-metadata.ts +0 -27
- package/src/db/migrations/014-secrets.ts +0 -44
- package/src/db/migrations/015-secrets-drop-host-pattern.ts +0 -18
- package/src/db/migrations/016-secret-assignments.ts +0 -30
- package/src/db/migrations/017-agent-activity.ts +0 -40
- package/src/db/migrations/018-oauth-app-configs.ts +0 -34
- package/src/db/migrations/019-oauth-app-connections.ts +0 -48
- package/src/db/migrations/020-agent-app-connections.ts +0 -28
- package/src/db/migrations/021-pending-oauth-states.ts +0 -35
- package/src/db/migrations/022-app-connections-provider.ts +0 -25
- package/src/db/migrations/023-agent-group-secret-mode.test.ts +0 -124
- package/src/db/migrations/023-agent-group-secret-mode.ts +0 -65
- package/src/db/migrations/024-collapse-approvals.test.ts +0 -249
- package/src/db/migrations/024-collapse-approvals.ts +0 -182
- package/src/db/migrations/025-secret-mode-check.test.ts +0 -155
- package/src/db/migrations/025-secret-mode-check.ts +0 -49
- package/src/db/migrations/026-user-dms-bot-id.test.ts +0 -116
- package/src/db/migrations/026-user-dms-bot-id.ts +0 -54
- package/src/db/migrations/027-provider-credentials.ts +0 -41
- package/src/db/migrations/_test-helpers.ts +0 -41
- package/src/db/migrations/index.ts +0 -127
- package/src/db/migrations/module-agent-to-agent-destinations.ts +0 -84
- package/src/db/migrations/module-approvals-pending-approvals.ts +0 -42
- package/src/db/migrations/module-approvals-title-options.ts +0 -40
- package/src/db/schema.ts +0 -258
- package/src/db/session-db.test.ts +0 -93
- package/src/db/session-db.ts +0 -325
- package/src/db/sessions.ts +0 -241
- package/src/delivery.test.ts +0 -148
- package/src/delivery.ts +0 -445
- package/src/env.ts +0 -74
- package/src/group-folder.test.ts +0 -35
- package/src/group-folder.ts +0 -44
- package/src/group-init.ts +0 -92
- package/src/host-core.test.ts +0 -456
- package/src/host-sweep.test.ts +0 -146
- package/src/host-sweep.ts +0 -287
- package/src/index.ts +0 -232
- package/src/install-slug.ts +0 -33
- package/src/log.test.ts +0 -81
- package/src/log.ts +0 -117
- package/src/mcp/http.ts +0 -72
- package/src/mcp/server.ts +0 -92
- package/src/mcp/stdio.ts +0 -51
- package/src/mcp/tools/activity.ts +0 -88
- package/src/mcp/tools/agent-groups.ts +0 -183
- package/src/mcp/tools/approvals.ts +0 -122
- package/src/mcp/tools/channels.test.ts +0 -126
- package/src/mcp/tools/channels.ts +0 -134
- package/src/mcp/tools/index.ts +0 -27
- package/src/mcp/tools/oauth.ts +0 -48
- package/src/mcp/tools/secrets.ts +0 -169
- package/src/mcp/tools/sessions.ts +0 -135
- package/src/mcp/types.ts +0 -51
- package/src/modules/agent-to-agent/agent-route.test.ts +0 -46
- package/src/modules/agent-to-agent/agent-route.ts +0 -223
- package/src/modules/agent-to-agent/create-agent.ts +0 -127
- package/src/modules/agent-to-agent/db/agent-destinations.ts +0 -135
- package/src/modules/agent-to-agent/index.ts +0 -22
- package/src/modules/agent-to-agent/write-destinations.ts +0 -59
- package/src/modules/approvals/agent.md +0 -45
- package/src/modules/approvals/index.ts +0 -21
- package/src/modules/approvals/picks.test.ts +0 -291
- package/src/modules/approvals/primitive.ts +0 -279
- package/src/modules/approvals/project.md +0 -27
- package/src/modules/approvals/response-handler.ts +0 -87
- package/src/modules/index.ts +0 -24
- package/src/modules/interactive/agent.md +0 -21
- package/src/modules/interactive/index.ts +0 -69
- package/src/modules/interactive/project.md +0 -12
- package/src/modules/mount-security/expand-path.test.ts +0 -82
- package/src/modules/mount-security/index.ts +0 -459
- package/src/modules/mount-security/migrate.test.ts +0 -91
- package/src/modules/permissions/access.ts +0 -28
- package/src/modules/permissions/channel-approval.test.ts +0 -389
- package/src/modules/permissions/channel-approval.ts +0 -188
- package/src/modules/permissions/db/agent-group-members.ts +0 -44
- package/src/modules/permissions/db/pending-channel-approvals.test.ts +0 -86
- package/src/modules/permissions/db/pending-channel-approvals.ts +0 -66
- package/src/modules/permissions/db/pending-sender-approvals.ts +0 -60
- package/src/modules/permissions/db/user-dms.ts +0 -58
- package/src/modules/permissions/db/user-roles.ts +0 -85
- package/src/modules/permissions/db/users.ts +0 -38
- package/src/modules/permissions/index.ts +0 -421
- package/src/modules/permissions/permissions.test.ts +0 -358
- package/src/modules/permissions/sender-approval.test.ts +0 -641
- package/src/modules/permissions/sender-approval.ts +0 -165
- package/src/modules/permissions/user-dm.ts +0 -200
- package/src/modules/provider-credentials/db.ts +0 -121
- package/src/modules/provider-credentials/index.ts +0 -12
- package/src/modules/provider-credentials/spawn.test.ts +0 -206
- package/src/modules/provider-credentials/spawn.ts +0 -114
- package/src/modules/scheduling/actions.ts +0 -113
- package/src/modules/scheduling/db.test.ts +0 -282
- package/src/modules/scheduling/db.ts +0 -148
- package/src/modules/scheduling/index.ts +0 -34
- package/src/modules/scheduling/recurrence.test.ts +0 -98
- package/src/modules/scheduling/recurrence.ts +0 -54
- package/src/modules/self-mod/agent.md +0 -30
- package/src/modules/self-mod/apply.ts +0 -85
- package/src/modules/self-mod/index.ts +0 -30
- package/src/modules/self-mod/project.md +0 -39
- package/src/modules/self-mod/request.ts +0 -91
- package/src/modules/typing/index.ts +0 -165
- package/src/oauth/agent-app-connections.ts +0 -103
- package/src/oauth/app-configs.test.ts +0 -64
- package/src/oauth/app-configs.ts +0 -114
- package/src/oauth/app-connections.test.ts +0 -109
- package/src/oauth/app-connections.ts +0 -178
- package/src/oauth/crypto.ts +0 -56
- package/src/oauth/flow.ts +0 -104
- package/src/oauth/providers/google.test.ts +0 -38
- package/src/oauth/providers/google.ts +0 -46
- package/src/oauth/providers/index.ts +0 -48
- package/src/oauth/state-store.test.ts +0 -54
- package/src/oauth/state-store.ts +0 -93
- package/src/parachute/README.md +0 -27
- package/src/parachute/create-agent.test.ts +0 -83
- package/src/parachute/create-agent.ts +0 -122
- package/src/parachute/group-status.test.ts +0 -165
- package/src/parachute/group-status.ts +0 -136
- package/src/parachute/types.ts +0 -41
- package/src/parachute/vault-mcp.test.ts +0 -251
- package/src/parachute/vault-mcp.ts +0 -232
- package/src/platform-id.test.ts +0 -104
- package/src/platform-id.ts +0 -109
- package/src/providers/index.ts +0 -6
- package/src/providers/provider-container-registry.ts +0 -58
- package/src/response-registry.ts +0 -45
- package/src/router.ts +0 -530
- package/src/secrets/crypto.test.ts +0 -45
- package/src/secrets/crypto.ts +0 -55
- package/src/secrets/index.ts +0 -461
- package/src/secrets/master-key.ts +0 -70
- package/src/secrets/secrets.test.ts +0 -651
- package/src/session-manager.attachments.test.ts +0 -171
- package/src/session-manager.dup-skip.test.ts +0 -173
- package/src/session-manager.migrate.test.ts +0 -59
- package/src/session-manager.ts +0 -451
- package/src/startup-bootstrap.test.ts +0 -226
- package/src/startup-bootstrap.ts +0 -207
- package/src/state-sqlite.ts +0 -182
- package/src/timezone.test.ts +0 -64
- package/src/timezone.ts +0 -37
- package/src/types.ts +0 -233
- package/src/web/auth.test.ts +0 -335
- package/src/web/auth.ts +0 -214
- package/src/web/discord-validate.test.ts +0 -77
- package/src/web/discord-validate.ts +0 -88
- package/src/web/hub-discovery.test.ts +0 -98
- package/src/web/hub-discovery.ts +0 -69
- package/src/web/routes/activity.ts +0 -106
- package/src/web/routes/agent-provider.test.ts +0 -282
- package/src/web/routes/agent-provider.ts +0 -309
- package/src/web/routes/approvals.ts +0 -185
- package/src/web/routes/apps.ts +0 -434
- package/src/web/routes/channels-mg-detail.test.ts +0 -324
- package/src/web/routes/channels-mga-detail.test.ts +0 -472
- package/src/web/routes/channels.ts +0 -311
- package/src/web/routes/oauth-providers.ts +0 -42
- package/src/web/routes/secrets.test.ts +0 -220
- package/src/web/routes/secrets.ts +0 -317
- package/src/web/routes/sessions.ts +0 -123
- package/src/web/routes/settings.test.ts +0 -106
- package/src/web/routes/settings.ts +0 -247
- package/src/web/routes/setup-status.ts +0 -205
- package/src/web/routes/vaults.test.ts +0 -389
- package/src/web/routes/vaults.ts +0 -225
- package/src/web/server-version.test.ts +0 -16
- package/src/web/server.ts +0 -1024
- package/src/web/services-manifest.test.ts +0 -148
- package/src/web/services-manifest.ts +0 -66
- package/src/web/static-serve.test.ts +0 -255
- package/src/web/static-serve.ts +0 -104
- package/src/web/telegram-validate.test.ts +0 -116
- package/src/web/telegram-validate.ts +0 -107
- package/src/web/vault-proxy.test.ts +0 -214
- package/src/web/vault-proxy.ts +0 -120
- package/src/web/wire-channel.ts +0 -181
- package/src/webhook-server.ts +0 -134
- package/vitest.config.ts +0 -18
- package/web/README.md +0 -63
- package/web/ui/index.html +0 -13
- package/web/ui/package.json +0 -35
- package/web/ui/pnpm-lock.yaml +0 -2164
- package/web/ui/scripts/verify-base.mjs +0 -31
- package/web/ui/src/App.tsx +0 -88
- package/web/ui/src/components/ActivityFeed.tsx +0 -444
- package/web/ui/src/components/AgentGroupPicker.tsx +0 -263
- package/web/ui/src/components/AgentProviderCards.tsx +0 -220
- package/web/ui/src/components/CredentialForm.tsx +0 -214
- package/web/ui/src/components/ScopeGrants.tsx +0 -74
- package/web/ui/src/components/StatusDot.tsx +0 -43
- package/web/ui/src/components/VaultPicker.tsx +0 -127
- package/web/ui/src/components/setup/AdapterInstallStep.tsx +0 -178
- package/web/ui/src/components/setup/AgentGroupStep.tsx +0 -43
- package/web/ui/src/components/setup/ChannelPickStep.tsx +0 -74
- package/web/ui/src/components/setup/DoneStep.tsx +0 -49
- package/web/ui/src/components/setup/PrereqStep.tsx +0 -129
- package/web/ui/src/components/setup/TestConnectionStep.tsx +0 -108
- package/web/ui/src/components/setup/TestMessageStep.tsx +0 -104
- package/web/ui/src/components/setup/WireChannelStep.tsx +0 -166
- package/web/ui/src/components/setup/types.ts +0 -105
- package/web/ui/src/lib/api.test.ts +0 -410
- package/web/ui/src/lib/api.ts +0 -1248
- package/web/ui/src/lib/auth.test.ts +0 -352
- package/web/ui/src/lib/auth.ts +0 -405
- package/web/ui/src/lib/channel-adapters.ts +0 -136
- package/web/ui/src/main.tsx +0 -19
- package/web/ui/src/routes/ApprovalsList.tsx +0 -294
- package/web/ui/src/routes/Apps.tsx +0 -613
- package/web/ui/src/routes/ChannelWireDetail.test.tsx +0 -233
- package/web/ui/src/routes/ChannelWireDetail.tsx +0 -403
- package/web/ui/src/routes/ChannelsList.tsx +0 -158
- package/web/ui/src/routes/GroupDetail.test.tsx +0 -206
- package/web/ui/src/routes/GroupDetail.tsx +0 -880
- package/web/ui/src/routes/GroupList.tsx +0 -187
- package/web/ui/src/routes/MessagingGroupDetail.test.tsx +0 -233
- package/web/ui/src/routes/MessagingGroupDetail.tsx +0 -306
- package/web/ui/src/routes/NewGroupWizard.tsx +0 -390
- package/web/ui/src/routes/OAuthCallback.tsx +0 -56
- package/web/ui/src/routes/SecretsList.tsx +0 -942
- package/web/ui/src/routes/SessionsList.tsx +0 -220
- package/web/ui/src/routes/SettingsAgentProvider.tsx +0 -109
- package/web/ui/src/routes/SettingsApprovals.tsx +0 -234
- package/web/ui/src/routes/SetupWizard.tsx +0 -219
- package/web/ui/src/routes/VaultDetail.test.tsx +0 -363
- package/web/ui/src/routes/VaultDetail.tsx +0 -960
- package/web/ui/src/routes/VaultsList.tsx +0 -295
- package/web/ui/src/routes/WireChannelPage.tsx +0 -413
- package/web/ui/src/styles.css +0 -608
- package/web/ui/src/test/setup.ts +0 -23
- package/web/ui/src/vite-env.d.ts +0 -10
- package/web/ui/vite.config.ts +0 -34
- package/web/ui/vitest.config.ts +0 -25
|
@@ -1,243 +0,0 @@
|
|
|
1
|
-
---
|
|
2
|
-
name: add-atomic-chat-tool
|
|
3
|
-
description: Add Atomic Chat MCP server so the container agent can call local models served by the Atomic Chat desktop app via its OpenAI-compatible API.
|
|
4
|
-
---
|
|
5
|
-
|
|
6
|
-
# Add Atomic Chat Integration
|
|
7
|
-
|
|
8
|
-
This skill adds a stdio-based MCP server that exposes models running in the local [Atomic Chat](https://github.com/AtomicBot-ai/Atomic-Chat) desktop app as tools for the container agent. Claude remains the orchestrator but can offload work to local models served by Atomic Chat on `http://127.0.0.1:1337/v1` (OpenAI-compatible).
|
|
9
|
-
|
|
10
|
-
Tools exposed:
|
|
11
|
-
- `atomic_chat_list_models` — list models currently available in Atomic Chat (`GET /v1/models`)
|
|
12
|
-
- `atomic_chat_generate` — send a prompt to a specified model and return the response (`POST /v1/chat/completions`)
|
|
13
|
-
|
|
14
|
-
Model management (download, delete) is done through the **Atomic Chat desktop UI** — the app is a fork of Jan and manages its own model library.
|
|
15
|
-
|
|
16
|
-
The skill ships the MCP server source in this folder and copies it into the agent-runner tree at install time, then wires it up with small edits to `index.ts`, `providers/claude.ts`, and `container-runner.ts`. No branch merge — all edits are additive and idempotent.
|
|
17
|
-
|
|
18
|
-
## Phase 1: Pre-flight
|
|
19
|
-
|
|
20
|
-
### Check if already applied
|
|
21
|
-
|
|
22
|
-
Check if `container/agent-runner/src/atomic-chat-mcp-stdio.ts` exists. If it does, skip to Phase 3 (Configure).
|
|
23
|
-
|
|
24
|
-
### Check prerequisites
|
|
25
|
-
|
|
26
|
-
Verify Atomic Chat is installed and its local API server is running. On the host:
|
|
27
|
-
|
|
28
|
-
```bash
|
|
29
|
-
curl -s http://127.0.0.1:1337/v1/models | head
|
|
30
|
-
```
|
|
31
|
-
|
|
32
|
-
If the request fails:
|
|
33
|
-
|
|
34
|
-
1. Install Atomic Chat from the [latest release](https://github.com/AtomicBot-ai/Atomic-Chat/releases) (macOS only for now — `atomic-chat.dmg`).
|
|
35
|
-
2. Open the app.
|
|
36
|
-
3. Open **Settings → Local API Server** and make sure it's enabled on port `1337`.
|
|
37
|
-
4. Go to the **Hub** (or **Models**) tab and download at least one model (e.g. Llama 3.2 3B, Qwen 2.5 Coder 7B).
|
|
38
|
-
5. Load the model once by sending any message in Atomic Chat's UI to warm it up.
|
|
39
|
-
|
|
40
|
-
## Phase 2: Apply Code Changes
|
|
41
|
-
|
|
42
|
-
### Copy the MCP server source
|
|
43
|
-
|
|
44
|
-
```bash
|
|
45
|
-
cp .claude/skills/add-atomic-chat-tool/atomic-chat-mcp-stdio.ts container/agent-runner/src/atomic-chat-mcp-stdio.ts
|
|
46
|
-
```
|
|
47
|
-
|
|
48
|
-
### Register the MCP server in the agent-runner
|
|
49
|
-
|
|
50
|
-
Edit `container/agent-runner/src/index.ts`. Find the `mcpServers` object that currently looks like this:
|
|
51
|
-
|
|
52
|
-
```ts
|
|
53
|
-
const mcpServers: Record<string, { command: string; args: string[]; env: Record<string, string> }> = {
|
|
54
|
-
nanoclaw: {
|
|
55
|
-
command: 'bun',
|
|
56
|
-
args: ['run', mcpServerPath],
|
|
57
|
-
env: {},
|
|
58
|
-
},
|
|
59
|
-
};
|
|
60
|
-
```
|
|
61
|
-
|
|
62
|
-
Add an `atomic_chat` entry alongside `nanoclaw`:
|
|
63
|
-
|
|
64
|
-
```ts
|
|
65
|
-
const mcpServers: Record<string, { command: string; args: string[]; env: Record<string, string> }> = {
|
|
66
|
-
nanoclaw: {
|
|
67
|
-
command: 'bun',
|
|
68
|
-
args: ['run', mcpServerPath],
|
|
69
|
-
env: {},
|
|
70
|
-
},
|
|
71
|
-
atomic_chat: {
|
|
72
|
-
command: 'bun',
|
|
73
|
-
args: ['run', path.join(__dirname, 'atomic-chat-mcp-stdio.ts')],
|
|
74
|
-
env: {
|
|
75
|
-
...(process.env.ATOMIC_CHAT_HOST ? { ATOMIC_CHAT_HOST: process.env.ATOMIC_CHAT_HOST } : {}),
|
|
76
|
-
...(process.env.ATOMIC_CHAT_API_KEY ? { ATOMIC_CHAT_API_KEY: process.env.ATOMIC_CHAT_API_KEY } : {}),
|
|
77
|
-
},
|
|
78
|
-
},
|
|
79
|
-
};
|
|
80
|
-
```
|
|
81
|
-
|
|
82
|
-
### Add the tool glob to the allowlist
|
|
83
|
-
|
|
84
|
-
Edit `container/agent-runner/src/providers/claude.ts`. Find `'mcp__nanoclaw__*',` in the `TOOL_ALLOWLIST` array and add `'mcp__atomic_chat__*',` on the following line:
|
|
85
|
-
|
|
86
|
-
```ts
|
|
87
|
-
'mcp__nanoclaw__*',
|
|
88
|
-
'mcp__atomic_chat__*',
|
|
89
|
-
];
|
|
90
|
-
```
|
|
91
|
-
|
|
92
|
-
### Forward host env vars into the container
|
|
93
|
-
|
|
94
|
-
Edit `src/container-runner.ts` in `buildContainerArgs`. Find the `TZ` env line:
|
|
95
|
-
|
|
96
|
-
```ts
|
|
97
|
-
args.push('-e', `TZ=${TIMEZONE}`);
|
|
98
|
-
```
|
|
99
|
-
|
|
100
|
-
Add ATOMIC_CHAT forwarding right after it:
|
|
101
|
-
|
|
102
|
-
```ts
|
|
103
|
-
args.push('-e', `TZ=${TIMEZONE}`);
|
|
104
|
-
|
|
105
|
-
// Atomic Chat MCP tool: forward host overrides if set (default is host.docker.internal:1337).
|
|
106
|
-
if (process.env.ATOMIC_CHAT_HOST) {
|
|
107
|
-
args.push('-e', `ATOMIC_CHAT_HOST=${process.env.ATOMIC_CHAT_HOST}`);
|
|
108
|
-
}
|
|
109
|
-
if (process.env.ATOMIC_CHAT_API_KEY) {
|
|
110
|
-
args.push('-e', `ATOMIC_CHAT_API_KEY=${process.env.ATOMIC_CHAT_API_KEY}`);
|
|
111
|
-
}
|
|
112
|
-
```
|
|
113
|
-
|
|
114
|
-
### Surface `[ATOMIC]` log lines at info level
|
|
115
|
-
|
|
116
|
-
In the same file, find the stderr logger:
|
|
117
|
-
|
|
118
|
-
```ts
|
|
119
|
-
container.stderr?.on('data', (data) => {
|
|
120
|
-
for (const line of data.toString().trim().split('\n')) {
|
|
121
|
-
if (line) log.debug(line, { container: agentGroup.folder });
|
|
122
|
-
}
|
|
123
|
-
});
|
|
124
|
-
```
|
|
125
|
-
|
|
126
|
-
Replace it with:
|
|
127
|
-
|
|
128
|
-
```ts
|
|
129
|
-
container.stderr?.on('data', (data) => {
|
|
130
|
-
for (const line of data.toString().trim().split('\n')) {
|
|
131
|
-
if (!line) continue;
|
|
132
|
-
if (line.includes('[ATOMIC]')) {
|
|
133
|
-
log.info(line, { container: agentGroup.folder });
|
|
134
|
-
} else {
|
|
135
|
-
log.debug(line, { container: agentGroup.folder });
|
|
136
|
-
}
|
|
137
|
-
}
|
|
138
|
-
});
|
|
139
|
-
```
|
|
140
|
-
|
|
141
|
-
### Add env-var stubs to `.env.example`
|
|
142
|
-
|
|
143
|
-
Append to `.env.example`:
|
|
144
|
-
|
|
145
|
-
```bash
|
|
146
|
-
# Atomic Chat MCP tool (.claude/skills/add-atomic-chat-tool)
|
|
147
|
-
# Override the host where Atomic Chat exposes its OpenAI-compatible API.
|
|
148
|
-
# Default: http://host.docker.internal:1337 (with fallback to localhost)
|
|
149
|
-
# ATOMIC_CHAT_HOST=http://host.docker.internal:1337
|
|
150
|
-
|
|
151
|
-
# Optional API key. Leave unset for a local Atomic Chat install — it does not require auth.
|
|
152
|
-
# ATOMIC_CHAT_API_KEY=
|
|
153
|
-
```
|
|
154
|
-
|
|
155
|
-
### Validate code changes
|
|
156
|
-
|
|
157
|
-
```bash
|
|
158
|
-
pnpm run build
|
|
159
|
-
pnpm exec tsc -p container/agent-runner/tsconfig.json --noEmit
|
|
160
|
-
./container/build.sh
|
|
161
|
-
```
|
|
162
|
-
|
|
163
|
-
All three must be clean before proceeding.
|
|
164
|
-
|
|
165
|
-
## Phase 3: Configure
|
|
166
|
-
|
|
167
|
-
### Set Atomic Chat host (optional)
|
|
168
|
-
|
|
169
|
-
By default, the MCP server connects to `http://host.docker.internal:1337` (Docker Desktop) with a fallback to `localhost`. To use a custom host, add to `.env`:
|
|
170
|
-
|
|
171
|
-
```bash
|
|
172
|
-
ATOMIC_CHAT_HOST=http://your-atomic-chat-host:1337
|
|
173
|
-
```
|
|
174
|
-
|
|
175
|
-
### Set API key (optional)
|
|
176
|
-
|
|
177
|
-
Atomic Chat does **not require authentication** when running locally — leave this unset. Only set it if you've put Atomic Chat behind a reverse proxy that enforces auth:
|
|
178
|
-
|
|
179
|
-
```bash
|
|
180
|
-
ATOMIC_CHAT_API_KEY=sk-...
|
|
181
|
-
```
|
|
182
|
-
|
|
183
|
-
### Restart the service
|
|
184
|
-
|
|
185
|
-
```bash
|
|
186
|
-
launchctl kickstart -k gui/$(id -u)/com.nanoclaw # macOS
|
|
187
|
-
# Linux: systemctl --user restart nanoclaw
|
|
188
|
-
```
|
|
189
|
-
|
|
190
|
-
## Phase 4: Verify
|
|
191
|
-
|
|
192
|
-
### Test inference
|
|
193
|
-
|
|
194
|
-
Tell the user:
|
|
195
|
-
|
|
196
|
-
> Send a message like: "use atomic chat to tell me the capital of France"
|
|
197
|
-
>
|
|
198
|
-
> The agent should use `atomic_chat_list_models` to find available models, then `atomic_chat_generate` to get a response.
|
|
199
|
-
|
|
200
|
-
### Check logs if needed
|
|
201
|
-
|
|
202
|
-
```bash
|
|
203
|
-
tail -f logs/nanoclaw.log | grep -i atomic
|
|
204
|
-
```
|
|
205
|
-
|
|
206
|
-
Look for:
|
|
207
|
-
- `[ATOMIC] Listing models...` — list request started
|
|
208
|
-
- `[ATOMIC] Found N models` — models discovered
|
|
209
|
-
- `[ATOMIC] >>> Generating with <model>` — generation started
|
|
210
|
-
- `[ATOMIC] <<< Done: <model> | Xs | N tokens | M chars` — generation completed
|
|
211
|
-
|
|
212
|
-
## Troubleshooting
|
|
213
|
-
|
|
214
|
-
### Agent says "Atomic Chat is not installed" or tries to run a CLI
|
|
215
|
-
|
|
216
|
-
The agent is looking for a CLI that doesn't exist instead of using the MCP tools. This means:
|
|
217
|
-
1. The MCP server wasn't copied — check `container/agent-runner/src/atomic-chat-mcp-stdio.ts` exists
|
|
218
|
-
2. The MCP server wasn't registered — check `container/agent-runner/src/index.ts` has the `atomic_chat` entry in `mcpServers`
|
|
219
|
-
3. The allowlist wasn't updated — check `container/agent-runner/src/providers/claude.ts` includes `mcp__atomic_chat__*` in `TOOL_ALLOWLIST`
|
|
220
|
-
4. The container wasn't rebuilt — run `./container/build.sh`
|
|
221
|
-
|
|
222
|
-
### "Failed to connect to Atomic Chat"
|
|
223
|
-
|
|
224
|
-
1. Verify the host API is reachable: `curl http://127.0.0.1:1337/v1/models`
|
|
225
|
-
2. Confirm the Local API Server is enabled in Atomic Chat's settings
|
|
226
|
-
3. Check Docker can reach the host: `docker run --rm curlimages/curl curl -s http://host.docker.internal:1337/v1/models`
|
|
227
|
-
4. If using a custom host, check `ATOMIC_CHAT_HOST` in `.env`
|
|
228
|
-
|
|
229
|
-
### `model not found` / 404 on generate
|
|
230
|
-
|
|
231
|
-
The model ID passed to `atomic_chat_generate` must exactly match one of the IDs returned by `atomic_chat_list_models`. Ask the agent to list models first, then pick one from that list.
|
|
232
|
-
|
|
233
|
-
### Slow first response
|
|
234
|
-
|
|
235
|
-
Atomic Chat lazy-loads models into memory on first use. The initial call may take longer while the model warms up. Subsequent calls against the same model are fast.
|
|
236
|
-
|
|
237
|
-
### Agent doesn't use Atomic Chat tools
|
|
238
|
-
|
|
239
|
-
The agent may not know about the tools. Try being explicit: "use the atomic_chat_generate tool with llama3.2-3b-instruct to answer: ..."
|
|
240
|
-
|
|
241
|
-
### Context window or output size issues
|
|
242
|
-
|
|
243
|
-
Atomic Chat respects each model's native context length. If you hit limits, pass `max_tokens` explicitly when calling `atomic_chat_generate`, or switch to a model with a larger context window in the Atomic Chat UI.
|
|
@@ -1,229 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Atomic Chat MCP Server for NanoClaw
|
|
3
|
-
* Exposes local Atomic Chat models (OpenAI-compatible, /v1) as tools for the container agent.
|
|
4
|
-
* Uses host.docker.internal to reach the host's Atomic Chat desktop app from Docker.
|
|
5
|
-
*/
|
|
6
|
-
|
|
7
|
-
import { McpServer } from '@modelcontextprotocol/sdk/server/mcp.js';
|
|
8
|
-
import { StdioServerTransport } from '@modelcontextprotocol/sdk/server/stdio.js';
|
|
9
|
-
import { z } from 'zod';
|
|
10
|
-
|
|
11
|
-
import fs from 'fs';
|
|
12
|
-
import path from 'path';
|
|
13
|
-
|
|
14
|
-
const ATOMIC_CHAT_HOST =
|
|
15
|
-
process.env.ATOMIC_CHAT_HOST || 'http://host.docker.internal:1337';
|
|
16
|
-
const ATOMIC_CHAT_API_KEY = process.env.ATOMIC_CHAT_API_KEY || '';
|
|
17
|
-
const ATOMIC_CHAT_STATUS_FILE = '/workspace/ipc/atomic_chat_status.json';
|
|
18
|
-
|
|
19
|
-
function log(msg: string): void {
|
|
20
|
-
console.error(`[ATOMIC] ${msg}`);
|
|
21
|
-
}
|
|
22
|
-
|
|
23
|
-
function writeStatus(status: string, detail?: string): void {
|
|
24
|
-
try {
|
|
25
|
-
const data = { status, detail, timestamp: new Date().toISOString() };
|
|
26
|
-
const tmpPath = `${ATOMIC_CHAT_STATUS_FILE}.tmp`;
|
|
27
|
-
fs.mkdirSync(path.dirname(ATOMIC_CHAT_STATUS_FILE), { recursive: true });
|
|
28
|
-
fs.writeFileSync(tmpPath, JSON.stringify(data));
|
|
29
|
-
fs.renameSync(tmpPath, ATOMIC_CHAT_STATUS_FILE);
|
|
30
|
-
} catch {
|
|
31
|
-
/* best-effort */
|
|
32
|
-
}
|
|
33
|
-
}
|
|
34
|
-
|
|
35
|
-
async function atomicFetch(
|
|
36
|
-
apiPath: string,
|
|
37
|
-
options?: RequestInit,
|
|
38
|
-
): Promise<Response> {
|
|
39
|
-
const url = `${ATOMIC_CHAT_HOST}${apiPath}`;
|
|
40
|
-
const headers: Record<string, string> = {
|
|
41
|
-
...((options?.headers as Record<string, string>) || {}),
|
|
42
|
-
};
|
|
43
|
-
if (ATOMIC_CHAT_API_KEY) {
|
|
44
|
-
headers.Authorization = `Bearer ${ATOMIC_CHAT_API_KEY}`;
|
|
45
|
-
}
|
|
46
|
-
const finalOptions: RequestInit = { ...options, headers };
|
|
47
|
-
try {
|
|
48
|
-
return await fetch(url, finalOptions);
|
|
49
|
-
} catch (err) {
|
|
50
|
-
// Fallback to localhost if host.docker.internal fails
|
|
51
|
-
if (ATOMIC_CHAT_HOST.includes('host.docker.internal')) {
|
|
52
|
-
const fallbackUrl = url.replace('host.docker.internal', 'localhost');
|
|
53
|
-
return await fetch(fallbackUrl, finalOptions);
|
|
54
|
-
}
|
|
55
|
-
throw err;
|
|
56
|
-
}
|
|
57
|
-
}
|
|
58
|
-
|
|
59
|
-
const server = new McpServer({
|
|
60
|
-
name: 'atomic_chat',
|
|
61
|
-
version: '1.0.0',
|
|
62
|
-
});
|
|
63
|
-
|
|
64
|
-
server.tool(
|
|
65
|
-
'atomic_chat_list_models',
|
|
66
|
-
'List all models available in the local Atomic Chat desktop app. Use this to see which models are loaded before calling atomic_chat_generate.',
|
|
67
|
-
{},
|
|
68
|
-
async () => {
|
|
69
|
-
log('Listing models...');
|
|
70
|
-
writeStatus('listing', 'Listing available models');
|
|
71
|
-
try {
|
|
72
|
-
const res = await atomicFetch('/v1/models');
|
|
73
|
-
if (!res.ok) {
|
|
74
|
-
return {
|
|
75
|
-
content: [
|
|
76
|
-
{
|
|
77
|
-
type: 'text' as const,
|
|
78
|
-
text: `Atomic Chat API error: ${res.status} ${res.statusText}`,
|
|
79
|
-
},
|
|
80
|
-
],
|
|
81
|
-
isError: true,
|
|
82
|
-
};
|
|
83
|
-
}
|
|
84
|
-
|
|
85
|
-
const data = (await res.json()) as {
|
|
86
|
-
data?: Array<{ id: string; owned_by?: string }>;
|
|
87
|
-
};
|
|
88
|
-
const models = data.data || [];
|
|
89
|
-
|
|
90
|
-
if (models.length === 0) {
|
|
91
|
-
return {
|
|
92
|
-
content: [
|
|
93
|
-
{
|
|
94
|
-
type: 'text' as const,
|
|
95
|
-
text: 'No models available. Open Atomic Chat on the host and download a model from the Hub.',
|
|
96
|
-
},
|
|
97
|
-
],
|
|
98
|
-
};
|
|
99
|
-
}
|
|
100
|
-
|
|
101
|
-
const list = models
|
|
102
|
-
.map((m) => `- ${m.id}${m.owned_by ? ` (${m.owned_by})` : ''}`)
|
|
103
|
-
.join('\n');
|
|
104
|
-
|
|
105
|
-
log(`Found ${models.length} models`);
|
|
106
|
-
return {
|
|
107
|
-
content: [
|
|
108
|
-
{ type: 'text' as const, text: `Available models:\n${list}` },
|
|
109
|
-
],
|
|
110
|
-
};
|
|
111
|
-
} catch (err) {
|
|
112
|
-
return {
|
|
113
|
-
content: [
|
|
114
|
-
{
|
|
115
|
-
type: 'text' as const,
|
|
116
|
-
text: `Failed to connect to Atomic Chat at ${ATOMIC_CHAT_HOST}: ${err instanceof Error ? err.message : String(err)}`,
|
|
117
|
-
},
|
|
118
|
-
],
|
|
119
|
-
isError: true,
|
|
120
|
-
};
|
|
121
|
-
}
|
|
122
|
-
},
|
|
123
|
-
);
|
|
124
|
-
|
|
125
|
-
server.tool(
|
|
126
|
-
'atomic_chat_generate',
|
|
127
|
-
'Send a prompt to a local Atomic Chat model and get a response. Good for cheaper/faster tasks like summarization, translation, or general queries. Use atomic_chat_list_models first to see available models.',
|
|
128
|
-
{
|
|
129
|
-
model: z
|
|
130
|
-
.string()
|
|
131
|
-
.describe(
|
|
132
|
-
'The model ID as returned by atomic_chat_list_models (e.g. "llama3.2-3b-instruct")',
|
|
133
|
-
),
|
|
134
|
-
prompt: z.string().describe('The prompt to send to the model'),
|
|
135
|
-
system: z
|
|
136
|
-
.string()
|
|
137
|
-
.optional()
|
|
138
|
-
.describe('Optional system prompt to set model behavior'),
|
|
139
|
-
temperature: z
|
|
140
|
-
.number()
|
|
141
|
-
.optional()
|
|
142
|
-
.describe('Sampling temperature (0.0–2.0). Defaults to model default.'),
|
|
143
|
-
max_tokens: z
|
|
144
|
-
.number()
|
|
145
|
-
.optional()
|
|
146
|
-
.describe('Maximum number of tokens to generate in the response.'),
|
|
147
|
-
},
|
|
148
|
-
async (args) => {
|
|
149
|
-
log(`>>> Generating with ${args.model} (${args.prompt.length} chars)...`);
|
|
150
|
-
writeStatus('generating', `Generating with ${args.model}`);
|
|
151
|
-
try {
|
|
152
|
-
const messages: Array<{ role: string; content: string }> = [];
|
|
153
|
-
if (args.system) {
|
|
154
|
-
messages.push({ role: 'system', content: args.system });
|
|
155
|
-
}
|
|
156
|
-
messages.push({ role: 'user', content: args.prompt });
|
|
157
|
-
|
|
158
|
-
const body: Record<string, unknown> = {
|
|
159
|
-
model: args.model,
|
|
160
|
-
messages,
|
|
161
|
-
stream: false,
|
|
162
|
-
};
|
|
163
|
-
if (args.temperature !== undefined) body.temperature = args.temperature;
|
|
164
|
-
if (args.max_tokens !== undefined) body.max_tokens = args.max_tokens;
|
|
165
|
-
|
|
166
|
-
const startedAt = Date.now();
|
|
167
|
-
const res = await atomicFetch('/v1/chat/completions', {
|
|
168
|
-
method: 'POST',
|
|
169
|
-
headers: { 'Content-Type': 'application/json' },
|
|
170
|
-
body: JSON.stringify(body),
|
|
171
|
-
});
|
|
172
|
-
|
|
173
|
-
if (!res.ok) {
|
|
174
|
-
const errorText = await res.text();
|
|
175
|
-
return {
|
|
176
|
-
content: [
|
|
177
|
-
{
|
|
178
|
-
type: 'text' as const,
|
|
179
|
-
text: `Atomic Chat error (${res.status}): ${errorText}`,
|
|
180
|
-
},
|
|
181
|
-
],
|
|
182
|
-
isError: true,
|
|
183
|
-
};
|
|
184
|
-
}
|
|
185
|
-
|
|
186
|
-
const data = (await res.json()) as {
|
|
187
|
-
choices?: Array<{ message?: { content?: string } }>;
|
|
188
|
-
usage?: {
|
|
189
|
-
prompt_tokens?: number;
|
|
190
|
-
completion_tokens?: number;
|
|
191
|
-
total_tokens?: number;
|
|
192
|
-
};
|
|
193
|
-
};
|
|
194
|
-
|
|
195
|
-
const response = data.choices?.[0]?.message?.content ?? '';
|
|
196
|
-
const elapsedSec = ((Date.now() - startedAt) / 1000).toFixed(1);
|
|
197
|
-
const completionTokens = data.usage?.completion_tokens;
|
|
198
|
-
|
|
199
|
-
const meta = `\n\n[${args.model} | ${elapsedSec}s${
|
|
200
|
-
completionTokens !== undefined ? ` | ${completionTokens} tokens` : ''
|
|
201
|
-
}]`;
|
|
202
|
-
|
|
203
|
-
log(
|
|
204
|
-
`<<< Done: ${args.model} | ${elapsedSec}s | ${
|
|
205
|
-
completionTokens ?? '?'
|
|
206
|
-
} tokens | ${response.length} chars`,
|
|
207
|
-
);
|
|
208
|
-
writeStatus(
|
|
209
|
-
'done',
|
|
210
|
-
`${args.model} | ${elapsedSec}s | ${completionTokens ?? '?'} tokens`,
|
|
211
|
-
);
|
|
212
|
-
|
|
213
|
-
return { content: [{ type: 'text' as const, text: response + meta }] };
|
|
214
|
-
} catch (err) {
|
|
215
|
-
return {
|
|
216
|
-
content: [
|
|
217
|
-
{
|
|
218
|
-
type: 'text' as const,
|
|
219
|
-
text: `Failed to call Atomic Chat: ${err instanceof Error ? err.message : String(err)}`,
|
|
220
|
-
},
|
|
221
|
-
],
|
|
222
|
-
isError: true,
|
|
223
|
-
};
|
|
224
|
-
}
|
|
225
|
-
},
|
|
226
|
-
);
|
|
227
|
-
|
|
228
|
-
const transport = new StdioServerTransport();
|
|
229
|
-
await server.connect(transport);
|
|
@@ -1,161 +0,0 @@
|
|
|
1
|
-
---
|
|
2
|
-
name: add-codex
|
|
3
|
-
description: Use Codex (CLI + AppServer) as the full agent provider — planning, tool orchestration, native compaction, MCP tools, session resume — in place of the Claude Agent SDK. ChatGPT subscription or OPENAI_API_KEY. Per-group via agent_provider. Distinct from using OpenAI as an MCP tool (where Claude remains the planner).
|
|
4
|
-
---
|
|
5
|
-
|
|
6
|
-
# Codex agent provider
|
|
7
|
-
|
|
8
|
-
NanoClaw runs agents in a long-lived **poll loop** inside the container. The backend is selected with **`AGENT_PROVIDER`** (`claude` | `opencode` | `codex` | `mock`).
|
|
9
|
-
|
|
10
|
-
Trunk ships with only the `claude` provider baked in. This skill copies the Codex provider files in from the `providers` branch, wires them into the host and container barrels, updates the Dockerfile to install the Codex CLI, and rebuilds the image.
|
|
11
|
-
|
|
12
|
-
The Codex provider runs `codex app-server` as a child process and speaks JSON-RPC over stdio. That gives it native session resume, streaming events, MCP tool access, and `thread/compact/start` compaction — same feature bar as the Claude Agent SDK, without the Anthropic-only lock-in.
|
|
13
|
-
|
|
14
|
-
## Install
|
|
15
|
-
|
|
16
|
-
### Pre-flight
|
|
17
|
-
|
|
18
|
-
If all of the following are already present, skip to **Configuration**:
|
|
19
|
-
|
|
20
|
-
- `src/providers/codex.ts`
|
|
21
|
-
- `container/agent-runner/src/providers/codex.ts`
|
|
22
|
-
- `container/agent-runner/src/providers/codex-app-server.ts`
|
|
23
|
-
- `container/agent-runner/src/providers/codex.factory.test.ts`
|
|
24
|
-
- `import './codex.js';` line in `src/providers/index.ts`
|
|
25
|
-
- `import './codex.js';` line in `container/agent-runner/src/providers/index.ts`
|
|
26
|
-
- `ARG CODEX_VERSION` and `"@openai/codex@${CODEX_VERSION}"` in the pnpm global-install block in `container/Dockerfile`
|
|
27
|
-
|
|
28
|
-
Missing pieces — continue below. All steps are idempotent; re-running is safe.
|
|
29
|
-
|
|
30
|
-
### 1. Fetch the providers branch
|
|
31
|
-
|
|
32
|
-
```bash
|
|
33
|
-
git fetch origin providers
|
|
34
|
-
```
|
|
35
|
-
|
|
36
|
-
### 2. Copy the Codex source files
|
|
37
|
-
|
|
38
|
-
Wholesale copies (owned entirely by this skill — user edits to these files won't survive a re-run, as designed):
|
|
39
|
-
|
|
40
|
-
```bash
|
|
41
|
-
git show origin/providers:src/providers/codex.ts > src/providers/codex.ts
|
|
42
|
-
git show origin/providers:container/agent-runner/src/providers/codex.ts > container/agent-runner/src/providers/codex.ts
|
|
43
|
-
git show origin/providers:container/agent-runner/src/providers/codex-app-server.ts > container/agent-runner/src/providers/codex-app-server.ts
|
|
44
|
-
git show origin/providers:container/agent-runner/src/providers/codex.factory.test.ts > container/agent-runner/src/providers/codex.factory.test.ts
|
|
45
|
-
```
|
|
46
|
-
|
|
47
|
-
### 3. Append the self-registration imports
|
|
48
|
-
|
|
49
|
-
Each barrel gets one line — alphabetical placement keeps diffs small.
|
|
50
|
-
|
|
51
|
-
`src/providers/index.ts`:
|
|
52
|
-
|
|
53
|
-
```typescript
|
|
54
|
-
import './codex.js';
|
|
55
|
-
```
|
|
56
|
-
|
|
57
|
-
`container/agent-runner/src/providers/index.ts`:
|
|
58
|
-
|
|
59
|
-
```typescript
|
|
60
|
-
import './codex.js';
|
|
61
|
-
```
|
|
62
|
-
|
|
63
|
-
### 4. Add the Codex CLI to the container Dockerfile
|
|
64
|
-
|
|
65
|
-
Two edits to `container/Dockerfile`, both idempotent (skip if already present):
|
|
66
|
-
|
|
67
|
-
**(a)** In the "Pin CLI versions" ARG block (around line 18), add after `ARG CLAUDE_CODE_VERSION=...`:
|
|
68
|
-
|
|
69
|
-
```dockerfile
|
|
70
|
-
ARG CODEX_VERSION=0.124.0
|
|
71
|
-
```
|
|
72
|
-
|
|
73
|
-
**(b)** Add a new standalone `RUN` block for the Codex CLI, after the existing per-CLI install blocks (around line 106, right after the `@anthropic-ai/claude-code` block). The Dockerfile splits each global CLI into its own layer for cache granularity — keep that pattern; do not collapse them into a single combined `pnpm install -g` call:
|
|
74
|
-
|
|
75
|
-
```dockerfile
|
|
76
|
-
RUN --mount=type=cache,target=/root/.cache/pnpm \
|
|
77
|
-
pnpm install -g "@openai/codex@${CODEX_VERSION}"
|
|
78
|
-
```
|
|
79
|
-
|
|
80
|
-
Note: **no agent-runner package dependency** — Codex is a CLI binary, not a library. Unlike OpenCode, there's nothing to add to `container/agent-runner/package.json`.
|
|
81
|
-
|
|
82
|
-
### 5. Build
|
|
83
|
-
|
|
84
|
-
```bash
|
|
85
|
-
pnpm run build # host
|
|
86
|
-
pnpm exec tsc -p container/agent-runner/tsconfig.json --noEmit # container typecheck
|
|
87
|
-
./container/build.sh # agent image
|
|
88
|
-
```
|
|
89
|
-
|
|
90
|
-
## Configuration
|
|
91
|
-
|
|
92
|
-
Codex supports two primary auth paths and one experimental BYO-endpoint path. Pick the one that matches your setup.
|
|
93
|
-
|
|
94
|
-
### Option A — ChatGPT subscription (recommended for individuals)
|
|
95
|
-
|
|
96
|
-
On the host (not inside the container), run Codex's OAuth login:
|
|
97
|
-
|
|
98
|
-
```bash
|
|
99
|
-
codex login
|
|
100
|
-
```
|
|
101
|
-
|
|
102
|
-
This writes `~/.codex/auth.json` with a subscription token. The host-side Codex provider ([src/providers/codex.ts](../../../src/providers/codex.ts)) copies `auth.json` into a per-session `~/.codex` directory mounted into the container — your host's own Codex CLI is never touched.
|
|
103
|
-
|
|
104
|
-
No `.env` variables required for this mode.
|
|
105
|
-
|
|
106
|
-
### Option B — API key (recommended for CI or API billing)
|
|
107
|
-
|
|
108
|
-
```env
|
|
109
|
-
OPENAI_API_KEY=sk-...
|
|
110
|
-
CODEX_MODEL=gpt-5.4-mini
|
|
111
|
-
```
|
|
112
|
-
|
|
113
|
-
The host forwards both variables into the container. If both subscription (`auth.json`) and `OPENAI_API_KEY` are present, Codex prefers the subscription.
|
|
114
|
-
|
|
115
|
-
### Option C — BYO OpenAI-compatible endpoint (experimental)
|
|
116
|
-
|
|
117
|
-
Codex's built-in `openai` provider honors the `OPENAI_BASE_URL` env var directly. Point it at any OpenAI-compatible endpoint — Groq, Together, self-hosted vLLM, an OpenAI proxy, etc.
|
|
118
|
-
|
|
119
|
-
```env
|
|
120
|
-
OPENAI_API_KEY=...
|
|
121
|
-
OPENAI_BASE_URL=https://api.groq.com/openai/v1
|
|
122
|
-
CODEX_MODEL=llama-3.3-70b-versatile
|
|
123
|
-
```
|
|
124
|
-
|
|
125
|
-
Codex also ships first-class local-runner flags — `codex --oss --local-provider ollama` or `--local-provider lmstudio` — that auto-detect a local server. To use those inside NanoClaw, set `CODEX_MODEL` to a model your local runner serves and add the corresponding base URL; see the Codex CLI docs for the full `model_provider = oss` configuration.
|
|
126
|
-
|
|
127
|
-
**Experimental caveat:** tool-calling quality depends on the model and endpoint. Not every OpenAI-compat provider implements the full function-calling spec, and smaller models (< 30B) often struggle with multi-step tool orchestration. Test before committing.
|
|
128
|
-
|
|
129
|
-
### Per group / per session
|
|
130
|
-
|
|
131
|
-
Set `"provider": "codex"` in the group's **`container.json`** (`groups/<folder>/container.json`) — the in-container runner reads `provider` from there, not from the DB. The DB columns **`agent_groups.agent_provider`** and **`sessions.agent_provider`** (session overrides group) only drive host-side provider contribution — per-session `~/.codex` mount, `OPENAI_*` / `CODEX_MODEL` env passthrough — and do not propagate into `container.json` at spawn time. Set both, or just edit `container.json`; if they disagree, the runner uses `container.json` and the host-side resolver falls back through session → group → `container.json` → `'claude'`.
|
|
132
|
-
|
|
133
|
-
`CODEX_MODEL` applies process-wide via `.env`; if you need different models for different groups, set them via `container_config.env` on the group.
|
|
134
|
-
|
|
135
|
-
Extra MCP servers still come from **`NANOCLAW_MCP_SERVERS`** / `container_config.mcpServers` on the host. The runner merges them into the same `mcpServers` object passed to all providers.
|
|
136
|
-
|
|
137
|
-
## Operational notes
|
|
138
|
-
|
|
139
|
-
- **Spawn-per-query:** Codex's app-server is spawned fresh per query invocation, matching the OpenCode pattern. No long-lived daemon to keep healthy across sessions.
|
|
140
|
-
- **Per-session `~/.codex` isolation:** each group gets its own copy of the host's `auth.json`. The container can rewrite `config.toml` freely on every wake without touching the host's Codex config.
|
|
141
|
-
- **Native compaction:** kicks in automatically at 40K cumulative input tokens between turns, via `thread/compact/start`. If compaction fails, the provider logs and continues uncompacted — no fatal error.
|
|
142
|
-
- **Approvals:** auto-accepted inside the container (the container is the sandbox; same posture as Claude/OpenCode).
|
|
143
|
-
- **Mid-turn input:** Codex turns don't accept mid-turn messages. Follow-up `push()` calls queue and drain between turns, matching the OpenCode pattern. The poll-loop only pushes between turns anyway, so no messages are dropped.
|
|
144
|
-
- **Stale thread recovery:** `isSessionInvalid` matches on stale-thread-ID errors (`thread not found`, `unknown thread`, etc.) so a cold-started app-server can recover cleanly when it sees a stored continuation it no longer has.
|
|
145
|
-
|
|
146
|
-
## Verify
|
|
147
|
-
|
|
148
|
-
```bash
|
|
149
|
-
grep -q "./codex.js" container/agent-runner/src/providers/index.ts && echo "container barrel: OK"
|
|
150
|
-
grep -q "./codex.js" src/providers/index.ts && echo "host barrel: OK"
|
|
151
|
-
grep -q "@openai/codex@" container/Dockerfile && echo "Dockerfile install: OK"
|
|
152
|
-
cd container/agent-runner && bun test src/providers/codex.factory.test.ts && cd -
|
|
153
|
-
```
|
|
154
|
-
|
|
155
|
-
After the image rebuild, set `agent_provider = 'codex'` on a test group and send a message. Successful round-trip looks like:
|
|
156
|
-
|
|
157
|
-
- `init` event with a stable thread ID as continuation
|
|
158
|
-
- One or more `activity` / `progress` events during the turn
|
|
159
|
-
- `result` event with the model's reply
|
|
160
|
-
|
|
161
|
-
If the agent hangs or errors, check `~/.codex/auth.json` exists on the host (Option A) or that `OPENAI_API_KEY` is forwarding correctly (Option B) — `docker exec` into a running container and `env | grep -i openai` to confirm.
|