nodmix 2026.5.25

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (827) hide show
  1. package/CHANGELOG.md +11573 -0
  2. package/LICENSE +21 -0
  3. package/README.md +486 -0
  4. package/docs/.i18n/README.md +81 -0
  5. package/docs/.i18n/ar-navigation.json +18 -0
  6. package/docs/.i18n/de-navigation.json +18 -0
  7. package/docs/.i18n/es-navigation.json +18 -0
  8. package/docs/.i18n/fr-navigation.json +18 -0
  9. package/docs/.i18n/glossary.ar.json +78 -0
  10. package/docs/.i18n/glossary.de.json +78 -0
  11. package/docs/.i18n/glossary.es.json +78 -0
  12. package/docs/.i18n/glossary.fa.json +78 -0
  13. package/docs/.i18n/glossary.fr.json +78 -0
  14. package/docs/.i18n/glossary.id.json +78 -0
  15. package/docs/.i18n/glossary.it.json +78 -0
  16. package/docs/.i18n/glossary.ja-JP.json +98 -0
  17. package/docs/.i18n/glossary.ko.json +78 -0
  18. package/docs/.i18n/glossary.nl.json +78 -0
  19. package/docs/.i18n/glossary.pl.json +78 -0
  20. package/docs/.i18n/glossary.pt-BR.json +78 -0
  21. package/docs/.i18n/glossary.th.json +78 -0
  22. package/docs/.i18n/glossary.tr.json +78 -0
  23. package/docs/.i18n/glossary.uk.json +78 -0
  24. package/docs/.i18n/glossary.vi.json +78 -0
  25. package/docs/.i18n/glossary.zh-CN.json +1002 -0
  26. package/docs/.i18n/glossary.zh-TW.json +78 -0
  27. package/docs/.i18n/id-navigation.json +18 -0
  28. package/docs/.i18n/it-navigation.json +18 -0
  29. package/docs/.i18n/ja-navigation.json +18 -0
  30. package/docs/.i18n/ko-navigation.json +18 -0
  31. package/docs/.i18n/pl-navigation.json +18 -0
  32. package/docs/.i18n/pt-BR-navigation.json +18 -0
  33. package/docs/.i18n/tr-navigation.json +18 -0
  34. package/docs/.i18n/translation-workflow.md +111 -0
  35. package/docs/.i18n/zh-Hans-navigation.json +542 -0
  36. package/docs/AGENTS.md +36 -0
  37. package/docs/announcements/bluebubbles-imessage.md +79 -0
  38. package/docs/assets/install-script.svg +1 -0
  39. package/docs/assets/macos-onboarding/01-macos-warning.jpeg +0 -0
  40. package/docs/assets/macos-onboarding/02-local-networks.jpeg +0 -0
  41. package/docs/assets/macos-onboarding/03-security-notice.png +0 -0
  42. package/docs/assets/macos-onboarding/04-choose-gateway.png +0 -0
  43. package/docs/assets/macos-onboarding/05-permissions.png +0 -0
  44. package/docs/assets/openclaw-logo-text-dark.png +0 -0
  45. package/docs/assets/openclaw-logo-text-dark.svg +418 -0
  46. package/docs/assets/openclaw-logo-text.png +0 -0
  47. package/docs/assets/openclaw-logo-text.svg +418 -0
  48. package/docs/assets/pixel-lobster.svg +60 -0
  49. package/docs/assets/pr/quick-settings-browser-tools.png +0 -0
  50. package/docs/assets/showcase/agents-ui.jpg +0 -0
  51. package/docs/assets/showcase/bambu-cli.png +0 -0
  52. package/docs/assets/showcase/codexmonitor.png +0 -0
  53. package/docs/assets/showcase/gohome-grafana.png +0 -0
  54. package/docs/assets/showcase/ios-testflight.jpg +0 -0
  55. package/docs/assets/showcase/oura-health.png +0 -0
  56. package/docs/assets/showcase/padel-cli.svg +11 -0
  57. package/docs/assets/showcase/padel-screenshot.jpg +0 -0
  58. package/docs/assets/showcase/papla-tts.jpg +0 -0
  59. package/docs/assets/showcase/pr-review-telegram.jpg +0 -0
  60. package/docs/assets/showcase/roborock-screenshot.jpg +0 -0
  61. package/docs/assets/showcase/roborock-status.svg +13 -0
  62. package/docs/assets/showcase/roof-camera-sky.jpg +0 -0
  63. package/docs/assets/showcase/snag.png +0 -0
  64. package/docs/assets/showcase/tesco-shop.jpg +0 -0
  65. package/docs/assets/showcase/wienerlinien.png +0 -0
  66. package/docs/assets/showcase/wine-cellar-skill.jpg +0 -0
  67. package/docs/assets/showcase/winix-air-purifier.jpg +0 -0
  68. package/docs/assets/showcase/xuezh-pronunciation.jpeg +0 -0
  69. package/docs/assets/sponsors/blacksmith-light.svg +14 -0
  70. package/docs/assets/sponsors/blacksmith.svg +14 -0
  71. package/docs/assets/sponsors/convex-light.svg +16 -0
  72. package/docs/assets/sponsors/convex.svg +16 -0
  73. package/docs/assets/sponsors/github-light.svg +3 -0
  74. package/docs/assets/sponsors/github.svg +3 -0
  75. package/docs/assets/sponsors/nvidia-dark.svg +9 -0
  76. package/docs/assets/sponsors/nvidia.svg +9 -0
  77. package/docs/assets/sponsors/openai-light.svg +3 -0
  78. package/docs/assets/sponsors/openai.svg +3 -0
  79. package/docs/assets/sponsors/vercel-light.svg +5 -0
  80. package/docs/assets/sponsors/vercel.svg +5 -0
  81. package/docs/auth-credential-semantics.md +124 -0
  82. package/docs/automation/auth-monitoring.md +11 -0
  83. package/docs/automation/clawflow.md +12 -0
  84. package/docs/automation/cron-jobs.md +500 -0
  85. package/docs/automation/cron-vs-heartbeat.md +11 -0
  86. package/docs/automation/gmail-pubsub.md +11 -0
  87. package/docs/automation/hooks.md +365 -0
  88. package/docs/automation/index.md +135 -0
  89. package/docs/automation/poll.md +12 -0
  90. package/docs/automation/standing-orders.md +250 -0
  91. package/docs/automation/taskflow.md +155 -0
  92. package/docs/automation/tasks.md +374 -0
  93. package/docs/automation/troubleshooting.md +12 -0
  94. package/docs/automation/webhook.md +12 -0
  95. package/docs/brave-search.md +11 -0
  96. package/docs/channels/access-groups.md +201 -0
  97. package/docs/channels/ambient-room-events.md +214 -0
  98. package/docs/channels/bot-loop-protection.md +131 -0
  99. package/docs/channels/broadcast-groups.md +472 -0
  100. package/docs/channels/channel-routing.md +162 -0
  101. package/docs/channels/clickclack.md +138 -0
  102. package/docs/channels/discord.md +1762 -0
  103. package/docs/channels/feishu.md +502 -0
  104. package/docs/channels/googlechat.md +284 -0
  105. package/docs/channels/group-messages.md +95 -0
  106. package/docs/channels/groups.md +519 -0
  107. package/docs/channels/imessage-from-bluebubbles.md +259 -0
  108. package/docs/channels/imessage.md +813 -0
  109. package/docs/channels/index.md +64 -0
  110. package/docs/channels/irc.md +253 -0
  111. package/docs/channels/line.md +243 -0
  112. package/docs/channels/location.md +71 -0
  113. package/docs/channels/matrix-migration.md +370 -0
  114. package/docs/channels/matrix-presentation.md +77 -0
  115. package/docs/channels/matrix-push-rules.md +150 -0
  116. package/docs/channels/matrix.md +921 -0
  117. package/docs/channels/mattermost.md +542 -0
  118. package/docs/channels/msteams.md +1042 -0
  119. package/docs/channels/nextcloud-talk.md +176 -0
  120. package/docs/channels/nostr.md +253 -0
  121. package/docs/channels/pairing.md +214 -0
  122. package/docs/channels/qqbot.md +309 -0
  123. package/docs/channels/signal.md +400 -0
  124. package/docs/channels/slack.md +1564 -0
  125. package/docs/channels/synology-chat.md +187 -0
  126. package/docs/channels/telegram.md +1107 -0
  127. package/docs/channels/tlon.md +296 -0
  128. package/docs/channels/troubleshooting.md +161 -0
  129. package/docs/channels/twitch.md +431 -0
  130. package/docs/channels/wechat.md +171 -0
  131. package/docs/channels/whatsapp.md +739 -0
  132. package/docs/channels/yuanbao.md +416 -0
  133. package/docs/channels/zalo.md +253 -0
  134. package/docs/channels/zalouser.md +199 -0
  135. package/docs/ci.md +612 -0
  136. package/docs/clawhub/publishing.md +96 -0
  137. package/docs/cli/acp.md +370 -0
  138. package/docs/cli/agent.md +103 -0
  139. package/docs/cli/agents.md +232 -0
  140. package/docs/cli/approvals.md +190 -0
  141. package/docs/cli/backup.md +97 -0
  142. package/docs/cli/browser.md +307 -0
  143. package/docs/cli/channels.md +154 -0
  144. package/docs/cli/clawbot.md +25 -0
  145. package/docs/cli/commitments.md +90 -0
  146. package/docs/cli/completion.md +39 -0
  147. package/docs/cli/config.md +504 -0
  148. package/docs/cli/configure.md +77 -0
  149. package/docs/cli/crestodian.md +332 -0
  150. package/docs/cli/cron.md +281 -0
  151. package/docs/cli/daemon.md +67 -0
  152. package/docs/cli/dashboard.md +33 -0
  153. package/docs/cli/devices.md +204 -0
  154. package/docs/cli/directory.md +68 -0
  155. package/docs/cli/dns.md +53 -0
  156. package/docs/cli/docs.md +73 -0
  157. package/docs/cli/doctor.md +237 -0
  158. package/docs/cli/flows.md +52 -0
  159. package/docs/cli/gateway.md +567 -0
  160. package/docs/cli/health.md +43 -0
  161. package/docs/cli/hooks.md +345 -0
  162. package/docs/cli/index.md +396 -0
  163. package/docs/cli/infer.md +364 -0
  164. package/docs/cli/logs.md +65 -0
  165. package/docs/cli/mcp.md +529 -0
  166. package/docs/cli/memory.md +183 -0
  167. package/docs/cli/message.md +317 -0
  168. package/docs/cli/migrate.md +290 -0
  169. package/docs/cli/models.md +224 -0
  170. package/docs/cli/node.md +177 -0
  171. package/docs/cli/nodes.md +76 -0
  172. package/docs/cli/onboard.md +245 -0
  173. package/docs/cli/pairing.md +77 -0
  174. package/docs/cli/path.md +502 -0
  175. package/docs/cli/plugins.md +454 -0
  176. package/docs/cli/policy.md +418 -0
  177. package/docs/cli/proxy.md +89 -0
  178. package/docs/cli/qr.md +56 -0
  179. package/docs/cli/reset.md +39 -0
  180. package/docs/cli/sandbox.md +208 -0
  181. package/docs/cli/secrets.md +202 -0
  182. package/docs/cli/security.md +124 -0
  183. package/docs/cli/sessions.md +164 -0
  184. package/docs/cli/setup.md +59 -0
  185. package/docs/cli/skills.md +102 -0
  186. package/docs/cli/status.md +45 -0
  187. package/docs/cli/system.md +89 -0
  188. package/docs/cli/tasks.md +111 -0
  189. package/docs/cli/tui.md +89 -0
  190. package/docs/cli/uninstall.md +44 -0
  191. package/docs/cli/update.md +242 -0
  192. package/docs/cli/voicecall.md +204 -0
  193. package/docs/cli/webhooks.md +117 -0
  194. package/docs/cli/wiki.md +256 -0
  195. package/docs/concepts/active-memory.md +856 -0
  196. package/docs/concepts/agent-loop.md +185 -0
  197. package/docs/concepts/agent-runtimes.md +243 -0
  198. package/docs/concepts/agent-workspace.md +230 -0
  199. package/docs/concepts/agent.md +136 -0
  200. package/docs/concepts/architecture.md +154 -0
  201. package/docs/concepts/channel-docking.md +145 -0
  202. package/docs/concepts/commitments.md +150 -0
  203. package/docs/concepts/compaction.md +203 -0
  204. package/docs/concepts/context-engine.md +306 -0
  205. package/docs/concepts/context.md +199 -0
  206. package/docs/concepts/delegate-architecture.md +319 -0
  207. package/docs/concepts/dreaming.md +261 -0
  208. package/docs/concepts/experimental-features.md +108 -0
  209. package/docs/concepts/features.md +91 -0
  210. package/docs/concepts/mantis-slack-desktop-runbook.md +202 -0
  211. package/docs/concepts/mantis.md +740 -0
  212. package/docs/concepts/markdown-formatting.md +139 -0
  213. package/docs/concepts/memory-builtin.md +146 -0
  214. package/docs/concepts/memory-honcho.md +144 -0
  215. package/docs/concepts/memory-qmd.md +271 -0
  216. package/docs/concepts/memory-search.md +166 -0
  217. package/docs/concepts/memory.md +258 -0
  218. package/docs/concepts/message-lifecycle-refactor.md +1128 -0
  219. package/docs/concepts/messages.md +214 -0
  220. package/docs/concepts/model-failover.md +385 -0
  221. package/docs/concepts/model-providers.md +715 -0
  222. package/docs/concepts/models.md +370 -0
  223. package/docs/concepts/multi-agent.md +619 -0
  224. package/docs/concepts/oauth.md +198 -0
  225. package/docs/concepts/openclaw-sdk.md +323 -0
  226. package/docs/concepts/parallel-specialist-lanes.md +127 -0
  227. package/docs/concepts/personal-agent-benchmark-pack.md +74 -0
  228. package/docs/concepts/presence.md +117 -0
  229. package/docs/concepts/progress-drafts.md +362 -0
  230. package/docs/concepts/qa-e2e-automation.md +820 -0
  231. package/docs/concepts/qa-matrix.md +139 -0
  232. package/docs/concepts/queue-steering.md +90 -0
  233. package/docs/concepts/queue.md +122 -0
  234. package/docs/concepts/retry.md +86 -0
  235. package/docs/concepts/session-pruning.md +104 -0
  236. package/docs/concepts/session-tool.md +190 -0
  237. package/docs/concepts/session.md +164 -0
  238. package/docs/concepts/soul.md +116 -0
  239. package/docs/concepts/streaming.md +251 -0
  240. package/docs/concepts/system-prompt.md +310 -0
  241. package/docs/concepts/timezone.md +47 -0
  242. package/docs/concepts/typebox.md +309 -0
  243. package/docs/concepts/typing-indicators.md +88 -0
  244. package/docs/concepts/usage-tracking.md +66 -0
  245. package/docs/date-time.md +126 -0
  246. package/docs/debug/node-issue.md +90 -0
  247. package/docs/diagnostics/flags.md +138 -0
  248. package/docs/docs.json +1832 -0
  249. package/docs/gateway/authentication.md +239 -0
  250. package/docs/gateway/background-process.md +147 -0
  251. package/docs/gateway/bonjour.md +303 -0
  252. package/docs/gateway/bridge-protocol.md +94 -0
  253. package/docs/gateway/cli-backends.md +420 -0
  254. package/docs/gateway/config-agents.md +1514 -0
  255. package/docs/gateway/config-channels.md +945 -0
  256. package/docs/gateway/config-tools.md +769 -0
  257. package/docs/gateway/configuration-examples.md +705 -0
  258. package/docs/gateway/configuration-reference.md +1393 -0
  259. package/docs/gateway/configuration.md +737 -0
  260. package/docs/gateway/diagnostics.md +213 -0
  261. package/docs/gateway/discovery.md +154 -0
  262. package/docs/gateway/doctor.md +574 -0
  263. package/docs/gateway/gateway-lock.md +37 -0
  264. package/docs/gateway/health.md +73 -0
  265. package/docs/gateway/heartbeat.md +493 -0
  266. package/docs/gateway/index.md +383 -0
  267. package/docs/gateway/local-model-services.md +205 -0
  268. package/docs/gateway/local-models.md +355 -0
  269. package/docs/gateway/logging.md +149 -0
  270. package/docs/gateway/multiple-gateways.md +178 -0
  271. package/docs/gateway/network-model.md +15 -0
  272. package/docs/gateway/openai-http-api.md +350 -0
  273. package/docs/gateway/openresponses-http-api.md +347 -0
  274. package/docs/gateway/openshell.md +316 -0
  275. package/docs/gateway/opentelemetry.md +404 -0
  276. package/docs/gateway/operator-scopes.md +111 -0
  277. package/docs/gateway/pairing.md +207 -0
  278. package/docs/gateway/prometheus.md +230 -0
  279. package/docs/gateway/protocol.md +803 -0
  280. package/docs/gateway/remote-gateway-readme.md +169 -0
  281. package/docs/gateway/remote.md +280 -0
  282. package/docs/gateway/sandbox-vs-tool-policy-vs-elevated.md +146 -0
  283. package/docs/gateway/sandboxing.md +545 -0
  284. package/docs/gateway/secrets-plan-contract.md +114 -0
  285. package/docs/gateway/secrets.md +609 -0
  286. package/docs/gateway/security/audit-checks.md +127 -0
  287. package/docs/gateway/security/index.md +1326 -0
  288. package/docs/gateway/security/secure-file-operations.md +76 -0
  289. package/docs/gateway/tailscale.md +156 -0
  290. package/docs/gateway/tools-invoke-http-api.md +169 -0
  291. package/docs/gateway/troubleshooting.md +772 -0
  292. package/docs/gateway/trusted-proxy-auth.md +451 -0
  293. package/docs/help/debugging.md +344 -0
  294. package/docs/help/environment.md +214 -0
  295. package/docs/help/faq-first-run.md +867 -0
  296. package/docs/help/faq-models.md +553 -0
  297. package/docs/help/faq.md +1975 -0
  298. package/docs/help/gpt55-codex-agentic-parity-maintainers.md +196 -0
  299. package/docs/help/gpt55-codex-agentic-parity.md +230 -0
  300. package/docs/help/index.md +39 -0
  301. package/docs/help/scripts.md +56 -0
  302. package/docs/help/testing-live.md +580 -0
  303. package/docs/help/testing-updates-plugins.md +291 -0
  304. package/docs/help/testing.md +928 -0
  305. package/docs/help/troubleshooting.md +424 -0
  306. package/docs/images/configure-model-picker-unsearchable.png +0 -0
  307. package/docs/images/feishu-get-group-id.png +0 -0
  308. package/docs/images/groups-flow.svg +52 -0
  309. package/docs/images/mobile-ui-screenshot.png +0 -0
  310. package/docs/index.md +196 -0
  311. package/docs/install/ansible.md +233 -0
  312. package/docs/install/azure.md +315 -0
  313. package/docs/install/bun.md +59 -0
  314. package/docs/install/clawdock.md +112 -0
  315. package/docs/install/development-channels.md +135 -0
  316. package/docs/install/digitalocean.md +174 -0
  317. package/docs/install/docker-vm-runtime.md +154 -0
  318. package/docs/install/docker.md +562 -0
  319. package/docs/install/exe-dev.md +201 -0
  320. package/docs/install/fly.md +524 -0
  321. package/docs/install/gcp.md +418 -0
  322. package/docs/install/hetzner.md +285 -0
  323. package/docs/install/hostinger.md +98 -0
  324. package/docs/install/index.md +221 -0
  325. package/docs/install/installer.md +455 -0
  326. package/docs/install/kubernetes.md +196 -0
  327. package/docs/install/macos-vm.md +281 -0
  328. package/docs/install/migrating-claude.md +165 -0
  329. package/docs/install/migrating-hermes.md +177 -0
  330. package/docs/install/migrating.md +137 -0
  331. package/docs/install/nix.md +112 -0
  332. package/docs/install/node.md +142 -0
  333. package/docs/install/northflank.mdx +44 -0
  334. package/docs/install/oracle.md +218 -0
  335. package/docs/install/podman.md +210 -0
  336. package/docs/install/railway.mdx +92 -0
  337. package/docs/install/raspberry-pi.md +234 -0
  338. package/docs/install/render.mdx +167 -0
  339. package/docs/install/uninstall.md +131 -0
  340. package/docs/install/updating.md +280 -0
  341. package/docs/logging.md +318 -0
  342. package/docs/nav-tabs-underline.js +100 -0
  343. package/docs/network.md +72 -0
  344. package/docs/nodes/audio.md +215 -0
  345. package/docs/nodes/camera.md +166 -0
  346. package/docs/nodes/images.md +77 -0
  347. package/docs/nodes/index.md +439 -0
  348. package/docs/nodes/location-command.md +102 -0
  349. package/docs/nodes/media-understanding.md +469 -0
  350. package/docs/nodes/talk.md +154 -0
  351. package/docs/nodes/troubleshooting.md +123 -0
  352. package/docs/nodes/voicewake.md +93 -0
  353. package/docs/perplexity.md +11 -0
  354. package/docs/pi-dev.md +82 -0
  355. package/docs/pi.md +573 -0
  356. package/docs/plan/codex-context-engine-harness.md +624 -0
  357. package/docs/plan/ui-channels.md +284 -0
  358. package/docs/platforms/android.md +285 -0
  359. package/docs/platforms/digitalocean.md +12 -0
  360. package/docs/platforms/index.md +60 -0
  361. package/docs/platforms/ios.md +283 -0
  362. package/docs/platforms/linux.md +141 -0
  363. package/docs/platforms/mac/bundled-gateway.md +79 -0
  364. package/docs/platforms/mac/canvas.md +128 -0
  365. package/docs/platforms/mac/child-process.md +72 -0
  366. package/docs/platforms/mac/dev-setup.md +112 -0
  367. package/docs/platforms/mac/health.md +39 -0
  368. package/docs/platforms/mac/icon.md +36 -0
  369. package/docs/platforms/mac/logging.md +62 -0
  370. package/docs/platforms/mac/menu-bar.md +93 -0
  371. package/docs/platforms/mac/peekaboo.md +92 -0
  372. package/docs/platforms/mac/permissions.md +53 -0
  373. package/docs/platforms/mac/remote.md +123 -0
  374. package/docs/platforms/mac/signing.md +52 -0
  375. package/docs/platforms/mac/skills.md +43 -0
  376. package/docs/platforms/mac/voice-overlay.md +66 -0
  377. package/docs/platforms/mac/voicewake.md +73 -0
  378. package/docs/platforms/mac/webchat.md +54 -0
  379. package/docs/platforms/mac/xpc.md +66 -0
  380. package/docs/platforms/macos.md +226 -0
  381. package/docs/platforms/oracle.md +12 -0
  382. package/docs/platforms/raspberry-pi.md +13 -0
  383. package/docs/platforms/windows.md +286 -0
  384. package/docs/plugins/adding-capabilities.md +133 -0
  385. package/docs/plugins/admin-http-rpc.md +216 -0
  386. package/docs/plugins/agent-tools.md +13 -0
  387. package/docs/plugins/architecture-internals.md +1195 -0
  388. package/docs/plugins/architecture.md +481 -0
  389. package/docs/plugins/building-extensions.md +13 -0
  390. package/docs/plugins/building-plugins.md +330 -0
  391. package/docs/plugins/bundles.md +310 -0
  392. package/docs/plugins/cli-backend-plugins.md +310 -0
  393. package/docs/plugins/codex-computer-use.md +293 -0
  394. package/docs/plugins/codex-harness-reference.md +409 -0
  395. package/docs/plugins/codex-harness-runtime.md +247 -0
  396. package/docs/plugins/codex-harness.md +746 -0
  397. package/docs/plugins/codex-native-plugins.md +276 -0
  398. package/docs/plugins/community.md +77 -0
  399. package/docs/plugins/compatibility.md +164 -0
  400. package/docs/plugins/dependency-resolution.md +143 -0
  401. package/docs/plugins/google-meet.md +1737 -0
  402. package/docs/plugins/hooks.md +459 -0
  403. package/docs/plugins/install-overrides.md +80 -0
  404. package/docs/plugins/manage-plugins.md +210 -0
  405. package/docs/plugins/manifest.md +1359 -0
  406. package/docs/plugins/memory-lancedb.md +385 -0
  407. package/docs/plugins/memory-wiki.md +529 -0
  408. package/docs/plugins/message-presentation.md +473 -0
  409. package/docs/plugins/oc-path.md +166 -0
  410. package/docs/plugins/plugin-inventory.md +182 -0
  411. package/docs/plugins/reference/acpx.md +23 -0
  412. package/docs/plugins/reference/admin-http-rpc.md +23 -0
  413. package/docs/plugins/reference/alibaba.md +23 -0
  414. package/docs/plugins/reference/amazon-bedrock-mantle.md +23 -0
  415. package/docs/plugins/reference/amazon-bedrock.md +23 -0
  416. package/docs/plugins/reference/anthropic-vertex.md +19 -0
  417. package/docs/plugins/reference/anthropic.md +23 -0
  418. package/docs/plugins/reference/arcee.md +23 -0
  419. package/docs/plugins/reference/azure-speech.md +23 -0
  420. package/docs/plugins/reference/bonjour.md +19 -0
  421. package/docs/plugins/reference/brave.md +23 -0
  422. package/docs/plugins/reference/browser.md +23 -0
  423. package/docs/plugins/reference/byteplus.md +19 -0
  424. package/docs/plugins/reference/canvas.md +19 -0
  425. package/docs/plugins/reference/cerebras.md +23 -0
  426. package/docs/plugins/reference/chutes.md +23 -0
  427. package/docs/plugins/reference/clickclack.md +23 -0
  428. package/docs/plugins/reference/cloudflare-ai-gateway.md +23 -0
  429. package/docs/plugins/reference/codex.md +23 -0
  430. package/docs/plugins/reference/comfy.md +23 -0
  431. package/docs/plugins/reference/copilot-proxy.md +19 -0
  432. package/docs/plugins/reference/deepgram.md +23 -0
  433. package/docs/plugins/reference/deepinfra.md +23 -0
  434. package/docs/plugins/reference/deepseek.md +23 -0
  435. package/docs/plugins/reference/diagnostics-otel.md +19 -0
  436. package/docs/plugins/reference/diagnostics-prometheus.md +19 -0
  437. package/docs/plugins/reference/diffs.md +19 -0
  438. package/docs/plugins/reference/discord.md +23 -0
  439. package/docs/plugins/reference/document-extract.md +23 -0
  440. package/docs/plugins/reference/duckduckgo.md +23 -0
  441. package/docs/plugins/reference/elevenlabs.md +23 -0
  442. package/docs/plugins/reference/exa.md +23 -0
  443. package/docs/plugins/reference/fal.md +23 -0
  444. package/docs/plugins/reference/feishu.md +23 -0
  445. package/docs/plugins/reference/file-transfer.md +19 -0
  446. package/docs/plugins/reference/firecrawl.md +23 -0
  447. package/docs/plugins/reference/fireworks.md +23 -0
  448. package/docs/plugins/reference/github-copilot.md +23 -0
  449. package/docs/plugins/reference/google-meet.md +23 -0
  450. package/docs/plugins/reference/google.md +23 -0
  451. package/docs/plugins/reference/googlechat.md +23 -0
  452. package/docs/plugins/reference/gradium.md +23 -0
  453. package/docs/plugins/reference/groq.md +23 -0
  454. package/docs/plugins/reference/huggingface.md +23 -0
  455. package/docs/plugins/reference/imessage.md +23 -0
  456. package/docs/plugins/reference/inworld.md +23 -0
  457. package/docs/plugins/reference/irc.md +23 -0
  458. package/docs/plugins/reference/kilocode.md +23 -0
  459. package/docs/plugins/reference/kimi.md +23 -0
  460. package/docs/plugins/reference/line.md +23 -0
  461. package/docs/plugins/reference/litellm.md +23 -0
  462. package/docs/plugins/reference/llm-task.md +19 -0
  463. package/docs/plugins/reference/lmstudio.md +23 -0
  464. package/docs/plugins/reference/lobster.md +19 -0
  465. package/docs/plugins/reference/matrix.md +23 -0
  466. package/docs/plugins/reference/mattermost.md +23 -0
  467. package/docs/plugins/reference/memory-core.md +19 -0
  468. package/docs/plugins/reference/memory-lancedb.md +23 -0
  469. package/docs/plugins/reference/memory-wiki.md +23 -0
  470. package/docs/plugins/reference/microsoft-foundry.md +19 -0
  471. package/docs/plugins/reference/microsoft.md +19 -0
  472. package/docs/plugins/reference/migrate-claude.md +19 -0
  473. package/docs/plugins/reference/migrate-hermes.md +19 -0
  474. package/docs/plugins/reference/minimax.md +23 -0
  475. package/docs/plugins/reference/mistral.md +23 -0
  476. package/docs/plugins/reference/moonshot.md +23 -0
  477. package/docs/plugins/reference/msteams.md +23 -0
  478. package/docs/plugins/reference/nextcloud-talk.md +23 -0
  479. package/docs/plugins/reference/nostr.md +23 -0
  480. package/docs/plugins/reference/nvidia.md +23 -0
  481. package/docs/plugins/reference/oc-path.md +23 -0
  482. package/docs/plugins/reference/ollama.md +23 -0
  483. package/docs/plugins/reference/open-prose.md +19 -0
  484. package/docs/plugins/reference/openai.md +23 -0
  485. package/docs/plugins/reference/opencode-go.md +23 -0
  486. package/docs/plugins/reference/opencode.md +23 -0
  487. package/docs/plugins/reference/openrouter.md +23 -0
  488. package/docs/plugins/reference/openshell.md +19 -0
  489. package/docs/plugins/reference/perplexity.md +23 -0
  490. package/docs/plugins/reference/policy.md +23 -0
  491. package/docs/plugins/reference/qa-channel.md +23 -0
  492. package/docs/plugins/reference/qa-lab.md +19 -0
  493. package/docs/plugins/reference/qa-matrix.md +19 -0
  494. package/docs/plugins/reference/qianfan.md +23 -0
  495. package/docs/plugins/reference/qqbot.md +23 -0
  496. package/docs/plugins/reference/qwen.md +23 -0
  497. package/docs/plugins/reference/runway.md +23 -0
  498. package/docs/plugins/reference/searxng.md +19 -0
  499. package/docs/plugins/reference/senseaudio.md +23 -0
  500. package/docs/plugins/reference/sglang.md +23 -0
  501. package/docs/plugins/reference/signal.md +23 -0
  502. package/docs/plugins/reference/skill-workshop.md +23 -0
  503. package/docs/plugins/reference/slack.md +23 -0
  504. package/docs/plugins/reference/stepfun.md +23 -0
  505. package/docs/plugins/reference/synology-chat.md +23 -0
  506. package/docs/plugins/reference/synthetic.md +23 -0
  507. package/docs/plugins/reference/tavily.md +23 -0
  508. package/docs/plugins/reference/telegram.md +23 -0
  509. package/docs/plugins/reference/tencent.md +23 -0
  510. package/docs/plugins/reference/tlon.md +23 -0
  511. package/docs/plugins/reference/together.md +23 -0
  512. package/docs/plugins/reference/tokenjuice.md +23 -0
  513. package/docs/plugins/reference/tts-local-cli.md +19 -0
  514. package/docs/plugins/reference/twitch.md +23 -0
  515. package/docs/plugins/reference/venice.md +23 -0
  516. package/docs/plugins/reference/vercel-ai-gateway.md +23 -0
  517. package/docs/plugins/reference/vllm.md +23 -0
  518. package/docs/plugins/reference/voice-call.md +23 -0
  519. package/docs/plugins/reference/volcengine.md +23 -0
  520. package/docs/plugins/reference/voyage.md +19 -0
  521. package/docs/plugins/reference/vydra.md +23 -0
  522. package/docs/plugins/reference/web-readability.md +19 -0
  523. package/docs/plugins/reference/webhooks.md +23 -0
  524. package/docs/plugins/reference/whatsapp.md +23 -0
  525. package/docs/plugins/reference/xai.md +23 -0
  526. package/docs/plugins/reference/xiaomi.md +23 -0
  527. package/docs/plugins/reference/zai.md +23 -0
  528. package/docs/plugins/reference/zalo.md +23 -0
  529. package/docs/plugins/reference/zalouser.md +24 -0
  530. package/docs/plugins/reference.md +138 -0
  531. package/docs/plugins/sdk-agent-harness.md +339 -0
  532. package/docs/plugins/sdk-channel-ingress.md +137 -0
  533. package/docs/plugins/sdk-channel-message.md +458 -0
  534. package/docs/plugins/sdk-channel-plugins.md +762 -0
  535. package/docs/plugins/sdk-channel-turn.md +580 -0
  536. package/docs/plugins/sdk-entrypoints.md +333 -0
  537. package/docs/plugins/sdk-migration.md +949 -0
  538. package/docs/plugins/sdk-overview.md +501 -0
  539. package/docs/plugins/sdk-provider-plugins.md +807 -0
  540. package/docs/plugins/sdk-runtime.md +676 -0
  541. package/docs/plugins/sdk-setup.md +550 -0
  542. package/docs/plugins/sdk-subpaths.md +396 -0
  543. package/docs/plugins/sdk-testing.md +401 -0
  544. package/docs/plugins/skill-workshop.md +713 -0
  545. package/docs/plugins/tool-plugins.md +411 -0
  546. package/docs/plugins/voice-call.md +943 -0
  547. package/docs/plugins/webhooks.md +192 -0
  548. package/docs/plugins/zalouser.md +86 -0
  549. package/docs/prose.md +137 -0
  550. package/docs/providers/alibaba.md +158 -0
  551. package/docs/providers/anthropic.md +344 -0
  552. package/docs/providers/arcee.md +144 -0
  553. package/docs/providers/azure-speech.md +119 -0
  554. package/docs/providers/bedrock-mantle.md +211 -0
  555. package/docs/providers/bedrock.md +414 -0
  556. package/docs/providers/cerebras.md +130 -0
  557. package/docs/providers/chutes.md +153 -0
  558. package/docs/providers/claude-max-api-proxy.md +188 -0
  559. package/docs/providers/cloudflare-ai-gateway.md +119 -0
  560. package/docs/providers/comfy.md +362 -0
  561. package/docs/providers/deepgram.md +184 -0
  562. package/docs/providers/deepinfra.md +87 -0
  563. package/docs/providers/deepseek.md +146 -0
  564. package/docs/providers/ds4.md +309 -0
  565. package/docs/providers/elevenlabs.md +130 -0
  566. package/docs/providers/fal.md +204 -0
  567. package/docs/providers/fireworks.md +144 -0
  568. package/docs/providers/github-copilot.md +225 -0
  569. package/docs/providers/glm.md +137 -0
  570. package/docs/providers/google.md +472 -0
  571. package/docs/providers/gradium.md +123 -0
  572. package/docs/providers/groq.md +180 -0
  573. package/docs/providers/huggingface.md +235 -0
  574. package/docs/providers/index.md +102 -0
  575. package/docs/providers/inferrs.md +272 -0
  576. package/docs/providers/inworld.md +120 -0
  577. package/docs/providers/kilocode.md +135 -0
  578. package/docs/providers/litellm.md +234 -0
  579. package/docs/providers/lmstudio.md +224 -0
  580. package/docs/providers/minimax.md +505 -0
  581. package/docs/providers/mistral.md +235 -0
  582. package/docs/providers/models.md +65 -0
  583. package/docs/providers/moonshot.md +413 -0
  584. package/docs/providers/nvidia.md +140 -0
  585. package/docs/providers/ollama.md +1180 -0
  586. package/docs/providers/openai.md +1057 -0
  587. package/docs/providers/opencode-go.md +123 -0
  588. package/docs/providers/opencode.md +149 -0
  589. package/docs/providers/openrouter.md +349 -0
  590. package/docs/providers/perplexity-provider.md +123 -0
  591. package/docs/providers/qianfan.md +132 -0
  592. package/docs/providers/qwen.md +332 -0
  593. package/docs/providers/runway.md +103 -0
  594. package/docs/providers/senseaudio.md +68 -0
  595. package/docs/providers/sglang.md +161 -0
  596. package/docs/providers/stepfun.md +229 -0
  597. package/docs/providers/synthetic.md +154 -0
  598. package/docs/providers/tencent.md +130 -0
  599. package/docs/providers/together.md +141 -0
  600. package/docs/providers/venice.md +315 -0
  601. package/docs/providers/vercel-ai-gateway.md +128 -0
  602. package/docs/providers/vllm.md +383 -0
  603. package/docs/providers/volcengine.md +199 -0
  604. package/docs/providers/vydra.md +180 -0
  605. package/docs/providers/xai.md +560 -0
  606. package/docs/providers/xiaomi.md +188 -0
  607. package/docs/providers/zai.md +203 -0
  608. package/docs/refactor/access.md +9 -0
  609. package/docs/refactor/acp.md +298 -0
  610. package/docs/refactor/canvas.md +131 -0
  611. package/docs/refactor/ingress-core.md +341 -0
  612. package/docs/reference/AGENTS.default.md +129 -0
  613. package/docs/reference/RELEASING.md +767 -0
  614. package/docs/reference/api-usage-costs.md +202 -0
  615. package/docs/reference/application-modernization-plan.md +208 -0
  616. package/docs/reference/code-mode.md +757 -0
  617. package/docs/reference/credits.md +33 -0
  618. package/docs/reference/device-models.md +50 -0
  619. package/docs/reference/full-release-validation.md +202 -0
  620. package/docs/reference/memory-config.md +630 -0
  621. package/docs/reference/openclaw-sdk-api-design.md +390 -0
  622. package/docs/reference/prompt-caching.md +358 -0
  623. package/docs/reference/rich-output-protocol.md +79 -0
  624. package/docs/reference/rpc.md +43 -0
  625. package/docs/reference/secretref-credential-surface.md +159 -0
  626. package/docs/reference/secretref-user-supplied-credentials-matrix.json +663 -0
  627. package/docs/reference/session-management-compaction.md +461 -0
  628. package/docs/reference/templates/AGENTS.dev.md +89 -0
  629. package/docs/reference/templates/AGENTS.md +225 -0
  630. package/docs/reference/templates/BOOT.md +16 -0
  631. package/docs/reference/templates/BOOTSTRAP.md +66 -0
  632. package/docs/reference/templates/HEARTBEAT.md +16 -0
  633. package/docs/reference/templates/IDENTITY.dev.md +52 -0
  634. package/docs/reference/templates/IDENTITY.md +34 -0
  635. package/docs/reference/templates/SOUL.dev.md +82 -0
  636. package/docs/reference/templates/SOUL.md +49 -0
  637. package/docs/reference/templates/TOOLS.dev.md +29 -0
  638. package/docs/reference/templates/TOOLS.md +51 -0
  639. package/docs/reference/templates/USER.dev.md +23 -0
  640. package/docs/reference/templates/USER.md +28 -0
  641. package/docs/reference/test.md +239 -0
  642. package/docs/reference/token-use.md +233 -0
  643. package/docs/reference/transcript-hygiene.md +214 -0
  644. package/docs/reference/wizard.md +252 -0
  645. package/docs/security/CONTRIBUTING-THREAT-MODEL.md +101 -0
  646. package/docs/security/THREAT-MODEL-ATLAS.md +611 -0
  647. package/docs/security/formal-verification.md +170 -0
  648. package/docs/security/incident-response.md +59 -0
  649. package/docs/security/network-proxy.md +268 -0
  650. package/docs/snippets/plugin-publish/minimal-openclaw.plugin.json +12 -0
  651. package/docs/snippets/plugin-publish/minimal-package.json +16 -0
  652. package/docs/start/bootstrapping.md +49 -0
  653. package/docs/start/docs-directory.md +69 -0
  654. package/docs/start/getting-started.md +152 -0
  655. package/docs/start/hubs.md +201 -0
  656. package/docs/start/lore.md +223 -0
  657. package/docs/start/onboarding-overview.md +72 -0
  658. package/docs/start/onboarding.md +95 -0
  659. package/docs/start/openclaw.md +244 -0
  660. package/docs/start/quickstart.md +25 -0
  661. package/docs/start/setup.md +178 -0
  662. package/docs/start/showcase.md +383 -0
  663. package/docs/start/wizard-cli-automation.md +232 -0
  664. package/docs/start/wizard-cli-reference.md +331 -0
  665. package/docs/start/wizard.md +141 -0
  666. package/docs/style.css +184 -0
  667. package/docs/superpowers/specs/2026-04-22-tweakcn-custom-theme-import-design.md +316 -0
  668. package/docs/tools/acp-agents-setup.md +352 -0
  669. package/docs/tools/acp-agents.md +847 -0
  670. package/docs/tools/agent-send.md +112 -0
  671. package/docs/tools/apply-patch.md +64 -0
  672. package/docs/tools/brave-search.md +139 -0
  673. package/docs/tools/browser-control.md +391 -0
  674. package/docs/tools/browser-linux-troubleshooting.md +173 -0
  675. package/docs/tools/browser-login.md +77 -0
  676. package/docs/tools/browser-wsl2-windows-remote-cdp-troubleshooting.md +219 -0
  677. package/docs/tools/browser.md +769 -0
  678. package/docs/tools/btw.md +159 -0
  679. package/docs/tools/capability-cookbook.md +12 -0
  680. package/docs/tools/clawhub.md +5 -0
  681. package/docs/tools/code-execution.md +173 -0
  682. package/docs/tools/creating-skills.md +120 -0
  683. package/docs/tools/diffs.md +506 -0
  684. package/docs/tools/duckduckgo-search.md +109 -0
  685. package/docs/tools/elevated.md +128 -0
  686. package/docs/tools/exa-search.md +152 -0
  687. package/docs/tools/exec-approvals-advanced.md +360 -0
  688. package/docs/tools/exec-approvals.md +474 -0
  689. package/docs/tools/exec.md +282 -0
  690. package/docs/tools/firecrawl.md +155 -0
  691. package/docs/tools/gemini-search.md +114 -0
  692. package/docs/tools/grok-search.md +113 -0
  693. package/docs/tools/image-generation.md +433 -0
  694. package/docs/tools/index.md +178 -0
  695. package/docs/tools/kimi-search.md +105 -0
  696. package/docs/tools/llm-task.md +137 -0
  697. package/docs/tools/lobster.md +365 -0
  698. package/docs/tools/loop-detection.md +154 -0
  699. package/docs/tools/media-overview.md +157 -0
  700. package/docs/tools/minimax-search.md +102 -0
  701. package/docs/tools/multi-agent-sandbox-tools.md +409 -0
  702. package/docs/tools/music-generation.md +371 -0
  703. package/docs/tools/ollama-search.md +153 -0
  704. package/docs/tools/pdf.md +195 -0
  705. package/docs/tools/perplexity-search.md +220 -0
  706. package/docs/tools/plugin.md +327 -0
  707. package/docs/tools/reactions.md +100 -0
  708. package/docs/tools/searxng-search.md +141 -0
  709. package/docs/tools/skills-config.md +195 -0
  710. package/docs/tools/skills.md +535 -0
  711. package/docs/tools/slash-commands.md +488 -0
  712. package/docs/tools/steer.md +84 -0
  713. package/docs/tools/subagents.md +650 -0
  714. package/docs/tools/tavily.md +162 -0
  715. package/docs/tools/thinking.md +140 -0
  716. package/docs/tools/tokenjuice.md +81 -0
  717. package/docs/tools/tool-search.md +269 -0
  718. package/docs/tools/trajectory.md +229 -0
  719. package/docs/tools/tts.md +1004 -0
  720. package/docs/tools/video-generation.md +552 -0
  721. package/docs/tools/web-fetch.md +195 -0
  722. package/docs/tools/web.md +459 -0
  723. package/docs/tts.md +11 -0
  724. package/docs/vps.md +139 -0
  725. package/docs/web/control-ui.md +503 -0
  726. package/docs/web/dashboard.md +107 -0
  727. package/docs/web/index.md +133 -0
  728. package/docs/web/tui.md +246 -0
  729. package/docs/web/webchat.md +99 -0
  730. package/docs/whatsapp-openclaw-ai-zh.jpg +0 -0
  731. package/docs/whatsapp-openclaw.jpg +0 -0
  732. package/nodmix.mjs +487 -0
  733. package/package.json +1852 -0
  734. package/patches/.gitkeep +0 -0
  735. package/patches/@agentclientprotocol__claude-agent-acp@0.36.1.patch +41 -0
  736. package/pnpm-workspace.yaml +63 -0
  737. package/scripts/crabbox-wrapper.mjs +353 -0
  738. package/scripts/lib/official-external-channel-catalog.json +559 -0
  739. package/scripts/lib/official-external-plugin-catalog.json +192 -0
  740. package/scripts/lib/official-external-provider-catalog.json +117 -0
  741. package/scripts/lib/package-dist-imports.mjs +171 -0
  742. package/scripts/npm-runner.mjs +91 -0
  743. package/scripts/postinstall-bundled-plugins.mjs +978 -0
  744. package/scripts/preinstall-package-manager-warning.mjs +64 -0
  745. package/scripts/windows-cmd-helpers.mjs +20 -0
  746. package/skills/1password/SKILL.md +70 -0
  747. package/skills/1password/references/cli-examples.md +29 -0
  748. package/skills/1password/references/get-started.md +17 -0
  749. package/skills/apple-notes/SKILL.md +77 -0
  750. package/skills/apple-reminders/SKILL.md +118 -0
  751. package/skills/bear-notes/SKILL.md +107 -0
  752. package/skills/blogwatcher/SKILL.md +69 -0
  753. package/skills/blucli/SKILL.md +47 -0
  754. package/skills/camsnap/SKILL.md +45 -0
  755. package/skills/canvas/SKILL.md +78 -0
  756. package/skills/clawhub/SKILL.md +77 -0
  757. package/skills/coding-agent/SKILL.md +149 -0
  758. package/skills/diagram-maker/SKILL.md +53 -0
  759. package/skills/diagram-maker/references/excalidraw-patterns.md +85 -0
  760. package/skills/diagram-maker/references/svg-template.md +112 -0
  761. package/skills/discord/SKILL.md +136 -0
  762. package/skills/eightctl/SKILL.md +50 -0
  763. package/skills/gemini/SKILL.md +47 -0
  764. package/skills/gh-issues/SKILL.md +213 -0
  765. package/skills/gifgrep/SKILL.md +85 -0
  766. package/skills/github/SKILL.md +84 -0
  767. package/skills/gog/SKILL.md +116 -0
  768. package/skills/goplaces/SKILL.md +52 -0
  769. package/skills/healthcheck/SKILL.md +105 -0
  770. package/skills/himalaya/SKILL.md +80 -0
  771. package/skills/himalaya/references/configuration.md +184 -0
  772. package/skills/himalaya/references/message-composition.md +199 -0
  773. package/skills/imsg/SKILL.md +122 -0
  774. package/skills/mcporter/SKILL.md +61 -0
  775. package/skills/meme-maker/SKILL.md +42 -0
  776. package/skills/meme-maker/references/templates.json +358 -0
  777. package/skills/meme-maker/scripts/meme.mjs +398 -0
  778. package/skills/model-usage/SKILL.md +69 -0
  779. package/skills/model-usage/references/codexbar-cli.md +33 -0
  780. package/skills/model-usage/scripts/model_usage.py +319 -0
  781. package/skills/model-usage/scripts/test_model_usage.py +40 -0
  782. package/skills/nano-pdf/SKILL.md +38 -0
  783. package/skills/node-connect/SKILL.md +142 -0
  784. package/skills/node-inspect-debugger/SKILL.md +85 -0
  785. package/skills/notion/SKILL.md +150 -0
  786. package/skills/obsidian/SKILL.md +119 -0
  787. package/skills/openai-whisper/SKILL.md +38 -0
  788. package/skills/openai-whisper-api/SKILL.md +71 -0
  789. package/skills/openai-whisper-api/scripts/transcribe.sh +154 -0
  790. package/skills/openhue/SKILL.md +112 -0
  791. package/skills/oracle/SKILL.md +126 -0
  792. package/skills/ordercli/SKILL.md +78 -0
  793. package/skills/peekaboo/SKILL.md +190 -0
  794. package/skills/pyproject.toml +10 -0
  795. package/skills/python-debugpy/SKILL.md +73 -0
  796. package/skills/sag/SKILL.md +87 -0
  797. package/skills/session-logs/SKILL.md +151 -0
  798. package/skills/sherpa-onnx-tts/SKILL.md +109 -0
  799. package/skills/sherpa-onnx-tts/bin/sherpa-onnx-tts +178 -0
  800. package/skills/skill-creator/SKILL.md +78 -0
  801. package/skills/skill-creator/license.txt +202 -0
  802. package/skills/skill-creator/scripts/init_skill.py +378 -0
  803. package/skills/skill-creator/scripts/package_skill.py +139 -0
  804. package/skills/skill-creator/scripts/quick_validate.py +169 -0
  805. package/skills/skill-creator/scripts/test_package_skill.py +161 -0
  806. package/skills/skill-creator/scripts/test_quick_validate.py +116 -0
  807. package/skills/slack/SKILL.md +78 -0
  808. package/skills/songsee/SKILL.md +49 -0
  809. package/skills/sonoscli/SKILL.md +65 -0
  810. package/skills/spike/SKILL.md +51 -0
  811. package/skills/spotify-player/SKILL.md +64 -0
  812. package/skills/summarize/SKILL.md +87 -0
  813. package/skills/taskflow/SKILL.md +149 -0
  814. package/skills/taskflow/examples/inbox-triage.lobster +33 -0
  815. package/skills/taskflow/examples/pr-intake.lobster +32 -0
  816. package/skills/taskflow-inbox-triage/SKILL.md +119 -0
  817. package/skills/things-mac/SKILL.md +86 -0
  818. package/skills/tmux/SKILL.md +91 -0
  819. package/skills/tmux/scripts/find-sessions.sh +112 -0
  820. package/skills/tmux/scripts/wait-for-text.sh +83 -0
  821. package/skills/trello/SKILL.md +108 -0
  822. package/skills/video-frames/SKILL.md +46 -0
  823. package/skills/video-frames/scripts/frame.sh +81 -0
  824. package/skills/voice-call/SKILL.md +45 -0
  825. package/skills/wacli/SKILL.md +72 -0
  826. package/skills/weather/SKILL.md +64 -0
  827. package/skills/xurl/SKILL.md +120 -0
@@ -0,0 +1,1326 @@
1
+ ---
2
+ summary: "Security considerations and threat model for running an AI gateway with shell access"
3
+ read_when:
4
+ - Adding features that widen access or automation
5
+ title: "Security"
6
+ ---
7
+
8
+ <Warning>
9
+ **Personal assistant trust model.** This guidance assumes one trusted
10
+ operator boundary per gateway (single-user, personal-assistant model).
11
+ Nodmix is **not** a hostile multi-tenant security boundary for multiple
12
+ adversarial users sharing one agent or gateway. If you need mixed-trust or
13
+ adversarial-user operation, split trust boundaries (separate gateway +
14
+ credentials, ideally separate OS users or hosts).
15
+ </Warning>
16
+
17
+ ## Scope first: personal assistant security model
18
+
19
+ Nodmix security guidance assumes a **personal assistant** deployment: one trusted operator boundary, potentially many agents.
20
+
21
+ - Supported security posture: one user/trust boundary per gateway (prefer one OS user/host/VPS per boundary).
22
+ - Not a supported security boundary: one shared gateway/agent used by mutually untrusted or adversarial users.
23
+ - If adversarial-user isolation is required, split by trust boundary (separate gateway + credentials, and ideally separate OS users/hosts).
24
+ - If multiple untrusted users can message one tool-enabled agent, treat them as sharing the same delegated tool authority for that agent.
25
+
26
+ This page explains hardening **within that model**. It does not claim hostile multi-tenant isolation on one shared gateway.
27
+
28
+ ## Quick check: `nodmix security audit`
29
+
30
+ See also: [Formal Verification (Security Models)](/security/formal-verification)
31
+
32
+ Run this regularly (especially after changing config or exposing network surfaces):
33
+
34
+ ```bash
35
+ nodmix security audit
36
+ nodmix security audit --deep
37
+ nodmix security audit --fix
38
+ nodmix security audit --json
39
+ ```
40
+
41
+ `security audit --fix` stays intentionally narrow: it flips common open group
42
+ policies to allowlists, restores `logging.redactSensitive: "tools"`, tightens
43
+ state/config/include-file permissions, and uses Windows ACL resets instead of
44
+ POSIX `chmod` when running on Windows.
45
+
46
+ It flags common footguns (Gateway auth exposure, browser control exposure, elevated allowlists, filesystem permissions, permissive exec approvals, and open-channel tool exposure).
47
+
48
+ Nodmix is both a product and an experiment: you're wiring frontier-model behavior into real messaging surfaces and real tools. **There is no "perfectly secure" setup.** The goal is to be deliberate about:
49
+
50
+ - who can talk to your bot
51
+ - where the bot is allowed to act
52
+ - what the bot can touch
53
+
54
+ Start with the smallest access that still works, then widen it as you gain confidence.
55
+
56
+ ### Deployment and host trust
57
+
58
+ Nodmix assumes the host and config boundary are trusted:
59
+
60
+ - If someone can modify Gateway host state/config (`~/.nodmix`, including `nodmix.json`), treat them as a trusted operator.
61
+ - Running one Gateway for multiple mutually untrusted/adversarial operators is **not a recommended setup**.
62
+ - For mixed-trust teams, split trust boundaries with separate gateways (or at minimum separate OS users/hosts).
63
+ - Recommended default: one user per machine/host (or VPS), one gateway for that user, and one or more agents in that gateway.
64
+ - Inside one Gateway instance, authenticated operator access is a trusted control-plane role, not a per-user tenant role.
65
+ - Session identifiers (`sessionKey`, session IDs, labels) are routing selectors, not authorization tokens.
66
+ - If several people can message one tool-enabled agent, each of them can steer that same permission set. Per-user session/memory isolation helps privacy, but does not convert a shared agent into per-user host authorization.
67
+
68
+ ### Secure file operations
69
+
70
+ Nodmix uses `@nodmix/fs-safe` for root-bounded file access, atomic writes, archive extraction, temp workspaces, and secret-file helpers. Nodmix defaults fs-safe's optional POSIX Python helper to **off**; set `NODMIX_FS_SAFE_PYTHON_MODE=auto` or `require` only when you want the extra fd-relative mutation hardening and can support a Python runtime.
71
+
72
+ Details: [Secure file operations](/gateway/security/secure-file-operations).
73
+
74
+ ### Shared Slack workspace: real risk
75
+
76
+ If "everyone in Slack can message the bot," the core risk is delegated tool authority:
77
+
78
+ - any allowed sender can induce tool calls (`exec`, browser, network/file tools) within the agent's policy;
79
+ - prompt/content injection from one sender can cause actions that affect shared state, devices, or outputs;
80
+ - if one shared agent has sensitive credentials/files, any allowed sender can potentially drive exfiltration via tool usage.
81
+
82
+ Use separate agents/gateways with minimal tools for team workflows; keep personal-data agents private.
83
+
84
+ ### Company-shared agent: acceptable pattern
85
+
86
+ This is acceptable when everyone using that agent is in the same trust boundary (for example one company team) and the agent is strictly business-scoped.
87
+
88
+ - run it on a dedicated machine/VM/container;
89
+ - use a dedicated OS user + dedicated browser/profile/accounts for that runtime;
90
+ - do not sign that runtime into personal Apple/Google accounts or personal password-manager/browser profiles.
91
+
92
+ If you mix personal and company identities on the same runtime, you collapse the separation and increase personal-data exposure risk.
93
+
94
+ ## Gateway and node trust concept
95
+
96
+ Treat Gateway and node as one operator trust domain, with different roles:
97
+
98
+ - **Gateway** is the control plane and policy surface (`gateway.auth`, tool policy, routing).
99
+ - **Node** is remote execution surface paired to that Gateway (commands, device actions, host-local capabilities).
100
+ - A caller authenticated to the Gateway is trusted at Gateway scope. After pairing, node actions are trusted operator actions on that node.
101
+ - Operator scope levels and approval-time checks are summarized in
102
+ [Operator scopes](/gateway/operator-scopes).
103
+ - Direct loopback backend clients authenticated with the shared gateway
104
+ token/password can make internal control-plane RPCs without presenting a user
105
+ device identity. This is not a remote or browser pairing bypass: network
106
+ clients, node clients, device-token clients, and explicit device identities
107
+ still go through pairing and scope-upgrade enforcement.
108
+ - `sessionKey` is routing/context selection, not per-user auth.
109
+ - Exec approvals (allowlist + ask) are guardrails for operator intent, not hostile multi-tenant isolation.
110
+ - Nodmix's product default for trusted single-operator setups is that host exec on `gateway`/`node` is allowed without approval prompts (`security="full"`, `ask="off"` unless you tighten it). That default is intentional UX, not a vulnerability by itself.
111
+ - Exec approvals bind exact request context and best-effort direct local file operands; they do not semantically model every runtime/interpreter loader path. Use sandboxing and host isolation for strong boundaries.
112
+
113
+ If you need hostile-user isolation, split trust boundaries by OS user/host and run separate gateways.
114
+
115
+ ## Trust boundary matrix
116
+
117
+ Use this as the quick model when triaging risk:
118
+
119
+ | Boundary or control | What it means | Common misread |
120
+ | --------------------------------------------------------- | ------------------------------------------------- | ----------------------------------------------------------------------------- |
121
+ | `gateway.auth` (token/password/trusted-proxy/device auth) | Authenticates callers to gateway APIs | "Needs per-message signatures on every frame to be secure" |
122
+ | `sessionKey` | Routing key for context/session selection | "Session key is a user auth boundary" |
123
+ | Prompt/content guardrails | Reduce model abuse risk | "Prompt injection alone proves auth bypass" |
124
+ | `canvas.eval` / browser evaluate | Intentional operator capability when enabled | "Any JS eval primitive is automatically a vuln in this trust model" |
125
+ | Local TUI `!` shell | Explicit operator-triggered local execution | "Local shell convenience command is remote injection" |
126
+ | Node pairing and node commands | Operator-level remote execution on paired devices | "Remote device control should be treated as untrusted user access by default" |
127
+ | `gateway.nodes.pairing.autoApproveCidrs` | Opt-in trusted-network node enrollment policy | "A disabled-by-default allowlist is an automatic pairing vulnerability" |
128
+
129
+ ## Not vulnerabilities by design
130
+
131
+ <Accordion title="Common findings that are out of scope">
132
+
133
+ These patterns get reported often and are usually closed as no-action unless
134
+ a real boundary bypass is demonstrated:
135
+
136
+ - Prompt-injection-only chains without a policy, auth, or sandbox bypass.
137
+ - Claims that assume hostile multi-tenant operation on one shared host or
138
+ config.
139
+ - Claims that classify normal operator read-path access (for example
140
+ `sessions.list` / `sessions.preview` / `chat.history`) as IDOR in a
141
+ shared-gateway setup.
142
+ - Localhost-only deployment findings (for example HSTS on a loopback-only
143
+ gateway).
144
+ - Discord inbound webhook signature findings for inbound paths that do not
145
+ exist in this repo.
146
+ - Reports that treat node pairing metadata as a hidden second per-command
147
+ approval layer for `system.run`, when the real execution boundary is still
148
+ the gateway's global node command policy plus the node's own exec
149
+ approvals.
150
+ - Reports that treat configured `gateway.nodes.pairing.autoApproveCidrs` as a
151
+ vulnerability by itself. This setting is disabled by default, requires
152
+ explicit CIDR/IP entries, only applies to first-time `role: node` pairing with
153
+ no requested scopes, and does not auto-approve operator/browser/Control UI,
154
+ WebChat, role upgrades, scope upgrades, metadata changes, public-key changes,
155
+ or same-host loopback trusted-proxy header paths unless loopback trusted-proxy auth was explicitly enabled.
156
+ - "Missing per-user authorization" findings that treat `sessionKey` as an
157
+ auth token.
158
+
159
+ </Accordion>
160
+
161
+ ## Hardened baseline in 60 seconds
162
+
163
+ Use this baseline first, then selectively re-enable tools per trusted agent:
164
+
165
+ ```json5
166
+ {
167
+ gateway: {
168
+ mode: "local",
169
+ bind: "loopback",
170
+ auth: { mode: "token", token: "replace-with-long-random-token" },
171
+ },
172
+ session: {
173
+ dmScope: "per-channel-peer",
174
+ },
175
+ tools: {
176
+ profile: "messaging",
177
+ deny: ["group:automation", "group:runtime", "group:fs", "sessions_spawn", "sessions_send"],
178
+ fs: { workspaceOnly: true },
179
+ exec: { security: "deny", ask: "always" },
180
+ elevated: { enabled: false },
181
+ },
182
+ channels: {
183
+ whatsapp: { dmPolicy: "pairing", groups: { "*": { requireMention: true } } },
184
+ },
185
+ }
186
+ ```
187
+
188
+ This keeps the Gateway local-only, isolates DMs, and disables control-plane/runtime tools by default.
189
+
190
+ ## Shared inbox quick rule
191
+
192
+ If more than one person can DM your bot:
193
+
194
+ - Set `session.dmScope: "per-channel-peer"` (or `"per-account-channel-peer"` for multi-account channels).
195
+ - Keep `dmPolicy: "pairing"` or strict allowlists.
196
+ - Never combine shared DMs with broad tool access.
197
+ - This hardens cooperative/shared inboxes, but is not designed as hostile co-tenant isolation when users share host/config write access.
198
+
199
+ ## Context visibility model
200
+
201
+ Nodmix separates two concepts:
202
+
203
+ - **Trigger authorization**: who can trigger the agent (`dmPolicy`, `groupPolicy`, allowlists, mention gates).
204
+ - **Context visibility**: what supplemental context is injected into model input (reply body, quoted text, thread history, forwarded metadata).
205
+
206
+ Allowlists gate triggers and command authorization. The `contextVisibility` setting controls how supplemental context (quoted replies, thread roots, fetched history) is filtered:
207
+
208
+ - `contextVisibility: "all"` (default) keeps supplemental context as received.
209
+ - `contextVisibility: "allowlist"` filters supplemental context to senders allowed by the active allowlist checks.
210
+ - `contextVisibility: "allowlist_quote"` behaves like `allowlist`, but still keeps one explicit quoted reply.
211
+
212
+ Set `contextVisibility` per channel or per room/conversation. See [Group Chats](/channels/groups#context-visibility-and-allowlists) for setup details.
213
+
214
+ Advisory triage guidance:
215
+
216
+ - Claims that only show "model can see quoted or historical text from non-allowlisted senders" are hardening findings addressable with `contextVisibility`, not auth or sandbox boundary bypasses by themselves.
217
+ - To be security-impacting, reports still need a demonstrated trust-boundary bypass (auth, policy, sandbox, approval, or another documented boundary).
218
+
219
+ ## What the audit checks (high level)
220
+
221
+ - **Inbound access** (DM policies, group policies, allowlists): can strangers trigger the bot?
222
+ - **Tool blast radius** (elevated tools + open rooms): could prompt injection turn into shell/file/network actions?
223
+ - **Exec filesystem drift**: are mutating filesystem tools denied while `exec`/`process` remain available without sandbox filesystem constraints?
224
+ - **Exec approval drift** (`security=full`, `autoAllowSkills`, interpreter allowlists without `strictInlineEval`): are host-exec guardrails still doing what you think they are?
225
+ - `security="full"` is a broad posture warning, not proof of a bug. It is the chosen default for trusted personal-assistant setups; tighten it only when your threat model needs approval or allowlist guardrails.
226
+ - **Network exposure** (Gateway bind/auth, Tailscale Serve/Funnel, weak/short auth tokens).
227
+ - **Browser control exposure** (remote nodes, relay ports, remote CDP endpoints).
228
+ - **Local disk hygiene** (permissions, symlinks, config includes, "synced folder" paths).
229
+ - **Plugins** (plugins load without an explicit allowlist).
230
+ - **Policy drift/misconfig** (sandbox docker settings configured but sandbox mode off; ineffective `gateway.nodes.denyCommands` patterns because matching is exact command-name only (for example `system.run`) and does not inspect shell text; dangerous `gateway.nodes.allowCommands` entries; global `tools.profile="minimal"` overridden by per-agent profiles; plugin-owned tools reachable under permissive tool policy).
231
+ - **Runtime expectation drift** (for example assuming implicit exec still means `sandbox` when `tools.exec.host` now defaults to `auto`, or explicitly setting `tools.exec.host="sandbox"` while sandbox mode is off).
232
+ - **Model hygiene** (warn when configured models look legacy; not a hard block).
233
+
234
+ If you run `--deep`, Nodmix also attempts a best-effort live Gateway probe.
235
+
236
+ ## Credential storage map
237
+
238
+ Use this when auditing access or deciding what to back up:
239
+
240
+ - **WhatsApp**: `~/.nodmix/credentials/whatsapp/<accountId>/creds.json`
241
+ - **Telegram bot token**: config/env or `channels.telegram.tokenFile` (regular file only; symlinks rejected)
242
+ - **Discord bot token**: config/env or SecretRef (env/file/exec providers)
243
+ - **Slack tokens**: config/env (`channels.slack.*`)
244
+ - **Pairing allowlists**:
245
+ - `~/.nodmix/credentials/<channel>-allowFrom.json` (default account)
246
+ - `~/.nodmix/credentials/<channel>-<accountId>-allowFrom.json` (non-default accounts)
247
+ - **Model auth profiles**: `~/.nodmix/agents/<agentId>/agent/auth-profiles.json`
248
+ - **Codex runtime state**: `~/.nodmix/agents/<agentId>/agent/codex-home/`
249
+ - **File-backed secrets payload (optional)**: `~/.nodmix/secrets.json`
250
+ - **Legacy OAuth import**: `~/.nodmix/credentials/oauth.json`
251
+
252
+ ## Security audit checklist
253
+
254
+ When the audit prints findings, treat this as a priority order:
255
+
256
+ 1. **Anything "open" + tools enabled**: lock down DMs/groups first (pairing/allowlists), then tighten tool policy/sandboxing.
257
+ 2. **Public network exposure** (LAN bind, Funnel, missing auth): fix immediately.
258
+ 3. **Browser control remote exposure**: treat it like operator access (tailnet-only, pair nodes deliberately, avoid public exposure).
259
+ 4. **Permissions**: make sure state/config/credentials/auth are not group/world-readable.
260
+ 5. **Plugins**: only load what you explicitly trust.
261
+ 6. **Model choice**: prefer modern, instruction-hardened models for any bot with tools.
262
+
263
+ ## Security audit glossary
264
+
265
+ Each audit finding is keyed by a structured `checkId` (for example
266
+ `gateway.bind_no_auth` or `tools.exec.security_full_configured`). Common
267
+ critical severity classes:
268
+
269
+ - `fs.*` - filesystem permissions on state, config, credentials, auth profiles.
270
+ - `gateway.*` - bind mode, auth, Tailscale, Control UI, trusted-proxy setup.
271
+ - `hooks.*`, `browser.*`, `sandbox.*`, `tools.exec.*` - per-surface hardening.
272
+ - `plugins.*`, `skills.*` - plugin/skill supply chain and scan findings.
273
+ - `security.exposure.*` - cross-cutting checks where access policy meets tool blast radius.
274
+
275
+ See the full catalog with severity levels, fix keys, and auto-fix support at
276
+ [Security audit checks](/gateway/security/audit-checks).
277
+
278
+ ## Control UI over HTTP
279
+
280
+ The Control UI needs a **secure context** (HTTPS or localhost) to generate device
281
+ identity. `gateway.controlUi.allowInsecureAuth` is a local compatibility toggle:
282
+
283
+ - On localhost, it allows Control UI auth without device identity when the page
284
+ is loaded over non-secure HTTP.
285
+ - It does not bypass pairing checks.
286
+ - It does not relax remote (non-localhost) device identity requirements.
287
+
288
+ Prefer HTTPS (Tailscale Serve) or open the UI on `127.0.0.1`.
289
+
290
+ For break-glass scenarios only, `gateway.controlUi.dangerouslyDisableDeviceAuth`
291
+ disables device identity checks entirely. This is a severe security downgrade;
292
+ keep it off unless you are actively debugging and can revert quickly.
293
+
294
+ Separate from those dangerous flags, successful `gateway.auth.mode: "trusted-proxy"`
295
+ can admit **operator** Control UI sessions without device identity. That is an
296
+ intentional auth-mode behavior, not an `allowInsecureAuth` shortcut, and it still
297
+ does not extend to node-role Control UI sessions.
298
+
299
+ `nodmix security audit` warns when this setting is enabled.
300
+
301
+ ## Insecure or dangerous flags summary
302
+
303
+ `nodmix security audit` raises `config.insecure_or_dangerous_flags` when
304
+ known insecure/dangerous debug switches are enabled. Keep these unset in
305
+ production. Each enabled flag is reported as its own finding. If audit
306
+ suppressions are configured, `security.audit.suppressions.active` remains in the
307
+ active audit output even when matching findings move to `suppressedFindings`.
308
+
309
+ <AccordionGroup>
310
+ <Accordion title="Flags tracked by the audit today">
311
+ - `gateway.controlUi.allowInsecureAuth=true`
312
+ - `gateway.controlUi.dangerouslyAllowHostHeaderOriginFallback=true`
313
+ - `gateway.controlUi.dangerouslyDisableDeviceAuth=true`
314
+ - `security.audit.suppressions configured (<count>)`
315
+ - `hooks.gmail.allowUnsafeExternalContent=true`
316
+ - `hooks.mappings[<index>].allowUnsafeExternalContent=true`
317
+ - `tools.exec.applyPatch.workspaceOnly=false`
318
+ - `plugins.entries.acpx.config.permissionMode=approve-all`
319
+
320
+ </Accordion>
321
+
322
+ <Accordion title="All `dangerous*` / `dangerously*` keys in the config schema">
323
+ Control UI and browser:
324
+
325
+ - `gateway.controlUi.dangerouslyAllowHostHeaderOriginFallback`
326
+ - `gateway.controlUi.dangerouslyDisableDeviceAuth`
327
+ - `browser.ssrfPolicy.dangerouslyAllowPrivateNetwork`
328
+
329
+ Channel name-matching (bundled and plugin channels; also available per
330
+ `accounts.<accountId>` where applicable):
331
+
332
+ - `channels.discord.dangerouslyAllowNameMatching`
333
+ - `channels.slack.dangerouslyAllowNameMatching`
334
+ - `channels.googlechat.dangerouslyAllowNameMatching`
335
+ - `channels.msteams.dangerouslyAllowNameMatching`
336
+ - `channels.synology-chat.dangerouslyAllowNameMatching` (plugin channel)
337
+ - `channels.synology-chat.dangerouslyAllowInheritedWebhookPath` (plugin channel)
338
+ - `channels.zalouser.dangerouslyAllowNameMatching` (plugin channel)
339
+ - `channels.irc.dangerouslyAllowNameMatching` (plugin channel)
340
+ - `channels.mattermost.dangerouslyAllowNameMatching` (plugin channel)
341
+
342
+ Network exposure:
343
+
344
+ - `channels.telegram.network.dangerouslyAllowPrivateNetwork` (also per account)
345
+
346
+ Sandbox Docker (defaults + per-agent):
347
+
348
+ - `agents.defaults.sandbox.docker.dangerouslyAllowReservedContainerTargets`
349
+ - `agents.defaults.sandbox.docker.dangerouslyAllowExternalBindSources`
350
+ - `agents.defaults.sandbox.docker.dangerouslyAllowContainerNamespaceJoin`
351
+
352
+ </Accordion>
353
+ </AccordionGroup>
354
+
355
+ ## Reverse proxy configuration
356
+
357
+ If you run the Gateway behind a reverse proxy (nginx, Caddy, Traefik, etc.), configure
358
+ `gateway.trustedProxies` for proper forwarded-client IP handling.
359
+
360
+ When the Gateway detects proxy headers from an address that is **not** in `trustedProxies`, it will **not** treat connections as local clients. If gateway auth is disabled, those connections are rejected. This prevents authentication bypass where proxied connections would otherwise appear to come from localhost and receive automatic trust.
361
+
362
+ `gateway.trustedProxies` also feeds `gateway.auth.mode: "trusted-proxy"`, but that auth mode is stricter:
363
+
364
+ - trusted-proxy auth **fails closed on loopback-source proxies by default**
365
+ - same-host loopback reverse proxies can use `gateway.trustedProxies` for local-client detection and forwarded IP handling
366
+ - same-host loopback reverse proxies can satisfy `gateway.auth.mode: "trusted-proxy"` only when `gateway.auth.trustedProxy.allowLoopback = true`; otherwise use token/password auth
367
+
368
+ ```yaml
369
+ gateway:
370
+ trustedProxies:
371
+ - "10.0.0.1" # reverse proxy IP
372
+ # Optional. Default false.
373
+ # Only enable if your proxy cannot provide X-Forwarded-For.
374
+ allowRealIpFallback: false
375
+ auth:
376
+ mode: password
377
+ password: ${NODMIX_GATEWAY_PASSWORD}
378
+ ```
379
+
380
+ When `trustedProxies` is configured, the Gateway uses `X-Forwarded-For` to determine the client IP. `X-Real-IP` is ignored by default unless `gateway.allowRealIpFallback: true` is explicitly set.
381
+
382
+ Trusted proxy headers do not make node device pairing automatically trusted.
383
+ `gateway.nodes.pairing.autoApproveCidrs` is a separate, disabled-by-default
384
+ operator policy. Even when enabled, loopback-source trusted-proxy header paths
385
+ are excluded from node auto-approval because local callers can forge those
386
+ headers, including when loopback trusted-proxy auth is explicitly enabled.
387
+
388
+ Good reverse proxy behavior (overwrite incoming forwarding headers):
389
+
390
+ ```nginx
391
+ proxy_set_header X-Forwarded-For $remote_addr;
392
+ proxy_set_header X-Real-IP $remote_addr;
393
+ ```
394
+
395
+ Bad reverse proxy behavior (append/preserve untrusted forwarding headers):
396
+
397
+ ```nginx
398
+ proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
399
+ ```
400
+
401
+ ## HSTS and origin notes
402
+
403
+ - Nodmix gateway is local/loopback first. If you terminate TLS at a reverse proxy, set HSTS on the proxy-facing HTTPS domain there.
404
+ - If the gateway itself terminates HTTPS, you can set `gateway.http.securityHeaders.strictTransportSecurity` to emit the HSTS header from Nodmix responses.
405
+ - Detailed deployment guidance is in [Trusted Proxy Auth](/gateway/trusted-proxy-auth#tls-termination-and-hsts).
406
+ - For non-loopback Control UI deployments, `gateway.controlUi.allowedOrigins` is required by default.
407
+ - `gateway.controlUi.allowedOrigins: ["*"]` is an explicit allow-all browser-origin policy, not a hardened default. Avoid it outside tightly controlled local testing.
408
+ - Browser-origin auth failures on loopback are still rate-limited even when the
409
+ general loopback exemption is enabled, but the lockout key is scoped per
410
+ normalized `Origin` value instead of one shared localhost bucket.
411
+ - `gateway.controlUi.dangerouslyAllowHostHeaderOriginFallback=true` enables Host-header origin fallback mode; treat it as a dangerous operator-selected policy.
412
+ - Treat DNS rebinding and proxy-host header behavior as deployment hardening concerns; keep `trustedProxies` tight and avoid exposing the gateway directly to the public internet.
413
+
414
+ ## Local session logs live on disk
415
+
416
+ Nodmix stores session transcripts on disk under `~/.nodmix/agents/<agentId>/sessions/*.jsonl`.
417
+ This is required for session continuity and (optionally) session memory indexing, but it also means
418
+ **any process/user with filesystem access can read those logs**. Treat disk access as the trust
419
+ boundary and lock down permissions on `~/.nodmix` (see the audit section below). If you need
420
+ stronger isolation between agents, run them under separate OS users or separate hosts.
421
+
422
+ ## Node execution (system.run)
423
+
424
+ If a macOS node is paired, the Gateway can invoke `system.run` on that node. This is **remote code execution** on the Mac:
425
+
426
+ - Requires node pairing (approval + token).
427
+ - Gateway node pairing is not a per-command approval surface. It establishes node identity/trust and token issuance.
428
+ - The Gateway applies a coarse global node command policy via `gateway.nodes.allowCommands` / `denyCommands`.
429
+ - Controlled on the Mac via **Settings → Exec approvals** (security + ask + allowlist).
430
+ - The per-node `system.run` policy is the node's own exec approvals file (`exec.approvals.node.*`), which can be stricter or looser than the gateway's global command-ID policy.
431
+ - A node running with `security="full"` and `ask="off"` is following the default trusted-operator model. Treat that as expected behavior unless your deployment explicitly requires a tighter approval or allowlist stance.
432
+ - Approval mode binds exact request context and, when possible, one concrete local script/file operand. If Nodmix cannot identify exactly one direct local file for an interpreter/runtime command, approval-backed execution is denied rather than promising full semantic coverage.
433
+ - For `host=node`, approval-backed runs also store a canonical prepared
434
+ `systemRunPlan`; later approved forwards reuse that stored plan, and gateway
435
+ validation rejects caller edits to command/cwd/session context after the
436
+ approval request was created.
437
+ - If you don't want remote execution, set security to **deny** and remove node pairing for that Mac.
438
+
439
+ This distinction matters for triage:
440
+
441
+ - A reconnecting paired node advertising a different command list is not, by itself, a vulnerability if the Gateway global policy and the node's local exec approvals still enforce the actual execution boundary.
442
+ - Reports that treat node pairing metadata as a second hidden per-command approval layer are usually policy/UX confusion, not a security boundary bypass.
443
+
444
+ ## Dynamic skills (watcher / remote nodes)
445
+
446
+ Nodmix can refresh the skills list mid-session:
447
+
448
+ - **Skills watcher**: changes to `SKILL.md` can update the skills snapshot on the next agent turn.
449
+ - **Remote nodes**: connecting a macOS node can make macOS-only skills eligible (based on bin probing).
450
+
451
+ Treat skill folders as **trusted code** and restrict who can modify them.
452
+
453
+ ## The threat model
454
+
455
+ Your AI assistant can:
456
+
457
+ - Execute arbitrary shell commands
458
+ - Read/write files
459
+ - Access network services
460
+ - Send messages to anyone (if you give it WhatsApp access)
461
+
462
+ People who message you can:
463
+
464
+ - Try to trick your AI into doing bad things
465
+ - Social engineer access to your data
466
+ - Probe for infrastructure details
467
+
468
+ ## Core concept: access control before intelligence
469
+
470
+ Most failures here are not fancy exploits - they're "someone messaged the bot and the bot did what they asked."
471
+
472
+ Nodmix's stance:
473
+
474
+ - **Identity first:** decide who can talk to the bot (DM pairing / allowlists / explicit "open").
475
+ - **Scope next:** decide where the bot is allowed to act (group allowlists + mention gating, tools, sandboxing, device permissions).
476
+ - **Model last:** assume the model can be manipulated; design so manipulation has limited blast radius.
477
+
478
+ ## Command authorization model
479
+
480
+ Slash commands and directives are only honored for **authorized senders**. Authorization is derived from
481
+ channel allowlists/pairing plus `commands.useAccessGroups` (see [Configuration](/gateway/configuration)
482
+ and [Slash commands](/tools/slash-commands)). If a channel allowlist is empty or includes `"*"`,
483
+ commands are effectively open for that channel.
484
+
485
+ `/exec` is a session-only convenience for authorized operators. It does **not** write config or
486
+ change other sessions.
487
+
488
+ ## Control plane tools risk
489
+
490
+ Two built-in tools can make persistent control-plane changes:
491
+
492
+ - `gateway` can inspect config with `config.schema.lookup` / `config.get`, and can make persistent changes with `config.apply`, `config.patch`, and `update.run`.
493
+ - `cron` can create scheduled jobs that keep running after the original chat/task ends.
494
+
495
+ The agent-facing `gateway` runtime tool still refuses to rewrite
496
+ `tools.exec.ask` or `tools.exec.security`; legacy `tools.bash.*` aliases are
497
+ normalized to the same protected exec paths before the write.
498
+ Agent-driven `gateway config.apply` and `gateway config.patch` edits are
499
+ fail-closed by default: only a narrow set of prompt, model, and mention-gating
500
+ paths are agent-tunable. New sensitive config trees are therefore protected
501
+ unless they are deliberately added to the allowlist.
502
+
503
+ For any agent/surface that handles untrusted content, deny these by default:
504
+
505
+ ```json5
506
+ {
507
+ tools: {
508
+ deny: ["gateway", "cron", "sessions_spawn", "sessions_send"],
509
+ },
510
+ }
511
+ ```
512
+
513
+ `commands.restart=false` only blocks restart actions. It does not disable `gateway` config/update actions.
514
+
515
+ ## Plugins
516
+
517
+ Plugins run **in-process** with the Gateway. Treat them as trusted code:
518
+
519
+ - Only install plugins from sources you trust.
520
+ - Prefer explicit `plugins.allow` allowlists.
521
+ - Review plugin config before enabling.
522
+ - Restart the Gateway after plugin changes.
523
+ - If you install or update plugins (`nodmix plugins install <package>`, `nodmix plugins update <id>`), treat it like running untrusted code:
524
+ - The install path is the per-plugin directory under the active plugin install root.
525
+ - Nodmix runs a built-in dangerous-code scan before install/update. `critical` findings block by default.
526
+ - npm and git plugin installs run package-manager dependency convergence only during the explicit install/update flow. Local paths and archives are treated as self-contained plugin packages; Nodmix copies/references them without running `npm install`.
527
+ - Prefer pinned, exact versions (`@scope/pkg@1.2.3`), and inspect the unpacked code on disk before enabling.
528
+ - `--dangerously-force-unsafe-install` is break-glass only for built-in scan false positives on plugin install/update flows. It does not bypass plugin `before_install` hook policy blocks and does not bypass scan failures.
529
+ - Gateway-backed skill dependency installs follow the same dangerous/suspicious split: built-in `critical` findings block unless the caller explicitly sets `dangerouslyForceUnsafeInstall`, while suspicious findings still warn only. `nodmix skills install` remains the separate ClawHub skill download/install flow.
530
+
531
+ Details: [Plugins](/tools/plugin)
532
+
533
+ ## DM access model: pairing, allowlist, open, disabled
534
+
535
+ All current DM-capable channels support a DM policy (`dmPolicy` or `*.dm.policy`) that gates inbound DMs **before** the message is processed:
536
+
537
+ - `pairing` (default): unknown senders receive a short pairing code and the bot ignores their message until approved. Codes expire after 1 hour; repeated DMs won't resend a code until a new request is created. Pending requests are capped at **3 per channel** by default.
538
+ - `allowlist`: unknown senders are blocked (no pairing handshake).
539
+ - `open`: allow anyone to DM (public). **Requires** the channel allowlist to include `"*"` (explicit opt-in).
540
+ - `disabled`: ignore inbound DMs entirely.
541
+
542
+ Approve via CLI:
543
+
544
+ ```bash
545
+ nodmix pairing list <channel>
546
+ nodmix pairing approve <channel> <code>
547
+ ```
548
+
549
+ Details + files on disk: [Pairing](/channels/pairing)
550
+
551
+ ## DM session isolation (multi-user mode)
552
+
553
+ By default, Nodmix routes **all DMs into the main session** so your assistant has continuity across devices and channels. If **multiple people** can DM the bot (open DMs or a multi-person allowlist), consider isolating DM sessions:
554
+
555
+ ```json5
556
+ {
557
+ session: { dmScope: "per-channel-peer" },
558
+ }
559
+ ```
560
+
561
+ This prevents cross-user context leakage while keeping group chats isolated.
562
+
563
+ This is a messaging-context boundary, not a host-admin boundary. If users are mutually adversarial and share the same Gateway host/config, run separate gateways per trust boundary instead.
564
+
565
+ ### Secure DM mode (recommended)
566
+
567
+ Treat the snippet above as **secure DM mode**:
568
+
569
+ - Default: `session.dmScope: "main"` (all DMs share one session for continuity).
570
+ - Local CLI onboarding default: writes `session.dmScope: "per-channel-peer"` when unset (keeps existing explicit values).
571
+ - Secure DM mode: `session.dmScope: "per-channel-peer"` (each channel+sender pair gets an isolated DM context).
572
+ - Cross-channel peer isolation: `session.dmScope: "per-peer"` (each sender gets one session across all channels of the same type).
573
+
574
+ If you run multiple accounts on the same channel, use `per-account-channel-peer` instead. If the same person contacts you on multiple channels, use `session.identityLinks` to collapse those DM sessions into one canonical identity. See [Session Management](/concepts/session) and [Configuration](/gateway/configuration).
575
+
576
+ ## Allowlists for DMs and groups
577
+
578
+ Nodmix has two separate "who can trigger me?" layers:
579
+
580
+ - **DM allowlist** (`allowFrom` / `channels.discord.allowFrom` / `channels.slack.allowFrom`; legacy: `channels.discord.dm.allowFrom`, `channels.slack.dm.allowFrom`): who is allowed to talk to the bot in direct messages.
581
+ - When `dmPolicy="pairing"`, approvals are written to the account-scoped pairing allowlist store under `~/.nodmix/credentials/` (`<channel>-allowFrom.json` for default account, `<channel>-<accountId>-allowFrom.json` for non-default accounts), merged with config allowlists.
582
+ - **Group allowlist** (channel-specific): which groups/channels/guilds the bot will accept messages from at all.
583
+ - Common patterns:
584
+ - `channels.whatsapp.groups`, `channels.telegram.groups`, `channels.imessage.groups`: per-group defaults like `requireMention`; when set, it also acts as a group allowlist (include `"*"` to keep allow-all behavior).
585
+ - `groupPolicy="allowlist"` + `groupAllowFrom`: restrict who can trigger the bot _inside_ a group session (WhatsApp/Telegram/Signal/iMessage/Microsoft Teams).
586
+ - `channels.discord.guilds` / `channels.slack.channels`: per-surface allowlists + mention defaults.
587
+ - Group checks run in this order: `groupPolicy`/group allowlists first, mention/reply activation second.
588
+ - Replying to a bot message (implicit mention) does **not** bypass sender allowlists like `groupAllowFrom`.
589
+ - **Security note:** treat `dmPolicy="open"` and `groupPolicy="open"` as last-resort settings. They should be barely used; prefer pairing + allowlists unless you fully trust every member of the room.
590
+
591
+ Details: [Configuration](/gateway/configuration) and [Groups](/channels/groups)
592
+
593
+ ## Prompt injection (what it is, why it matters)
594
+
595
+ Prompt injection is when an attacker crafts a message that manipulates the model into doing something unsafe ("ignore your instructions", "dump your filesystem", "follow this link and run commands", etc.).
596
+
597
+ Even with strong system prompts, **prompt injection is not solved**. System prompt guardrails are soft guidance only; hard enforcement comes from tool policy, exec approvals, sandboxing, and channel allowlists (and operators can disable these by design). What helps in practice:
598
+
599
+ - Keep inbound DMs locked down (pairing/allowlists).
600
+ - Prefer mention gating in groups; avoid "always-on" bots in public rooms.
601
+ - Treat links, attachments, and pasted instructions as hostile by default.
602
+ - Run sensitive tool execution in a sandbox; keep secrets out of the agent's reachable filesystem.
603
+ - Note: sandboxing is opt-in. If sandbox mode is off, implicit `host=auto` resolves to the gateway host. Explicit `host=sandbox` still fails closed because no sandbox runtime is available. Set `host=gateway` if you want that behavior to be explicit in config.
604
+ - Limit high-risk tools (`exec`, `browser`, `web_fetch`, `web_search`) to trusted agents or explicit allowlists.
605
+ - If you allowlist interpreters (`python`, `node`, `ruby`, `perl`, `php`, `lua`, `osascript`), enable `tools.exec.strictInlineEval` so inline eval forms still need explicit approval.
606
+ - Shell approval analysis also rejects POSIX parameter-expansion forms (`$VAR`, `$?`, `$$`, `$1`, `$@`, `${…}`) inside **unquoted heredocs**, so an allowlisted heredoc body cannot sneak shell expansion past allowlist review as plain text. Quote the heredoc terminator (for example `<<'EOF'`) to opt into literal body semantics; unquoted heredocs that would have expanded variables are rejected.
607
+ - **Model choice matters:** older/smaller/legacy models are significantly less robust against prompt injection and tool misuse. For tool-enabled agents, use the strongest latest-generation, instruction-hardened model available.
608
+
609
+ Red flags to treat as untrusted:
610
+
611
+ - "Read this file/URL and do exactly what it says."
612
+ - "Ignore your system prompt or safety rules."
613
+ - "Reveal your hidden instructions or tool outputs."
614
+ - "Paste the full contents of ~/.nodmix or your logs."
615
+
616
+ ## External content special-token sanitization
617
+
618
+ Nodmix strips common self-hosted LLM chat-template special-token literals from wrapped external content and metadata before they reach the model. Covered marker families include Qwen/ChatML, Llama, Gemma, Mistral, Phi, and GPT-OSS role/turn tokens.
619
+
620
+ Why:
621
+
622
+ - OpenAI-compatible backends that front self-hosted models sometimes preserve special tokens that appear in user text, instead of masking them. An attacker who can write into inbound external content (a fetched page, an email body, a file contents tool output) could otherwise inject a synthetic `assistant` or `system` role boundary and escape the wrapped-content guardrails.
623
+ - Sanitization happens at the external-content wrapping layer, so it applies uniformly across fetch/read tools and inbound channel content rather than being per-provider.
624
+ - Outbound model responses already have a separate sanitizer that strips leaked `<tool_call>`, `<function_calls>`, `<system-reminder>`, `<previous_response>`, and similar internal runtime scaffolding from user-visible replies at the final channel delivery boundary. The external-content sanitizer is the inbound counterpart.
625
+
626
+ This does not replace the other hardening on this page - `dmPolicy`, allowlists, exec approvals, sandboxing, and `contextVisibility` still do the primary work. It closes one specific tokenizer-layer bypass against self-hosted stacks that forward user text with special tokens intact.
627
+
628
+ ## Unsafe external content bypass flags
629
+
630
+ Nodmix includes explicit bypass flags that disable external-content safety wrapping:
631
+
632
+ - `hooks.mappings[].allowUnsafeExternalContent`
633
+ - `hooks.gmail.allowUnsafeExternalContent`
634
+ - Cron payload field `allowUnsafeExternalContent`
635
+
636
+ Guidance:
637
+
638
+ - Keep these unset/false in production.
639
+ - Only enable temporarily for tightly scoped debugging.
640
+ - If enabled, isolate that agent (sandbox + minimal tools + dedicated session namespace).
641
+
642
+ Hooks risk note:
643
+
644
+ - Hook payloads are untrusted content, even when delivery comes from systems you control (mail/docs/web content can carry prompt injection).
645
+ - Weak model tiers increase this risk. For hook-driven automation, prefer strong modern model tiers and keep tool policy tight (`tools.profile: "messaging"` or stricter), plus sandboxing where possible.
646
+
647
+ ### Prompt injection does not require public DMs
648
+
649
+ Even if **only you** can message the bot, prompt injection can still happen via
650
+ any **untrusted content** the bot reads (web search/fetch results, browser pages,
651
+ emails, docs, attachments, pasted logs/code). In other words: the sender is not
652
+ the only threat surface; the **content itself** can carry adversarial instructions.
653
+
654
+ When tools are enabled, the typical risk is exfiltrating context or triggering
655
+ tool calls. Reduce the blast radius by:
656
+
657
+ - Using a read-only or tool-disabled **reader agent** to summarize untrusted content,
658
+ then pass the summary to your main agent.
659
+ - Keeping `web_search` / `web_fetch` / `browser` off for tool-enabled agents unless needed.
660
+ - For OpenResponses URL inputs (`input_file` / `input_image`), set tight
661
+ `gateway.http.endpoints.responses.files.urlAllowlist` and
662
+ `gateway.http.endpoints.responses.images.urlAllowlist`, and keep `maxUrlParts` low.
663
+ Empty allowlists are treated as unset; use `files.allowUrl: false` / `images.allowUrl: false`
664
+ if you want to disable URL fetching entirely.
665
+ - For OpenResponses file inputs, decoded `input_file` text is still injected as
666
+ **untrusted external content**. Do not rely on file text being trusted just because
667
+ the Gateway decoded it locally. The injected block still carries explicit
668
+ `<<<EXTERNAL_UNTRUSTED_CONTENT ...>>>` boundary markers plus `Source: External`
669
+ metadata, even though this path omits the longer `SECURITY NOTICE:` banner.
670
+ - The same marker-based wrapping is applied when media-understanding extracts text
671
+ from attached documents before appending that text to the media prompt.
672
+ - Enabling sandboxing and strict tool allowlists for any agent that touches untrusted input.
673
+ - Keeping secrets out of prompts; pass them via env/config on the gateway host instead.
674
+
675
+ ### Self-hosted LLM backends
676
+
677
+ OpenAI-compatible self-hosted backends such as vLLM, SGLang, TGI, LM Studio,
678
+ or custom Hugging Face tokenizer stacks can differ from hosted providers in how
679
+ chat-template special tokens are handled. If a backend tokenizes literal strings
680
+ such as `<|im_start|>`, `<|start_header_id|>`, or `<start_of_turn>` as
681
+ structural chat-template tokens inside user content, untrusted text can try to
682
+ forge role boundaries at the tokenizer layer.
683
+
684
+ Nodmix strips common model-family special-token literals from wrapped
685
+ external content before dispatching it to the model. Keep external-content
686
+ wrapping enabled, and prefer backend settings that split or escape special
687
+ tokens in user-provided content when available. Hosted providers such as OpenAI
688
+ and Anthropic already apply their own request-side sanitization.
689
+
690
+ ### Model strength (security note)
691
+
692
+ Prompt injection resistance is **not** uniform across model tiers. Smaller/cheaper models are generally more susceptible to tool misuse and instruction hijacking, especially under adversarial prompts.
693
+
694
+ <Warning>
695
+ For tool-enabled agents or agents that read untrusted content, prompt-injection risk with older/smaller models is often too high. Do not run those workloads on weak model tiers.
696
+ </Warning>
697
+
698
+ Recommendations:
699
+
700
+ - **Use the latest generation, best-tier model** for any bot that can run tools or touch files/networks.
701
+ - **Do not use older/weaker/smaller tiers** for tool-enabled agents or untrusted inboxes; the prompt-injection risk is too high.
702
+ - If you must use a smaller model, **reduce blast radius** (read-only tools, strong sandboxing, minimal filesystem access, strict allowlists).
703
+ - When running small models, **enable sandboxing for all sessions** and **disable web_search/web_fetch/browser** unless inputs are tightly controlled.
704
+ - For chat-only personal assistants with trusted input and no tools, smaller models are usually fine.
705
+
706
+ ## Reasoning and verbose output in groups
707
+
708
+ `/reasoning`, `/verbose`, and `/trace` can expose internal reasoning, tool
709
+ output, or plugin diagnostics that
710
+ was not meant for a public channel. In group settings, treat them as **debug
711
+ only** and keep them off unless you explicitly need them.
712
+
713
+ Guidance:
714
+
715
+ - Keep `/reasoning`, `/verbose`, and `/trace` disabled in public rooms.
716
+ - If you enable them, do so only in trusted DMs or tightly controlled rooms.
717
+ - Remember: verbose and trace output can include tool args, URLs, plugin diagnostics, and data the model saw.
718
+
719
+ ## Configuration hardening examples
720
+
721
+ ### File permissions
722
+
723
+ Keep config + state private on the gateway host:
724
+
725
+ - `~/.nodmix/nodmix.json`: `600` (user read/write only)
726
+ - `~/.nodmix`: `700` (user only)
727
+
728
+ `nodmix doctor` can warn and offer to tighten these permissions.
729
+
730
+ ### Network exposure (bind, port, firewall)
731
+
732
+ The Gateway multiplexes **WebSocket + HTTP** on a single port:
733
+
734
+ - Default: `18789`
735
+ - Config/flags/env: `gateway.port`, `--port`, `NODMIX_GATEWAY_PORT`
736
+
737
+ This HTTP surface includes the Control UI and the canvas host:
738
+
739
+ - Control UI (SPA assets) (default base path `/`)
740
+ - Canvas host: `/__nodmix__/canvas/` and `/__nodmix__/a2ui/` (arbitrary HTML/JS; treat as untrusted content)
741
+
742
+ If you load canvas content in a normal browser, treat it like any other untrusted web page:
743
+
744
+ - Don't expose the canvas host to untrusted networks/users.
745
+ - Don't make canvas content share the same origin as privileged web surfaces unless you fully understand the implications.
746
+
747
+ Bind mode controls where the Gateway listens:
748
+
749
+ - `gateway.bind: "loopback"` (default): only local clients can connect.
750
+ - Non-loopback binds (`"lan"`, `"tailnet"`, `"custom"`) expand the attack surface. Only use them with gateway auth (shared token/password or a correctly configured trusted proxy) and a real firewall.
751
+
752
+ Rules of thumb:
753
+
754
+ - Prefer Tailscale Serve over LAN binds (Serve keeps the Gateway on loopback, and Tailscale handles access).
755
+ - If you must bind to LAN, firewall the port to a tight allowlist of source IPs; do not port-forward it broadly.
756
+ - Never expose the Gateway unauthenticated on `0.0.0.0`.
757
+
758
+ ### Docker port publishing with UFW
759
+
760
+ If you run Nodmix with Docker on a VPS, remember that published container ports
761
+ (`-p HOST:CONTAINER` or Compose `ports:`) are routed through Docker's forwarding
762
+ chains, not only host `INPUT` rules.
763
+
764
+ To keep Docker traffic aligned with your firewall policy, enforce rules in
765
+ `DOCKER-USER` (this chain is evaluated before Docker's own accept rules).
766
+ On many modern distros, `iptables`/`ip6tables` use the `iptables-nft` frontend
767
+ and still apply these rules to the nftables backend.
768
+
769
+ Minimal allowlist example (IPv4):
770
+
771
+ ```bash
772
+ # /etc/ufw/after.rules (append as its own *filter section)
773
+ *filter
774
+ :DOCKER-USER - [0:0]
775
+ -A DOCKER-USER -m conntrack --ctstate ESTABLISHED,RELATED -j RETURN
776
+ -A DOCKER-USER -s 127.0.0.0/8 -j RETURN
777
+ -A DOCKER-USER -s 10.0.0.0/8 -j RETURN
778
+ -A DOCKER-USER -s 172.16.0.0/12 -j RETURN
779
+ -A DOCKER-USER -s 192.168.0.0/16 -j RETURN
780
+ -A DOCKER-USER -s 100.64.0.0/10 -j RETURN
781
+ -A DOCKER-USER -p tcp --dport 80 -j RETURN
782
+ -A DOCKER-USER -p tcp --dport 443 -j RETURN
783
+ -A DOCKER-USER -m conntrack --ctstate NEW -j DROP
784
+ -A DOCKER-USER -j RETURN
785
+ COMMIT
786
+ ```
787
+
788
+ IPv6 has separate tables. Add a matching policy in `/etc/ufw/after6.rules` if
789
+ Docker IPv6 is enabled.
790
+
791
+ Avoid hardcoding interface names like `eth0` in docs snippets. Interface names
792
+ vary across VPS images (`ens3`, `enp*`, etc.) and mismatches can accidentally
793
+ skip your deny rule.
794
+
795
+ Quick validation after reload:
796
+
797
+ ```bash
798
+ ufw reload
799
+ iptables -S DOCKER-USER
800
+ ip6tables -S DOCKER-USER
801
+ nmap -sT -p 1-65535 <public-ip> --open
802
+ ```
803
+
804
+ Expected external ports should be only what you intentionally expose (for most
805
+ setups: SSH + your reverse proxy ports).
806
+
807
+ ### mDNS/Bonjour discovery
808
+
809
+ When the bundled `bonjour` plugin is enabled, the Gateway broadcasts its presence via mDNS (`_nodmix-gw._tcp` on port 5353) for local device discovery. In full mode, this includes TXT records that may expose operational details:
810
+
811
+ - `cliPath`: full filesystem path to the CLI binary (reveals username and install location)
812
+ - `sshPort`: advertises SSH availability on the host
813
+ - `displayName`, `lanHost`: hostname information
814
+
815
+ **Operational security consideration:** Broadcasting infrastructure details makes reconnaissance easier for anyone on the local network. Even "harmless" info like filesystem paths and SSH availability helps attackers map your environment.
816
+
817
+ **Recommendations:**
818
+
819
+ 1. **Keep Bonjour disabled unless LAN discovery is needed.** Bonjour auto-starts on macOS hosts and is opt-in elsewhere; direct Gateway URLs, Tailnet, SSH, or wide-area DNS-SD avoid local multicast.
820
+
821
+ 2. **Minimal mode** (default when Bonjour is enabled, recommended for exposed gateways): omit sensitive fields from mDNS broadcasts:
822
+
823
+ ```json5
824
+ {
825
+ discovery: {
826
+ mdns: { mode: "minimal" },
827
+ },
828
+ }
829
+ ```
830
+
831
+ 3. **Disable mDNS mode** if you want to keep the plugin enabled but suppress local device discovery:
832
+
833
+ ```json5
834
+ {
835
+ discovery: {
836
+ mdns: { mode: "off" },
837
+ },
838
+ }
839
+ ```
840
+
841
+ 4. **Full mode** (opt-in): include `cliPath` + `sshPort` in TXT records:
842
+
843
+ ```json5
844
+ {
845
+ discovery: {
846
+ mdns: { mode: "full" },
847
+ },
848
+ }
849
+ ```
850
+
851
+ 5. **Environment variable** (alternative): set `NODMIX_DISABLE_BONJOUR=1` to disable mDNS without config changes.
852
+
853
+ When Bonjour is enabled in minimal mode, the Gateway broadcasts enough for device discovery (`role`, `gatewayPort`, `transport`) but omits `cliPath` and `sshPort`. Apps that need CLI path information can fetch it via the authenticated WebSocket connection instead.
854
+
855
+ ### Lock down the Gateway WebSocket (local auth)
856
+
857
+ Gateway auth is **required by default**. If no valid gateway auth path is configured,
858
+ the Gateway refuses WebSocket connections (fail-closed).
859
+
860
+ Onboarding generates a token by default (even for loopback) so
861
+ local clients must authenticate.
862
+
863
+ Set a token so **all** WS clients must authenticate:
864
+
865
+ ```json5
866
+ {
867
+ gateway: {
868
+ auth: { mode: "token", token: "your-token" },
869
+ },
870
+ }
871
+ ```
872
+
873
+ Doctor can generate one for you: `nodmix doctor --generate-gateway-token`.
874
+
875
+ <Note>
876
+ `gateway.remote.token` and `gateway.remote.password` are client credential sources. They do **not** protect local WS access by themselves. Local call paths can use `gateway.remote.*` as fallback only when `gateway.auth.*` is unset. If `gateway.auth.token` or `gateway.auth.password` is explicitly configured via SecretRef and unresolved, resolution fails closed (no remote fallback masking).
877
+ </Note>
878
+ Optional: pin remote TLS with `gateway.remote.tlsFingerprint` when using `wss://`.
879
+ Plaintext `ws://` is accepted for loopback, private IP literals, `.local`, and
880
+ Tailnet `*.ts.net` gateway URLs. For other trusted private-DNS names, set
881
+ `NODMIX_ALLOW_INSECURE_PRIVATE_WS=1` on the client process as break-glass.
882
+ This is intentionally process environment only, not an `nodmix.json` config
883
+ key.
884
+ Mobile pairing and Android manual or scanned gateway routes are stricter:
885
+ cleartext is accepted for loopback, but private-LAN, link-local, `.local`, and
886
+ dotless hostnames must use TLS unless you explicitly opt into the trusted
887
+ private-network cleartext path.
888
+
889
+ Local device pairing:
890
+
891
+ - Device pairing is auto-approved for direct local loopback connects to keep
892
+ same-host clients smooth.
893
+ - Nodmix also has a narrow backend/container-local self-connect path for
894
+ trusted shared-secret helper flows.
895
+ - Tailnet and LAN connects, including same-host tailnet binds, are treated as
896
+ remote for pairing and still need approval.
897
+ - Forwarded-header evidence on a loopback request disqualifies loopback
898
+ locality. Metadata-upgrade auto-approval is scoped narrowly. See
899
+ [Gateway pairing](/gateway/pairing) for both rules.
900
+
901
+ Auth modes:
902
+
903
+ - `gateway.auth.mode: "token"`: shared bearer token (recommended for most setups).
904
+ - `gateway.auth.mode: "password"`: password auth (prefer setting via env: `NODMIX_GATEWAY_PASSWORD`).
905
+ - `gateway.auth.mode: "trusted-proxy"`: trust an identity-aware reverse proxy to authenticate users and pass identity via headers (see [Trusted Proxy Auth](/gateway/trusted-proxy-auth)).
906
+
907
+ Rotation checklist (token/password):
908
+
909
+ 1. Generate/set a new secret (`gateway.auth.token` or `NODMIX_GATEWAY_PASSWORD`).
910
+ 2. Restart the Gateway (or restart the macOS app if it supervises the Gateway).
911
+ 3. Update any remote clients (`gateway.remote.token` / `.password` on machines that call into the Gateway).
912
+ 4. Verify you can no longer connect with the old credentials.
913
+
914
+ ### Tailscale Serve identity headers
915
+
916
+ When `gateway.auth.allowTailscale` is `true` (default for Serve), Nodmix
917
+ accepts Tailscale Serve identity headers (`tailscale-user-login`) for Control
918
+ UI/WebSocket authentication. Nodmix verifies the identity by resolving the
919
+ `x-forwarded-for` address through the local Tailscale daemon (`tailscale whois`)
920
+ and matching it to the header. This only triggers for requests that hit loopback
921
+ and include `x-forwarded-for`, `x-forwarded-proto`, and `x-forwarded-host` as
922
+ injected by Tailscale.
923
+ For this async identity check path, failed attempts for the same `{scope, ip}`
924
+ are serialized before the limiter records the failure. Concurrent bad retries
925
+ from one Serve client can therefore lock out the second attempt immediately
926
+ instead of racing through as two plain mismatches.
927
+ HTTP API endpoints (for example `/v1/*`, `/tools/invoke`, and `/api/channels/*`)
928
+ do **not** use Tailscale identity-header auth. They still follow the gateway's
929
+ configured HTTP auth mode.
930
+
931
+ Important boundary note:
932
+
933
+ - Gateway HTTP bearer auth is effectively all-or-nothing operator access.
934
+ - Treat credentials that can call `/v1/chat/completions`, `/v1/responses`, plugin routes such as `/api/v1/admin/rpc`, or `/api/channels/*` as full-access operator secrets for that gateway.
935
+ - On the OpenAI-compatible HTTP surface, shared-secret bearer auth restores the full default operator scopes (`operator.admin`, `operator.approvals`, `operator.pairing`, `operator.read`, `operator.talk.secrets`, `operator.write`) and owner semantics for agent turns; narrower `x-nodmix-scopes` values do not reduce that shared-secret path.
936
+ - Per-request scope semantics on HTTP only apply when the request comes from an identity-bearing mode such as trusted proxy auth, or from an explicitly no-auth private ingress.
937
+ - In those identity-bearing modes, omitting `x-nodmix-scopes` falls back to the normal operator default scope set; send the header explicitly when you want a narrower scope set.
938
+ - `/tools/invoke` and HTTP session history endpoints follow the same shared-secret rule: token/password bearer auth is treated as full operator access there too, while identity-bearing modes still honor declared scopes.
939
+ - Do not share these credentials with untrusted callers; prefer separate gateways per trust boundary.
940
+
941
+ **Trust assumption:** tokenless Serve auth assumes the gateway host is trusted.
942
+ Do not treat this as protection against hostile same-host processes. If untrusted
943
+ local code may run on the gateway host, disable `gateway.auth.allowTailscale`
944
+ and require explicit shared-secret auth with `gateway.auth.mode: "token"` or
945
+ `"password"`.
946
+
947
+ **Security rule:** do not forward these headers from your own reverse proxy. If
948
+ you terminate TLS or proxy in front of the gateway, disable
949
+ `gateway.auth.allowTailscale` and use shared-secret auth (`gateway.auth.mode:
950
+ "token"` or `"password"`) or [Trusted Proxy Auth](/gateway/trusted-proxy-auth)
951
+ instead.
952
+
953
+ Trusted proxies:
954
+
955
+ - If you terminate TLS in front of the Gateway, set `gateway.trustedProxies` to your proxy IPs.
956
+ - Nodmix will trust `x-forwarded-for` (or `x-real-ip`) from those IPs to determine the client IP for local pairing checks and HTTP auth/local checks.
957
+ - Ensure your proxy **overwrites** `x-forwarded-for` and blocks direct access to the Gateway port.
958
+
959
+ See [Tailscale](/gateway/tailscale) and [Web overview](/web).
960
+
961
+ ### Browser control via node host (recommended)
962
+
963
+ If your Gateway is remote but the browser runs on another machine, run a **node host**
964
+ on the browser machine and let the Gateway proxy browser actions (see [Browser tool](/tools/browser)).
965
+ Treat node pairing like admin access.
966
+
967
+ Recommended pattern:
968
+
969
+ - Keep the Gateway and node host on the same tailnet (Tailscale).
970
+ - Pair the node intentionally; disable browser proxy routing if you don't need it.
971
+
972
+ Avoid:
973
+
974
+ - Exposing relay/control ports over LAN or public Internet.
975
+ - Tailscale Funnel for browser control endpoints (public exposure).
976
+
977
+ ### Secrets on disk
978
+
979
+ Assume anything under `~/.nodmix/` (or `$NODMIX_STATE_DIR/`) may contain secrets or private data:
980
+
981
+ - `nodmix.json`: config may include tokens (gateway, remote gateway), provider settings, and allowlists.
982
+ - `credentials/**`: channel credentials (example: WhatsApp creds), pairing allowlists, legacy OAuth imports.
983
+ - `agents/<agentId>/agent/auth-profiles.json`: API keys, token profiles, OAuth tokens, and optional `keyRef`/`tokenRef`.
984
+ - `agents/<agentId>/agent/codex-home/**`: per-agent Codex app-server account, config, skills, plugins, native thread state, and diagnostics.
985
+ - `secrets.json` (optional): file-backed secret payload used by `file` SecretRef providers (`secrets.providers`).
986
+ - `agents/<agentId>/agent/auth.json`: legacy compatibility file. Static `api_key` entries are scrubbed when discovered.
987
+ - `agents/<agentId>/sessions/**`: session transcripts (`*.jsonl`) + routing metadata (`sessions.json`) that can contain private messages and tool output.
988
+ - bundled plugin packages: installed plugins (plus their `node_modules/`).
989
+ - `sandboxes/**`: tool sandbox workspaces; can accumulate copies of files you read/write inside the sandbox.
990
+
991
+ Hardening tips:
992
+
993
+ - Keep permissions tight (`700` on dirs, `600` on files).
994
+ - Use full-disk encryption on the gateway host.
995
+ - Prefer a dedicated OS user account for the Gateway if the host is shared.
996
+
997
+ ### Workspace `.env` files
998
+
999
+ Nodmix loads workspace-local `.env` files for agents and tools, but never lets those files silently override gateway runtime controls.
1000
+
1001
+ - Any key that starts with `NODMIX_*` is blocked from untrusted workspace `.env` files.
1002
+ - Channel endpoint settings for Matrix, Mattermost, IRC, and Synology Chat are also blocked from workspace `.env` overrides, so cloned workspaces cannot redirect bundled connector traffic through local endpoint config. Endpoint env keys (such as `MATRIX_HOMESERVER`, `MATTERMOST_URL`, `IRC_HOST`, `SYNOLOGY_CHAT_INCOMING_URL`) must come from the gateway process environment or `env.shellEnv`, not from a workspace-loaded `.env`.
1003
+ - The block is fail-closed: a new runtime-control variable added in a future release cannot be inherited from a checked-in or attacker-supplied `.env`; the key is ignored and the gateway keeps its own value.
1004
+ - Trusted process/OS environment variables (the gateway's own shell, launchd/systemd unit, app bundle) still apply - this only constrains `.env` file loading.
1005
+
1006
+ Why: workspace `.env` files frequently live next to agent code, get committed by accident, or get written by tools. Blocking the whole `NODMIX_*` prefix means adding a new `NODMIX_*` flag later can never regress into silent inheritance from workspace state.
1007
+
1008
+ ### Logs and transcripts (redaction and retention)
1009
+
1010
+ Logs and transcripts can leak sensitive info even when access controls are correct:
1011
+
1012
+ - Gateway logs may include tool summaries, errors, and URLs.
1013
+ - Session transcripts can include pasted secrets, file contents, command output, and links.
1014
+
1015
+ Recommendations:
1016
+
1017
+ - Keep log and transcript redaction on (`logging.redactSensitive: "tools"`; default).
1018
+ - Add custom patterns for your environment via `logging.redactPatterns` (tokens, hostnames, internal URLs).
1019
+ - When sharing diagnostics, prefer `nodmix status --all` (pasteable, secrets redacted) over raw logs.
1020
+ - Prune old session transcripts and log files if you don't need long retention.
1021
+
1022
+ Details: [Logging](/gateway/logging)
1023
+
1024
+ ### DMs: pairing by default
1025
+
1026
+ ```json5
1027
+ {
1028
+ channels: { whatsapp: { dmPolicy: "pairing" } },
1029
+ }
1030
+ ```
1031
+
1032
+ ### Groups: require mention everywhere
1033
+
1034
+ ```json
1035
+ {
1036
+ "channels": {
1037
+ "whatsapp": {
1038
+ "groups": {
1039
+ "*": { "requireMention": true }
1040
+ }
1041
+ }
1042
+ },
1043
+ "agents": {
1044
+ "list": [
1045
+ {
1046
+ "id": "main",
1047
+ "groupChat": { "mentionPatterns": ["@nodmix", "@mybot"] }
1048
+ }
1049
+ ]
1050
+ }
1051
+ }
1052
+ ```
1053
+
1054
+ In group chats, only respond when explicitly mentioned.
1055
+
1056
+ ### Separate numbers (WhatsApp, Signal, Telegram)
1057
+
1058
+ For phone-number-based channels, consider running your AI on a separate phone number from your personal one:
1059
+
1060
+ - Personal number: Your conversations stay private
1061
+ - Bot number: AI handles these, with appropriate boundaries
1062
+
1063
+ ### Read-only mode (via sandbox and tools)
1064
+
1065
+ You can build a read-only profile by combining:
1066
+
1067
+ - `agents.defaults.sandbox.workspaceAccess: "ro"` (or `"none"` for no workspace access)
1068
+ - tool allow/deny lists that block `write`, `edit`, `apply_patch`, `exec`, `process`, etc.
1069
+
1070
+ Additional hardening options:
1071
+
1072
+ - `tools.exec.applyPatch.workspaceOnly: true` (default): ensures `apply_patch` cannot write/delete outside the workspace directory even when sandboxing is off. Set to `false` only if you intentionally want `apply_patch` to touch files outside the workspace.
1073
+ - `tools.fs.workspaceOnly: true` (optional): restricts `read`/`write`/`edit`/`apply_patch` paths and native prompt image auto-load paths to the workspace directory (useful if you allow absolute paths today and want a single guardrail).
1074
+ - Keep filesystem roots narrow: avoid broad roots like your home directory for agent workspaces/sandbox workspaces. Broad roots can expose sensitive local files (for example state/config under `~/.nodmix`) to filesystem tools.
1075
+
1076
+ ### Secure baseline (copy/paste)
1077
+
1078
+ One "safe default" config that keeps the Gateway private, requires DM pairing, and avoids always-on group bots:
1079
+
1080
+ ```json5
1081
+ {
1082
+ gateway: {
1083
+ mode: "local",
1084
+ bind: "loopback",
1085
+ port: 18789,
1086
+ auth: { mode: "token", token: "your-long-random-token" },
1087
+ },
1088
+ channels: {
1089
+ whatsapp: {
1090
+ dmPolicy: "pairing",
1091
+ groups: { "*": { requireMention: true } },
1092
+ },
1093
+ },
1094
+ }
1095
+ ```
1096
+
1097
+ If you want "safer by default" tool execution too, add a sandbox + deny dangerous tools for any non-owner agent (example below under "Per-agent access profiles").
1098
+
1099
+ Built-in baseline for chat-driven agent turns: non-owner senders cannot use the `cron` or `gateway` tools.
1100
+
1101
+ ## Sandboxing (recommended)
1102
+
1103
+ Dedicated doc: [Sandboxing](/gateway/sandboxing)
1104
+
1105
+ Two complementary approaches:
1106
+
1107
+ - **Run the full Gateway in Docker** (container boundary): [Docker](/install/docker)
1108
+ - **Tool sandbox** (`agents.defaults.sandbox`, host gateway + sandbox-isolated tools; Docker is the default backend): [Sandboxing](/gateway/sandboxing)
1109
+
1110
+ <Note>
1111
+ To prevent cross-agent access, keep `agents.defaults.sandbox.scope` at `"agent"` (default) or `"session"` for stricter per-session isolation. `scope: "shared"` uses a single container or workspace.
1112
+ </Note>
1113
+
1114
+ Also consider agent workspace access inside the sandbox:
1115
+
1116
+ - `agents.defaults.sandbox.workspaceAccess: "none"` (default) keeps the agent workspace off-limits; tools run against a sandbox workspace under `~/.nodmix/sandboxes`
1117
+ - `agents.defaults.sandbox.workspaceAccess: "ro"` mounts the agent workspace read-only at `/agent` (disables `write`/`edit`/`apply_patch`)
1118
+ - `agents.defaults.sandbox.workspaceAccess: "rw"` mounts the agent workspace read/write at `/workspace`
1119
+ - Extra `sandbox.docker.binds` are validated against normalized and canonicalized source paths. Parent-symlink tricks and canonical home aliases still fail closed if they resolve into blocked roots such as `/etc`, `/var/run`, or credential directories under the OS home.
1120
+
1121
+ <Warning>
1122
+ `tools.elevated` is the global baseline escape hatch that runs exec outside the sandbox. The effective host is `gateway` by default, or `node` when the exec target is configured to `node`. Keep `tools.elevated.allowFrom` tight and do not enable it for strangers. You can further restrict elevated per agent via `agents.list[].tools.elevated`. See [Elevated mode](/tools/elevated).
1123
+ </Warning>
1124
+
1125
+ ### Sub-agent delegation guardrail
1126
+
1127
+ If you allow session tools, treat delegated sub-agent runs as another boundary decision:
1128
+
1129
+ - Deny `sessions_spawn` unless the agent truly needs delegation.
1130
+ - Keep `agents.defaults.subagents.allowAgents` and any per-agent `agents.list[].subagents.allowAgents` overrides restricted to known-safe target agents.
1131
+ - For any workflow that must remain sandboxed, call `sessions_spawn` with `sandbox: "require"` (default is `inherit`).
1132
+ - `sandbox: "require"` fails fast when the target child runtime is not sandboxed.
1133
+
1134
+ ## Browser control risks
1135
+
1136
+ Enabling browser control gives the model the ability to drive a real browser.
1137
+ If that browser profile already contains logged-in sessions, the model can
1138
+ access those accounts and data. Treat browser profiles as **sensitive state**:
1139
+
1140
+ - Prefer a dedicated profile for the agent (the default `nodmix` profile).
1141
+ - Avoid pointing the agent at your personal daily-driver profile.
1142
+ - Keep host browser control disabled for sandboxed agents unless you trust them.
1143
+ - The standalone loopback browser control API only honors shared-secret auth
1144
+ (gateway token bearer auth or gateway password). It does not consume
1145
+ trusted-proxy or Tailscale Serve identity headers.
1146
+ - Treat browser downloads as untrusted input; prefer an isolated downloads directory.
1147
+ - Disable browser sync/password managers in the agent profile if possible (reduces blast radius).
1148
+ - For remote gateways, assume "browser control" is equivalent to "operator access" to whatever that profile can reach.
1149
+ - Keep the Gateway and node hosts tailnet-only; avoid exposing browser control ports to LAN or public Internet.
1150
+ - Disable browser proxy routing when you don't need it (`gateway.nodes.browser.mode="off"`).
1151
+ - Chrome MCP existing-session mode is **not** "safer"; it can act as you in whatever that host Chrome profile can reach.
1152
+
1153
+ ### Browser SSRF policy (strict by default)
1154
+
1155
+ Nodmix's browser navigation policy is strict by default: private/internal destinations stay blocked unless you explicitly opt in.
1156
+
1157
+ - Default: `browser.ssrfPolicy.dangerouslyAllowPrivateNetwork` is unset, so browser navigation keeps private/internal/special-use destinations blocked.
1158
+ - Legacy alias: `browser.ssrfPolicy.allowPrivateNetwork` is still accepted for compatibility.
1159
+ - Opt-in mode: set `browser.ssrfPolicy.dangerouslyAllowPrivateNetwork: true` to allow private/internal/special-use destinations.
1160
+ - In strict mode, use `hostnameAllowlist` (patterns like `*.example.com`) and `allowedHostnames` (exact host exceptions, including blocked names like `localhost`) for explicit exceptions.
1161
+ - Navigation is checked before request and best-effort re-checked on the final `http(s)` URL after navigation to reduce redirect-based pivots.
1162
+
1163
+ Example strict policy:
1164
+
1165
+ ```json5
1166
+ {
1167
+ browser: {
1168
+ ssrfPolicy: {
1169
+ dangerouslyAllowPrivateNetwork: false,
1170
+ hostnameAllowlist: ["*.example.com", "example.com"],
1171
+ allowedHostnames: ["localhost"],
1172
+ },
1173
+ },
1174
+ }
1175
+ ```
1176
+
1177
+ ## Per-agent access profiles (multi-agent)
1178
+
1179
+ With multi-agent routing, each agent can have its own sandbox + tool policy:
1180
+ use this to give **full access**, **read-only**, or **no access** per agent.
1181
+ See [Multi-Agent Sandbox & Tools](/tools/multi-agent-sandbox-tools) for full details
1182
+ and precedence rules.
1183
+
1184
+ Common use cases:
1185
+
1186
+ - Personal agent: full access, no sandbox
1187
+ - Family/work agent: sandboxed + read-only tools
1188
+ - Public agent: sandboxed + no filesystem/shell tools
1189
+
1190
+ ### Example: full access (no sandbox)
1191
+
1192
+ ```json5
1193
+ {
1194
+ agents: {
1195
+ list: [
1196
+ {
1197
+ id: "personal",
1198
+ workspace: "~/.nodmix/workspace-personal",
1199
+ sandbox: { mode: "off" },
1200
+ },
1201
+ ],
1202
+ },
1203
+ }
1204
+ ```
1205
+
1206
+ ### Example: read-only tools + read-only workspace
1207
+
1208
+ ```json5
1209
+ {
1210
+ agents: {
1211
+ list: [
1212
+ {
1213
+ id: "family",
1214
+ workspace: "~/.nodmix/workspace-family",
1215
+ sandbox: {
1216
+ mode: "all",
1217
+ scope: "agent",
1218
+ workspaceAccess: "ro",
1219
+ },
1220
+ tools: {
1221
+ allow: ["read"],
1222
+ deny: ["write", "edit", "apply_patch", "exec", "process", "browser"],
1223
+ },
1224
+ },
1225
+ ],
1226
+ },
1227
+ }
1228
+ ```
1229
+
1230
+ ### Example: no filesystem/shell access (provider messaging allowed)
1231
+
1232
+ ```json5
1233
+ {
1234
+ agents: {
1235
+ list: [
1236
+ {
1237
+ id: "public",
1238
+ workspace: "~/.nodmix/workspace-public",
1239
+ sandbox: {
1240
+ mode: "all",
1241
+ scope: "agent",
1242
+ workspaceAccess: "none",
1243
+ },
1244
+ // Session tools can reveal sensitive data from transcripts. By default Nodmix limits these tools
1245
+ // to the current session + spawned subagent sessions, but you can clamp further if needed.
1246
+ // See `tools.sessions.visibility` in the configuration reference.
1247
+ tools: {
1248
+ sessions: { visibility: "tree" }, // self | tree | agent | all
1249
+ allow: [
1250
+ "sessions_list",
1251
+ "sessions_history",
1252
+ "sessions_send",
1253
+ "sessions_spawn",
1254
+ "session_status",
1255
+ "whatsapp",
1256
+ "telegram",
1257
+ "slack",
1258
+ "discord",
1259
+ ],
1260
+ deny: [
1261
+ "read",
1262
+ "write",
1263
+ "edit",
1264
+ "apply_patch",
1265
+ "exec",
1266
+ "process",
1267
+ "browser",
1268
+ "canvas",
1269
+ "nodes",
1270
+ "cron",
1271
+ "gateway",
1272
+ "image",
1273
+ ],
1274
+ },
1275
+ },
1276
+ ],
1277
+ },
1278
+ }
1279
+ ```
1280
+
1281
+ ## Incident response
1282
+
1283
+ If your AI does something bad:
1284
+
1285
+ ### Contain
1286
+
1287
+ 1. **Stop it:** stop the macOS app (if it supervises the Gateway) or terminate your `nodmix gateway` process.
1288
+ 2. **Close exposure:** set `gateway.bind: "loopback"` (or disable Tailscale Funnel/Serve) until you understand what happened.
1289
+ 3. **Freeze access:** switch risky DMs/groups to `dmPolicy: "disabled"` / require mentions, and remove `"*"` allow-all entries if you had them.
1290
+
1291
+ ### Rotate (assume compromise if secrets leaked)
1292
+
1293
+ 1. Rotate Gateway auth (`gateway.auth.token` / `NODMIX_GATEWAY_PASSWORD`) and restart.
1294
+ 2. Rotate remote client secrets (`gateway.remote.token` / `.password`) on any machine that can call the Gateway.
1295
+ 3. Rotate provider/API credentials (WhatsApp creds, Slack/Discord tokens, model/API keys in `auth-profiles.json`, and encrypted secrets payload values when used).
1296
+
1297
+ ### Audit
1298
+
1299
+ 1. Check Gateway logs: `/tmp/nodmix/nodmix-YYYY-MM-DD.log` (or `logging.file`).
1300
+ 2. Review the relevant transcript(s): `~/.nodmix/agents/<agentId>/sessions/*.jsonl`.
1301
+ 3. Review recent config changes (anything that could have widened access: `gateway.bind`, `gateway.auth`, dm/group policies, `tools.elevated`, plugin changes).
1302
+ 4. Re-run `nodmix security audit --deep` and confirm critical findings are resolved.
1303
+
1304
+ ### Collect for a report
1305
+
1306
+ - Timestamp, gateway host OS + Nodmix version
1307
+ - The session transcript(s) + a short log tail (after redacting)
1308
+ - What the attacker sent + what the agent did
1309
+ - Whether the Gateway was exposed beyond loopback (LAN/Tailscale Funnel/Serve)
1310
+
1311
+ ## Secret scanning
1312
+
1313
+ CI runs the pre-commit `detect-private-key` hook over the repository. If it
1314
+ fails, remove or rotate the committed key material, then reproduce locally:
1315
+
1316
+ ```bash
1317
+ pre-commit run --all-files detect-private-key
1318
+ ```
1319
+
1320
+ ## Reporting security issues
1321
+
1322
+ Found a vulnerability in Nodmix? Please report responsibly:
1323
+
1324
+ 1. Email: [security@nodmix.ai](mailto:security@nodmix.ai)
1325
+ 2. Don't post publicly until fixed
1326
+ 3. We'll credit you (unless you prefer anonymity)