@bolloon/bolloon-agent 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (808) hide show
  1. package/README.md +462 -0
  2. package/dist/agents/constraint-layer.js +211 -0
  3. package/dist/agents/constraint-layer.js.map +1 -0
  4. package/dist/agents/pi-sdk.js +1199 -0
  5. package/dist/agents/pi-sdk.js.map +1 -0
  6. package/dist/agents/protocol.js +229 -0
  7. package/dist/agents/subagent-manager.js +361 -0
  8. package/dist/agents/workflow-engine.js +222 -0
  9. package/dist/agents/workflow-engine.js.map +1 -0
  10. package/dist/bollharness-integration/bollharness-integration/context-router-judgment.d.ts +48 -0
  11. package/dist/bollharness-integration/bollharness-integration/context-router-judgment.js +261 -0
  12. package/dist/bollharness-integration/bollharness-integration/context-router.d.ts +110 -0
  13. package/dist/bollharness-integration/bollharness-integration/context-router.js +542 -0
  14. package/dist/bollharness-integration/bollharness-integration/gate-state-machine.d.ts +87 -0
  15. package/dist/bollharness-integration/bollharness-integration/gate-state-machine.js +231 -0
  16. package/dist/bollharness-integration/bollharness-integration/gate-transition-hooks.d.ts +30 -0
  17. package/dist/bollharness-integration/bollharness-integration/gate-transition-hooks.js +91 -0
  18. package/dist/bollharness-integration/bollharness-integration/guard-checker.d.ts +105 -0
  19. package/dist/bollharness-integration/bollharness-integration/guard-checker.js +353 -0
  20. package/dist/bollharness-integration/bollharness-integration/index.d.ts +66 -0
  21. package/dist/bollharness-integration/bollharness-integration/index.js +32 -0
  22. package/dist/bollharness-integration/bollharness-integration/integration.d.ts +219 -0
  23. package/dist/bollharness-integration/bollharness-integration/integration.js +420 -0
  24. package/dist/bollharness-integration/bollharness-integration/skill-adapter.d.ts +151 -0
  25. package/dist/bollharness-integration/bollharness-integration/skill-adapter.js +518 -0
  26. package/dist/bollharness-integration/context-router-judgment.js +269 -0
  27. package/dist/bollharness-integration/context-router-judgment.js.map +1 -0
  28. package/dist/bollharness-integration/context-router.js +584 -0
  29. package/dist/bollharness-integration/context-router.js.map +1 -0
  30. package/dist/bollharness-integration/gate-state-machine.js +349 -0
  31. package/dist/bollharness-integration/gate-state-machine.js.map +1 -0
  32. package/dist/bollharness-integration/gate-transition-hooks.js +136 -0
  33. package/dist/bollharness-integration/gate-transition-hooks.js.map +1 -0
  34. package/dist/bollharness-integration/guard-checker.js +400 -0
  35. package/dist/bollharness-integration/guard-checker.js.map +1 -0
  36. package/dist/bollharness-integration/index.js +142 -0
  37. package/dist/bollharness-integration/index.js.map +1 -0
  38. package/dist/bollharness-integration/integration.js +427 -0
  39. package/dist/bollharness-integration/integration.js.map +1 -0
  40. package/dist/bollharness-integration/llm/pi-ai.d.ts +60 -0
  41. package/dist/bollharness-integration/llm/pi-ai.js +397 -0
  42. package/dist/bollharness-integration/pi-ecosystem-colony/index.d.ts +153 -0
  43. package/dist/bollharness-integration/pi-ecosystem-colony/index.js +365 -0
  44. package/dist/bollharness-integration/pi-ecosystem-goals/index.d.ts +136 -0
  45. package/dist/bollharness-integration/pi-ecosystem-goals/index.js +458 -0
  46. package/dist/bollharness-integration/pi-ecosystem-judgment/decision.d.ts +117 -0
  47. package/dist/bollharness-integration/pi-ecosystem-judgment/decision.js +300 -0
  48. package/dist/bollharness-integration/pi-ecosystem-judgment/distillation.d.ts +78 -0
  49. package/dist/bollharness-integration/pi-ecosystem-judgment/distillation.js +291 -0
  50. package/dist/bollharness-integration/pi-ecosystem-judgment/index.d.ts +139 -0
  51. package/dist/bollharness-integration/pi-ecosystem-judgment/index.js +445 -0
  52. package/dist/bollharness-integration/pi-ecosystem-mcp/index.d.ts +127 -0
  53. package/dist/bollharness-integration/pi-ecosystem-mcp/index.js +331 -0
  54. package/dist/bollharness-integration/pi-ecosystem-subagents/index.d.ts +95 -0
  55. package/dist/bollharness-integration/pi-ecosystem-subagents/index.js +303 -0
  56. package/dist/bollharness-integration/skill-adapter.js +564 -0
  57. package/dist/bollharness-integration/skill-adapter.js.map +1 -0
  58. package/dist/cli/interface.js +181 -0
  59. package/dist/constraint-runtime/src/_archive_helper.js +9 -0
  60. package/dist/constraint-runtime/src/agent/coordinator.js +48 -0
  61. package/dist/constraint-runtime/src/agent/index.js +1 -0
  62. package/dist/constraint-runtime/src/assistant/index.js +12 -0
  63. package/dist/constraint-runtime/src/bootstrap/index.js +12 -0
  64. package/dist/constraint-runtime/src/bootstrap_graph.js +13 -0
  65. package/dist/constraint-runtime/src/bridge/index.js +12 -0
  66. package/dist/constraint-runtime/src/buddy/index.js +12 -0
  67. package/dist/constraint-runtime/src/cli/index.js +12 -0
  68. package/dist/constraint-runtime/src/command_graph.js +10 -0
  69. package/dist/constraint-runtime/src/commands.js +50 -0
  70. package/dist/constraint-runtime/src/components/index.js +12 -0
  71. package/dist/constraint-runtime/src/constants/index.js +12 -0
  72. package/dist/constraint-runtime/src/constraint/budget.js +27 -0
  73. package/dist/constraint-runtime/src/constraint/budget.js.map +1 -0
  74. package/dist/constraint-runtime/src/constraint/index.js +2 -0
  75. package/dist/constraint-runtime/src/constraint/permission.js +25 -0
  76. package/dist/constraint-runtime/src/constraint/permission.js.map +1 -0
  77. package/dist/constraint-runtime/src/context.js +30 -0
  78. package/dist/constraint-runtime/src/coordinator/index.js +12 -0
  79. package/dist/constraint-runtime/src/cost_hook.js +4 -0
  80. package/dist/constraint-runtime/src/cost_tracker.js +8 -0
  81. package/dist/constraint-runtime/src/deferred_init.js +10 -0
  82. package/dist/constraint-runtime/src/direct_modes.js +6 -0
  83. package/dist/constraint-runtime/src/dynamic-tool-loader.js +85 -0
  84. package/dist/constraint-runtime/src/entrypoints/index.js +12 -0
  85. package/dist/constraint-runtime/src/execution_registry.js +44 -0
  86. package/dist/constraint-runtime/src/history.js +9 -0
  87. package/dist/constraint-runtime/src/hooks/index.js +12 -0
  88. package/dist/constraint-runtime/src/index.js +26 -0
  89. package/dist/constraint-runtime/src/ink.js +4 -0
  90. package/dist/constraint-runtime/src/keybindings/index.js +12 -0
  91. package/dist/constraint-runtime/src/memdir/index.js +12 -0
  92. package/dist/constraint-runtime/src/migrations/index.js +12 -0
  93. package/dist/constraint-runtime/src/models.js +3 -0
  94. package/dist/constraint-runtime/src/models.js.map +1 -0
  95. package/dist/constraint-runtime/src/moreright/index.js +12 -0
  96. package/dist/constraint-runtime/src/native_ts/index.js +12 -0
  97. package/dist/constraint-runtime/src/output_styles/index.js +12 -0
  98. package/dist/constraint-runtime/src/parity_audit.js +12 -0
  99. package/dist/constraint-runtime/src/plugins/index.js +12 -0
  100. package/dist/constraint-runtime/src/port_manifest.js +11 -0
  101. package/dist/constraint-runtime/src/prefetch.js +9 -0
  102. package/dist/constraint-runtime/src/query.js +1 -0
  103. package/dist/constraint-runtime/src/remote/index.js +12 -0
  104. package/dist/constraint-runtime/src/remote_runtime.js +9 -0
  105. package/dist/constraint-runtime/src/runtime/index.js +1 -0
  106. package/dist/constraint-runtime/src/runtime/session.js +35 -0
  107. package/dist/constraint-runtime/src/schemas/index.js +12 -0
  108. package/dist/constraint-runtime/src/screens/index.js +12 -0
  109. package/dist/constraint-runtime/src/server/index.js +12 -0
  110. package/dist/constraint-runtime/src/services/index.js +12 -0
  111. package/dist/constraint-runtime/src/session_store.js +22 -0
  112. package/dist/constraint-runtime/src/setup.js +30 -0
  113. package/dist/constraint-runtime/src/skills/index.js +1 -0
  114. package/dist/constraint-runtime/src/skills/skill-registry.js +28 -0
  115. package/dist/constraint-runtime/src/state/index.js +12 -0
  116. package/dist/constraint-runtime/src/system_init.js +20 -0
  117. package/dist/constraint-runtime/src/thinking/engine.js +42 -0
  118. package/dist/constraint-runtime/src/thinking/index.js +1 -0
  119. package/dist/constraint-runtime/src/tool_pool.js +8 -0
  120. package/dist/constraint-runtime/src/tools/OpenCLI/execAdapter.js +7 -0
  121. package/dist/constraint-runtime/src/tools/OpenCLI/listAdapters.js +7 -0
  122. package/dist/constraint-runtime/src/tools/OpenCLI/runCommand.js +7 -0
  123. package/dist/constraint-runtime/src/tools/PolymarketSDK/cancelOrder.js +6 -0
  124. package/dist/constraint-runtime/src/tools/PolymarketSDK/createOrder.js +6 -0
  125. package/dist/constraint-runtime/src/tools/PolymarketSDK/getMarket.js +5 -0
  126. package/dist/constraint-runtime/src/tools/PolymarketSDK/getOrders.js +6 -0
  127. package/dist/constraint-runtime/src/tools/PolymarketSDK/listMarkets.js +4 -0
  128. package/dist/constraint-runtime/src/tools/SafeSDK/confirmTransaction.js +6 -0
  129. package/dist/constraint-runtime/src/tools/SafeSDK/createTransaction.js +8 -0
  130. package/dist/constraint-runtime/src/tools/SafeSDK/deploySafe.js +6 -0
  131. package/dist/constraint-runtime/src/tools/SafeSDK/executeTransaction.js +6 -0
  132. package/dist/constraint-runtime/src/tools/SafeSDK/getBalance.js +6 -0
  133. package/dist/constraint-runtime/src/tools/SafeSDK/getPendingTransactions.js +6 -0
  134. package/dist/constraint-runtime/src/tools/SafeSDK/proposeTransaction.js +6 -0
  135. package/dist/constraint-runtime/src/tools/WalletTools/autoPay.js +29 -0
  136. package/dist/constraint-runtime/src/tools/WalletTools/createWallet.js +10 -0
  137. package/dist/constraint-runtime/src/tools/WalletTools/getBalance.js +13 -0
  138. package/dist/constraint-runtime/src/tools/WalletTools/importWallet.js +22 -0
  139. package/dist/constraint-runtime/src/tools/WalletTools/sendTransaction.js +25 -0
  140. package/dist/constraint-runtime/src/tools/WalletTools/signMessage.js +10 -0
  141. package/dist/constraint-runtime/src/tools/WalletTools/transferToken.js +25 -0
  142. package/dist/constraint-runtime/src/tools.js +70 -0
  143. package/dist/constraint-runtime/src/transcript.js +19 -0
  144. package/dist/constraint-runtime/src/types/index.js +12 -0
  145. package/dist/constraint-runtime/src/upstream_proxy/index.js +12 -0
  146. package/dist/constraint-runtime/src/utils/index.js +12 -0
  147. package/dist/constraint-runtime/src/vim/index.js +12 -0
  148. package/dist/constraint-runtime/src/voice/index.js +12 -0
  149. package/dist/constraint-runtime/tests/agent.test.js +16 -0
  150. package/dist/constraint-runtime/tests/constraint.test.js +41 -0
  151. package/dist/constraint-runtime/tests/skill.test.js +19 -0
  152. package/dist/constraint-runtime/tests/thinking.test.js +22 -0
  153. package/dist/constraints/commands.js +100 -0
  154. package/dist/constraints/index.js +11 -0
  155. package/dist/constraints/index.js.map +1 -0
  156. package/dist/constraints/permissions.js +37 -0
  157. package/dist/constraints/runtime.js +135 -0
  158. package/dist/constraints/session.js +48 -0
  159. package/dist/constraints/system-init.js +51 -0
  160. package/dist/constraints/tools.js +104 -0
  161. package/dist/documents/reader.js +104 -0
  162. package/dist/documents/reader.js.map +1 -0
  163. package/dist/electron-preload.js +15 -0
  164. package/dist/electron-preload.js.map +1 -0
  165. package/dist/electron.js +206 -0
  166. package/dist/electron.js.map +1 -0
  167. package/dist/index.js +1053 -0
  168. package/dist/llm/config-store.js +301 -0
  169. package/dist/llm/config-store.js.map +1 -0
  170. package/dist/llm/minimax-provider.js +46 -0
  171. package/dist/llm/minimax.js +45 -0
  172. package/dist/llm/pi-ai.js +471 -0
  173. package/dist/llm/pi-ai.js.map +1 -0
  174. package/dist/network/agent-network.js +692 -0
  175. package/dist/network/agent-network.js.map +1 -0
  176. package/dist/network/iroh-integration.js +117 -0
  177. package/dist/network/iroh-integration.js.map +1 -0
  178. package/dist/network/iroh-transport.js +506 -0
  179. package/dist/network/iroh-transport.js.map +1 -0
  180. package/dist/network/p2p.js +864 -0
  181. package/dist/network/p2p.js.map +1 -0
  182. package/dist/network/storage/adapters/json-adapter.js +401 -0
  183. package/dist/network/storage/adapters/json-adapter.js.map +1 -0
  184. package/dist/pi-ecosystem-colony/index.js +426 -0
  185. package/dist/pi-ecosystem-colony/index.js.map +1 -0
  186. package/dist/pi-ecosystem-goals/index.js +512 -0
  187. package/dist/pi-ecosystem-goals/index.js.map +1 -0
  188. package/dist/pi-ecosystem-judgment/decision.js +318 -0
  189. package/dist/pi-ecosystem-judgment/decision.js.map +1 -0
  190. package/dist/pi-ecosystem-judgment/distillation.js +304 -0
  191. package/dist/pi-ecosystem-judgment/distillation.js.map +1 -0
  192. package/dist/pi-ecosystem-judgment/human-value-store.js +463 -0
  193. package/dist/pi-ecosystem-judgment/human-value-store.js.map +1 -0
  194. package/dist/pi-ecosystem-judgment/index.js +581 -0
  195. package/dist/pi-ecosystem-judgment/index.js.map +1 -0
  196. package/dist/pi-ecosystem-judgment/value-injection.js +592 -0
  197. package/dist/pi-ecosystem-judgment/value-injection.js.map +1 -0
  198. package/dist/pi-ecosystem-mcp/index.js +385 -0
  199. package/dist/pi-ecosystem-mcp/index.js.map +1 -0
  200. package/dist/pi-ecosystem-subagents/index.js +352 -0
  201. package/dist/pi-ecosystem-subagents/index.js.map +1 -0
  202. package/dist/runtime/context/minimax-prompt.js +178 -0
  203. package/dist/runtime/context/sys-prompt.js +1 -0
  204. package/dist/social/ant-colony/AdaptiveHeartbeat.js +106 -0
  205. package/dist/social/ant-colony/AdaptiveHeartbeat.js.map +1 -0
  206. package/dist/social/ant-colony/PheromoneEngine.js +265 -0
  207. package/dist/social/ant-colony/PheromoneEngine.js.map +1 -0
  208. package/dist/social/ant-colony/types.js +28 -0
  209. package/dist/social/ant-colony/types.js.map +1 -0
  210. package/dist/social/channels/ChannelManager.js +444 -0
  211. package/dist/social/channels/ChannelManager.js.map +1 -0
  212. package/dist/social/channels/DiapChannelBridge.js +451 -0
  213. package/dist/social/channels/DiapChannelBridge.js.map +1 -0
  214. package/dist/social/channels/InterestMatcher.js +135 -0
  215. package/dist/social/channels/InterestMatcher.js.map +1 -0
  216. package/dist/social/channels/types.js +21 -0
  217. package/dist/social/channels/types.js.map +1 -0
  218. package/dist/social/global-shared-context.js +368 -0
  219. package/dist/social/global-shared-context.js.map +1 -0
  220. package/dist/social/heartbeat.js +865 -0
  221. package/dist/social/heartbeat.js.map +1 -0
  222. package/dist/test/constraint-layer.test.js +164 -0
  223. package/dist/test/global-shared-context.test.js +315 -0
  224. package/dist/test/pi-sdk.test.js +47 -0
  225. package/dist/test/set-persona.test.js +38 -0
  226. package/dist/test/subagent-manager.test.js +276 -0
  227. package/dist/test/workflow-engine.test.js +87 -0
  228. package/dist/web/api-config.html +265 -0
  229. package/dist/web/client.js +1401 -0
  230. package/dist/web/components/p2p/index.js +357 -0
  231. package/dist/web/components/p2p/p2p-connection.js +161 -0
  232. package/dist/web/components/p2p/p2p-identity.js +78 -0
  233. package/dist/web/components/p2p/p2p-manager.js +82 -0
  234. package/dist/web/components/p2p/p2p-messages.js +186 -0
  235. package/dist/web/components/p2p/p2p-store-memory.js +129 -0
  236. package/dist/web/components/p2p/types.js +23 -0
  237. package/dist/web/index.html +192 -0
  238. package/dist/web/server.js +1647 -0
  239. package/dist/web/server.js.map +1 -0
  240. package/dist/web/style.css +3260 -0
  241. package/dist/workflows/collaboration.js +374 -0
  242. package/dist/workflows/index.js +54 -0
  243. package/docs/agent-communication.md +333 -0
  244. package/docs/plans/2026-05-15-document-agent-design.md +479 -0
  245. package/docs/plans/2026-05-15-document-agent-implementation-plan.md +792 -0
  246. package/docs/plans/2026-05-16-chat-ui-design.md +86 -0
  247. package/docs/plans/2026-05-16-constraint-runtime-design.md +106 -0
  248. package/docs/plans/2026-05-16-constraint-runtime-implementation.md +441 -0
  249. package/docs//346/225/260/345/255/246/350/276/205/345/212/251/346/231/272/350/203/275/344/275/223-/346/240/270/345/277/203/346/225/210/346/236/234/345/256/232/344/271/211.md +287 -0
  250. package/package.json +121 -0
  251. package/scripts/build-web.ts +61 -0
  252. package/src/agents/constraint-layer.ts +309 -0
  253. package/src/agents/pi-sdk.ts +1591 -0
  254. package/src/agents/protocol.ts +304 -0
  255. package/src/agents/subagent-manager.ts +553 -0
  256. package/src/agents/workflow-engine.ts +332 -0
  257. package/src/bollharness/.boll/CLAUDE.md.template +34 -0
  258. package/src/bollharness/.boll/MANIFEST.yaml +213 -0
  259. package/src/bollharness/.boll/active-review-agents/.gitkeep +0 -0
  260. package/src/bollharness/.boll/agents/review-base.yaml +108 -0
  261. package/src/bollharness/.boll/deploy-allowlist.yaml +38 -0
  262. package/src/bollharness/.boll/inbox/schema/message-v1.json +99 -0
  263. package/src/bollharness/.boll/install-staging/.gitkeep +0 -0
  264. package/src/bollharness/.boll/issue-adapter.yaml +31 -0
  265. package/src/bollharness/.boll/plugins/boll-mode-toolkit/contracts/mode-contract.md +85 -0
  266. package/src/bollharness/.boll/plugins/boll-review-toolkit/contracts/evidence-packet-schema.json +102 -0
  267. package/src/bollharness/.boll/plugins/boll-review-toolkit/contracts/review-contract.yaml +247 -0
  268. package/src/bollharness/.boll/rules/backend-routes.md +31 -0
  269. package/src/bollharness/.boll/rules/closure-semantics.md +30 -0
  270. package/src/bollharness/.boll/rules/env-vars.md +32 -0
  271. package/src/bollharness/.boll/rules/hanis-protocol.md +145 -0
  272. package/src/bollharness/.boll/rules/repo-structure.md +42 -0
  273. package/src/bollharness/.boll/rules/review-agent-isolation.md +73 -0
  274. package/src/bollharness/.boll/rules/source-of-truth.md +33 -0
  275. package/src/bollharness/.boll/settings.json +180 -0
  276. package/src/bollharness/.boll/settings.json.template +31 -0
  277. package/src/bollharness/.boll/skills/arch/SKILL.md +372 -0
  278. package/src/bollharness/.boll/skills/bug-pipeline/SKILL.md +168 -0
  279. package/src/bollharness/.boll/skills/bug-triage/SKILL.md +161 -0
  280. package/src/bollharness/.boll/skills/context-chains/SKILL.md +250 -0
  281. package/src/bollharness/.boll/skills/context-chains/context-chain-index.md +48 -0
  282. package/src/bollharness/.boll/skills/context-chains/work-type-extractors/code_change_extractor.ts +142 -0
  283. package/src/bollharness/.boll/skills/context-chains/work-type-extractors/debugging_extractor.ts +126 -0
  284. package/src/bollharness/.boll/skills/context-chains/work-type-extractors/design_extractor.ts +148 -0
  285. package/src/bollharness/.boll/skills/context-chains/work-type-extractors/planning_extractor.ts +162 -0
  286. package/src/bollharness/.boll/skills/context-chains/work-type-extractors/question_extractor.ts +116 -0
  287. package/src/bollharness/.boll/skills/context-chains/work-type-extractors/review_extractor.ts +136 -0
  288. package/src/bollharness/.boll/skills/crystal-learn/SKILL.md +93 -0
  289. package/src/bollharness/.boll/skills/crystal-learn/invariants/INV-0.md +34 -0
  290. package/src/bollharness/.boll/skills/crystal-learn/invariants/INV-1.md +34 -0
  291. package/src/bollharness/.boll/skills/crystal-learn/invariants/INV-2.md +35 -0
  292. package/src/bollharness/.boll/skills/crystal-learn/invariants/INV-3.md +34 -0
  293. package/src/bollharness/.boll/skills/crystal-learn/invariants/INV-4.md +43 -0
  294. package/src/bollharness/.boll/skills/crystal-learn/invariants/INV-5.md +34 -0
  295. package/src/bollharness/.boll/skills/crystal-learn/invariants/INV-6.md +37 -0
  296. package/src/bollharness/.boll/skills/crystal-learn/invariants/INV-7.md +46 -0
  297. package/src/bollharness/.boll/skills/guardian-fixer/PROMPT.md +415 -0
  298. package/src/bollharness/.boll/skills/guardian-fixer/SKILL.md +320 -0
  299. package/src/bollharness/.boll/skills/harness-dev/SKILL.md +93 -0
  300. package/src/bollharness/.boll/skills/harness-dev/examples/README.md +227 -0
  301. package/src/bollharness/.boll/skills/harness-dev-handoff/SKILL.md +165 -0
  302. package/src/bollharness/.boll/skills/harness-eng/SKILL.md +110 -0
  303. package/src/bollharness/.boll/skills/harness-eng-test/SKILL.md +79 -0
  304. package/src/bollharness/.boll/skills/harness-lab/SKILL.md +170 -0
  305. package/src/bollharness/.boll/skills/harness-ops/SKILL.md +57 -0
  306. package/src/bollharness/.boll/skills/harness-voice/SKILL.md +183 -0
  307. package/src/bollharness/.boll/skills/judgment/SKILL.md +115 -0
  308. package/src/bollharness/.boll/skills/lead/SKILL.md +245 -0
  309. package/src/bollharness/.boll/skills/lead/install-wow-harness.md +77 -0
  310. package/src/bollharness/.boll/skills/lead/ref-review-sop.md +91 -0
  311. package/src/bollharness/.boll/skills/lead/ref-stages.md +129 -0
  312. package/src/bollharness/.boll/skills/skill-discovery/SKILL.md +169 -0
  313. package/src/bollharness/.boll/skills/task-arch/SKILL.md +106 -0
  314. package/src/bollharness/.boll/skills/toolkit/SKILL.md +57 -0
  315. package/src/bollharness/.boll/tasks/.gitkeep +0 -0
  316. package/src/bollharness/.boll/toolkit-index.yaml +112 -0
  317. package/src/bollharness/.claude/agents/review-base.yaml +108 -0
  318. package/src/bollharness/.claude/plugins/boll-mode-toolkit/.claude-plugin/plugin.json +44 -0
  319. package/src/bollharness/.claude/plugins/boll-review-toolkit/.claude-plugin/plugin.json +24 -0
  320. package/src/bollharness/.claude/plugins/boll-review-toolkit/contracts/evidence-packet-schema.json +102 -0
  321. package/src/bollharness/.claude/plugins/boll-review-toolkit/contracts/review-contract.yaml +247 -0
  322. package/src/bollharness/.claude/settings.json +157 -0
  323. package/src/bollharness/.claude/skills/arch/SKILL.md +64 -0
  324. package/src/bollharness/.claude/skills/crystal-learn/SKILL.md +93 -0
  325. package/src/bollharness/.claude/skills/guardian-fixer/PROMPT.md +44 -0
  326. package/src/bollharness/.claude/skills/guardian-fixer/SKILL.md +324 -0
  327. package/src/bollharness/.claude/skills/harness-dev/SKILL.md +93 -0
  328. package/src/bollharness/.claude/skills/harness-dev/examples/README.md +17 -0
  329. package/src/bollharness/.claude/skills/harness-dev-handoff/SKILL.md +165 -0
  330. package/src/bollharness/.claude/skills/harness-eng/SKILL.md +183 -0
  331. package/src/bollharness/.claude/skills/harness-eng-test/SKILL.md +57 -0
  332. package/src/bollharness/.claude/skills/harness-ops/SKILL.md +57 -0
  333. package/src/bollharness/.claude/skills/harness-voice/SKILL.md +84 -0
  334. package/src/bollharness/.claude/skills/lead/INDEX.md +28 -0
  335. package/src/bollharness/.claude/skills/lead/SKILL.md +24 -0
  336. package/src/bollharness/.claude/skills/lead/install-wow-harness.md +77 -0
  337. package/src/bollharness/.claude/skills/lead/ref-review-sop.md +48 -0
  338. package/src/bollharness/.claude/skills/lead/ref-stages.md +58 -0
  339. package/src/bollharness/.claude/skills/plan-lock/SKILL.md +74 -0
  340. package/src/bollharness/.claude/skills/skill-discovery/SKILL.md +120 -0
  341. package/src/bollharness/.claude/skills/task-arch/SKILL.md +106 -0
  342. package/src/bollharness/.claude/skills/toolkit/SKILL.md +57 -0
  343. package/src/bollharness/.claude/skills/toolkit/list.sh +92 -0
  344. package/src/bollharness/.githooks/pre-commit +21 -0
  345. package/src/bollharness/.github/workflows/ci.yml +88 -0
  346. package/src/bollharness/.sanitize-report.json +13 -0
  347. package/src/bollharness/CLAUDE.md +73 -0
  348. package/src/bollharness/LICENSE +21 -0
  349. package/src/bollharness/README.md +143 -0
  350. package/src/bollharness/README.zh-CN.md +131 -0
  351. package/src/bollharness/docs/decisions/ADR-030-guard-signal-protocol-and-governance-reload.md +1076 -0
  352. package/src/bollharness/docs/decisions/ADR-038-harness-optimization-strategy.md +2039 -0
  353. package/src/bollharness/docs/decisions/ADR-041-codex-claude-code-division-of-labor.md +128 -0
  354. package/src/bollharness/docs/decisions/ADR-H1-crystal-learn-revival.md +188 -0
  355. package/src/bollharness/docs/decisions/ADR-H2-identity-isolation.md +183 -0
  356. package/src/bollharness/docs/decisions/ADR-H3-memory-scope.md +133 -0
  357. package/src/bollharness/docs/decisions/ADR-H4-prompt-governance.md +146 -0
  358. package/src/bollharness/docs/decisions/ADR-H5-gate-quantization.md +212 -0
  359. package/src/bollharness/docs/decisions/ADR-H6-state-file-health.md +211 -0
  360. package/src/bollharness/docs/decisions/ADR-H8-issue-and-doc-compliance.md +202 -0
  361. package/src/bollharness/docs/decisions/ADR-H9-mailbox.md +231 -0
  362. package/src/bollharness/docs/decisions/PLAN-H1-crystal-learn-revival.md +270 -0
  363. package/src/bollharness/docs/decisions/PLAN-H2-identity-isolation.md +291 -0
  364. package/src/bollharness/docs/decisions/PLAN-H3-memory-scope.md +228 -0
  365. package/src/bollharness/docs/decisions/PLAN-H4-prompt-governance.md +227 -0
  366. package/src/bollharness/docs/decisions/PLAN-H5-gate-quantization.md +239 -0
  367. package/src/bollharness/docs/decisions/PLAN-H6-state-file-health.md +325 -0
  368. package/src/bollharness/docs/decisions/PLAN-H8-issue-and-doc-compliance.md +242 -0
  369. package/src/bollharness/docs/decisions/PLAN-H9-mailbox.md +378 -0
  370. package/src/bollharness/docs/launch-article-en.md +276 -0
  371. package/src/bollharness/docs/launch-article-zh.md +305 -0
  372. package/src/bollharness/docs/practice.html +356 -0
  373. package/src/bollharness/docs/practice.md +82 -0
  374. package/src/bollharness/docs/research/round-1/README.md +11 -0
  375. package/src/bollharness/docs/research/round-2/README.md +11 -0
  376. package/src/bollharness/docs/research/round-3/README.md +11 -0
  377. package/src/bollharness/docs/research/round-4/README.md +11 -0
  378. package/src/bollharness/docs/research/round-5/README.md +11 -0
  379. package/src/bollharness/docs/research/round-6/README.md +11 -0
  380. package/src/bollharness/package-lock.json +48 -0
  381. package/src/bollharness/package.json +20 -0
  382. package/src/bollharness/reference/SOURCE-COMMIT.txt +3 -0
  383. package/src/bollharness/reference/boll-reference/.claude/rules/backend-routes.md +268 -0
  384. package/src/bollharness/reference/boll-reference/.claude/rules/bridge.md +20 -0
  385. package/src/bollharness/reference/boll-reference/.claude/rules/closure-semantics.md +30 -0
  386. package/src/bollharness/reference/boll-reference/.claude/rules/coaching.md +13 -0
  387. package/src/bollharness/reference/boll-reference/.claude/rules/env-vars.md +50 -0
  388. package/src/bollharness/reference/boll-reference/.claude/rules/hackathon.md +12 -0
  389. package/src/bollharness/reference/boll-reference/.claude/rules/repo-structure.md +184 -0
  390. package/src/bollharness/reference/boll-reference/.claude/rules/review-agent-isolation.md +112 -0
  391. package/src/bollharness/reference/boll-reference/.claude/rules/scenes.md +12 -0
  392. package/src/bollharness/reference/boll-reference/.claude/skills/arch/SKILL.md +551 -0
  393. package/src/bollharness/reference/boll-reference/.claude/skills/boll-animation/SKILL.md +26 -0
  394. package/src/bollharness/reference/boll-reference/.claude/skills/boll-bridge/SKILL.md +227 -0
  395. package/src/bollharness/reference/boll-reference/.claude/skills/boll-bridge/agents/openai.yaml +4 -0
  396. package/src/bollharness/reference/boll-reference/.claude/skills/boll-bridge/references/bridge-failure-taxonomy.md +142 -0
  397. package/src/bollharness/reference/boll-reference/.claude/skills/boll-bridge/references/bridge-validation-ladder.md +107 -0
  398. package/src/bollharness/reference/boll-reference/.claude/skills/boll-crystal/SKILL.md +893 -0
  399. package/src/bollharness/reference/boll-reference/.claude/skills/boll-crystal-learn/SKILL.md +89 -0
  400. package/src/bollharness/reference/boll-reference/.claude/skills/boll-dev/SKILL.md +93 -0
  401. package/src/bollharness/reference/boll-reference/.claude/skills/boll-dev/examples/README.md +209 -0
  402. package/src/bollharness/reference/boll-reference/.claude/skills/boll-dev-handoff/SKILL.md +165 -0
  403. package/src/bollharness/reference/boll-reference/.claude/skills/boll-eng/SKILL.md +110 -0
  404. package/src/bollharness/reference/boll-reference/.claude/skills/boll-eng-frontend/SKILL.md +203 -0
  405. package/src/bollharness/reference/boll-reference/.claude/skills/boll-eng-hdc/SKILL.md +27 -0
  406. package/src/bollharness/reference/boll-reference/.claude/skills/boll-eng-orchestrator/SKILL.md +28 -0
  407. package/src/bollharness/reference/boll-reference/.claude/skills/boll-eng-prompt/SKILL.md +27 -0
  408. package/src/bollharness/reference/boll-reference/.claude/skills/boll-eng-test/SKILL.md +79 -0
  409. package/src/bollharness/reference/boll-reference/.claude/skills/boll-lab/SKILL.md +372 -0
  410. package/src/bollharness/reference/boll-reference/.claude/skills/boll-run/SKILL.md +437 -0
  411. package/src/bollharness/reference/boll-reference/.claude/skills/boll-ux-appstore/SKILL.md +27 -0
  412. package/src/bollharness/reference/boll-reference/.claude/skills/boll-voice/SKILL.md +442 -0
  413. package/src/bollharness/reference/boll-reference/.claude/skills/guardian-fixer/PROMPT.md +421 -0
  414. package/src/bollharness/reference/boll-reference/.claude/skills/guardian-fixer/SKILL.md +326 -0
  415. package/src/bollharness/reference/boll-reference/.claude/skills/lead/SKILL.md +155 -0
  416. package/src/bollharness/reference/boll-reference/.claude/skills/lead/ref-review-sop.md +91 -0
  417. package/src/bollharness/reference/boll-reference/.claude/skills/lead/ref-stages.md +129 -0
  418. package/src/bollharness/reference/boll-reference/.claude/skills/nature-designer/output/skill-map-preview.png +0 -0
  419. package/src/bollharness/reference/boll-reference/.claude/skills/nature-designer/output/skill-map-v2.png +0 -0
  420. package/src/bollharness/reference/boll-reference/.claude/skills/nature-designer/output/skill-map-v3.png +0 -0
  421. package/src/bollharness/reference/boll-reference/.claude/skills/nature-designer/output/skill-map-v4.png +0 -0
  422. package/src/bollharness/reference/boll-reference/.claude/skills/plan-lock/SKILL.md +425 -0
  423. package/src/bollharness/reference/boll-reference/.claude/skills/plan-lock/ref-three-checks.md +62 -0
  424. package/src/bollharness/reference/boll-reference/.claude/skills/plan-lock/ref-wp-templates.md +78 -0
  425. package/src/bollharness/reference/boll-reference/.claude/skills/task-arch/SKILL.md +76 -0
  426. package/src/bollharness/reference/boll-reference/.claude/skills/vibedevteam-graph/SKILL.md +57 -0
  427. package/src/bollharness/reference/boll-reference/.claude/skills/vibedevteam-graph/beads-graph.sh +153 -0
  428. package/src/bollharness/reference/boll-reference/.claude/skills/vibedevteam-init/SKILL.md +52 -0
  429. package/src/bollharness/reference/boll-reference/.claude/skills/vibedevteam-init/beads-auto-link.sh +76 -0
  430. package/src/bollharness/reference/boll-reference/.claude/skills/vibedevteam-sync/SKILL.md +50 -0
  431. package/src/bollharness/reference/boll-reference/.claude/skills/vibedevteam-sync/beads-sync-proj.sh +108 -0
  432. package/src/bollharness/reference/boll-reference/docs/architecture/AGENT-PROFILE.md +151 -0
  433. package/src/bollharness/reference/boll-reference/docs/architecture/COST-STRUCTURE.md +56 -0
  434. package/src/bollharness/reference/boll-reference/docs/architecture/INDEX.md +76 -0
  435. package/src/bollharness/reference/boll-reference/docs/architecture/MODULE1-INTENT-FIELD.md +116 -0
  436. package/src/bollharness/reference/boll-reference/docs/architecture/MODULE2-CRYSTALLIZATION.md +200 -0
  437. package/src/bollharness/reference/boll-reference/docs/architecture/PRINCIPLES.md +84 -0
  438. package/src/bollharness/reference/boll-reference/docs/architecture/PROTOCOL-CORE.md +209 -0
  439. package/src/bollharness/reference/boll-reference/docs/architecture/VISION.md +181 -0
  440. package/src/bollharness/reference/boll-reference/docs/architecture/discussions/D-01-MARKET-SCENE-PROTOCOL.md +754 -0
  441. package/src/bollharness/reference/boll-reference/scripts/hooks/.sanitize-report.json +12 -0
  442. package/src/bollharness/reference/boll-reference/scripts/hooks/find-boll-root.sh +27 -0
  443. package/src/bollharness/reference/boll-reference/scripts/hooks/precompact.sh +57 -0
  444. package/src/bollharness/reference/boll-reference/scripts/hooks/stop-evaluator.md +57 -0
  445. package/src/bollharness/schemas/metrics-jsonl-allowlist.json +67 -0
  446. package/src/bollharness/scripts/checks/next_decision_number.sh +48 -0
  447. package/src/bollharness/scripts/ci/count-components.sh +65 -0
  448. package/src/bollharness/scripts/context-fragments/artifact-linkage.md +14 -0
  449. package/src/bollharness/scripts/context-fragments/auth-consumers.md +17 -0
  450. package/src/bollharness/scripts/context-fragments/bridge-constitution.md +13 -0
  451. package/src/bollharness/scripts/context-fragments/catalyst-distributed.md +18 -0
  452. package/src/bollharness/scripts/context-fragments/closure-checklist.md +13 -0
  453. package/src/bollharness/scripts/context-fragments/contract-consumers.md +15 -0
  454. package/src/bollharness/scripts/context-fragments/db-shared-structures.md +15 -0
  455. package/src/bollharness/scripts/context-fragments/fixed-three-layers.md +19 -0
  456. package/src/bollharness/scripts/context-fragments/general-dev-principles.md +11 -0
  457. package/src/bollharness/scripts/context-fragments/issue-first.md +8 -0
  458. package/src/bollharness/scripts/context-fragments/mcp-parity.md +16 -0
  459. package/src/bollharness/scripts/context-fragments/pi-agent-operations.md +74 -0
  460. package/src/bollharness/scripts/context-fragments/protocol-consumers.md +15 -0
  461. package/src/bollharness/scripts/context-fragments/run-events-consumers.md +15 -0
  462. package/src/bollharness/scripts/context-fragments/scene-fidelity.md +13 -0
  463. package/src/bollharness/scripts/context-fragments/truth-source-hierarchy.md +15 -0
  464. package/src/bollharness/scripts/context-fragments/two-language.md +15 -0
  465. package/src/bollharness/scripts/context-fragments/version-sources.md +14 -0
  466. package/src/bollharness/scripts/hooks/find-project-root.sh +47 -0
  467. package/src/bollharness/scripts/hooks/inbox-poll.sh +78 -0
  468. package/src/bollharness/scripts/hooks/precompact.sh +56 -0
  469. package/src/bollharness/scripts/hooks/stop-evaluator.md +83 -0
  470. package/src/bollharness/scripts/sync-from-upstream.sh +281 -0
  471. package/src/bollharness/src/index.ts +5 -0
  472. package/src/bollharness/src/scripts/checks/check_adr_plan_numbering.ts +11 -0
  473. package/src/bollharness/src/scripts/checks/check_api_types.ts +52 -0
  474. package/src/bollharness/src/scripts/checks/check_artifact_link.ts +156 -0
  475. package/src/bollharness/src/scripts/checks/check_bridge_deps.ts +11 -0
  476. package/src/bollharness/src/scripts/checks/check_bugfix_binding.ts +11 -0
  477. package/src/bollharness/src/scripts/checks/check_bugfix_binding_ci.ts +11 -0
  478. package/src/bollharness/src/scripts/checks/check_doc_file_references.ts +11 -0
  479. package/src/bollharness/src/scripts/checks/check_doc_freshness.ts +141 -0
  480. package/src/bollharness/src/scripts/checks/check_doc_links.ts +36 -0
  481. package/src/bollharness/src/scripts/checks/check_file_existence_claims.ts +11 -0
  482. package/src/bollharness/src/scripts/checks/check_fragment_integrity.ts +40 -0
  483. package/src/bollharness/src/scripts/checks/check_hook_installed.ts +70 -0
  484. package/src/bollharness/src/scripts/checks/check_issue_closure.ts +51 -0
  485. package/src/bollharness/src/scripts/checks/check_mcp_parity.ts +11 -0
  486. package/src/bollharness/src/scripts/checks/check_security.ts +54 -0
  487. package/src/bollharness/src/scripts/checks/check_skill_parity.ts +11 -0
  488. package/src/bollharness/src/scripts/checks/check_versions.ts +11 -0
  489. package/src/bollharness/src/scripts/checks/finding.ts +35 -0
  490. package/src/bollharness/src/scripts/checks/next_decision_number.ts +24 -0
  491. package/src/bollharness/src/scripts/checks/regenerate_magic_docs.ts +11 -0
  492. package/src/bollharness/src/scripts/ci/detect_rebaseline_triggers.ts +14 -0
  493. package/src/bollharness/src/scripts/ci/scan_subprocess_cfg.ts +14 -0
  494. package/src/bollharness/src/scripts/ci/scan_verify_artifacts.ts +14 -0
  495. package/src/bollharness/src/scripts/ci/scan_yaml_schema.ts +14 -0
  496. package/src/bollharness/src/scripts/context_router.ts +76 -0
  497. package/src/bollharness/src/scripts/deploy-guard.ts +182 -0
  498. package/src/bollharness/src/scripts/guard-feedback.ts +215 -0
  499. package/src/bollharness/src/scripts/guard_router.ts +194 -0
  500. package/src/bollharness/src/scripts/hooks/_hook_output.js +3 -0
  501. package/src/bollharness/src/scripts/hooks/_hook_output.ts +11 -0
  502. package/src/bollharness/src/scripts/hooks/auto-python3.ts +10 -0
  503. package/src/bollharness/src/scripts/hooks/deploy-progress-on-session-end.ts +10 -0
  504. package/src/bollharness/src/scripts/hooks/failure-analyzer.ts +10 -0
  505. package/src/bollharness/src/scripts/hooks/gate-judgment-inject.ts +111 -0
  506. package/src/bollharness/src/scripts/hooks/gate-transition-judgment.ts +74 -0
  507. package/src/bollharness/src/scripts/hooks/inbox-ack.ts +10 -0
  508. package/src/bollharness/src/scripts/hooks/inbox-inject-on-start.ts +10 -0
  509. package/src/bollharness/src/scripts/hooks/inbox-validate.ts +10 -0
  510. package/src/bollharness/src/scripts/hooks/inbox-write-ledger.ts +10 -0
  511. package/src/bollharness/src/scripts/hooks/initializer-agent.ts +10 -0
  512. package/src/bollharness/src/scripts/hooks/loop-detection.ts +83 -0
  513. package/src/bollharness/src/scripts/hooks/owner-guard.ts +10 -0
  514. package/src/bollharness/src/scripts/hooks/precompact.ts +10 -0
  515. package/src/bollharness/src/scripts/hooks/review-agent-gatekeeper.ts +10 -0
  516. package/src/bollharness/src/scripts/hooks/risk-tracker.ts +121 -0
  517. package/src/bollharness/src/scripts/hooks/sanitize-on-read.ts +10 -0
  518. package/src/bollharness/src/scripts/hooks/session-reflection.ts +12 -0
  519. package/src/bollharness/src/scripts/hooks/session-start-magic-docs.ts +12 -0
  520. package/src/bollharness/src/scripts/hooks/session-start-reset-risk.ts +12 -0
  521. package/src/bollharness/src/scripts/hooks/session-start-toolkit-reminder.ts +12 -0
  522. package/src/bollharness/src/scripts/hooks/stop-evaluator.ts +164 -0
  523. package/src/bollharness/src/scripts/hooks/tool-call-counter.ts +10 -0
  524. package/src/bollharness/src/scripts/hooks/trace-analyzer.ts +14 -0
  525. package/src/bollharness/src/scripts/install/install-trust-token.ts +13 -0
  526. package/src/bollharness/src/scripts/install/multi_project_registry.ts +13 -0
  527. package/src/bollharness/src/scripts/install/phase2_auto.ts +28 -0
  528. package/src/bollharness/src/scripts/install/pre_commit_installer.ts +10 -0
  529. package/src/bollharness/src/scripts/install/tier_selector.ts +10 -0
  530. package/src/bollharness/src/scripts/install/transcript_miner.ts +13 -0
  531. package/src/bollharness/src/scripts/lib/claim_patterns.ts +11 -0
  532. package/src/bollharness/src/scripts/lib/sanitize_patterns.ts +13 -0
  533. package/src/bollharness/src/scripts/sanitize.ts +9 -0
  534. package/src/bollharness/templates/persona/default.json +19 -0
  535. package/src/bollharness/templates/scaffold/.boll/guard/.gitkeep +0 -0
  536. package/src/bollharness/templates/scaffold/.boll/metrics/.gitkeep +0 -0
  537. package/src/bollharness/templates/scaffold/.boll/state/.gitkeep +0 -0
  538. package/src/bollharness/templates/scaffold/.gitignore.append +16 -0
  539. package/src/bollharness/templates/scaffold/CLAUDE.md +89 -0
  540. package/src/bollharness/templates/scaffold/docs/INDEX.md +3 -0
  541. package/src/bollharness/templates/scaffold/docs/decisions/ADR_TEMPLATE.md +38 -0
  542. package/src/bollharness/templates/scaffold/docs/decisions/PLAN_TEMPLATE.md +45 -0
  543. package/src/bollharness/templates/scaffold/docs/decisions/tasks/.gitkeep +2 -0
  544. package/src/bollharness/templates/scaffold/docs/issues/.gitkeep +0 -0
  545. package/src/bollharness/templates/scaffold/docs/issues/GUARD_ISSUE_TEMPLATE.md +35 -0
  546. package/src/bollharness/templates/scaffold/docs/issues/ISSUE_TEMPLATE.md +51 -0
  547. package/src/bollharness/tsconfig.json +26 -0
  548. package/src/bollharness-integration/channel-judgment-engine.ts +634 -0
  549. package/src/bollharness-integration/context-chain-router.ts +474 -0
  550. package/src/bollharness-integration/context-router-judgment.ts +339 -0
  551. package/src/bollharness-integration/context-router.ts +583 -0
  552. package/src/bollharness-integration/gate-state-machine.ts +444 -0
  553. package/src/bollharness-integration/gate-transition-hooks.ts +137 -0
  554. package/src/bollharness-integration/guard-checker.ts +451 -0
  555. package/src/bollharness-integration/index.ts +219 -0
  556. package/src/bollharness-integration/integration.ts +538 -0
  557. package/src/bollharness-integration/judgment-prompts.yaml +535 -0
  558. package/src/bollharness-integration/llm-judgment-engine.ts +712 -0
  559. package/src/bollharness-integration/skill-adapter.ts +646 -0
  560. package/src/cli/interface.ts +211 -0
  561. package/src/constraint-runtime/package-lock.json +48 -0
  562. package/src/constraint-runtime/package.json +34 -0
  563. package/src/constraint-runtime/src/_archive_helper.ts +16 -0
  564. package/src/constraint-runtime/src/agent/coordinator.ts +71 -0
  565. package/src/constraint-runtime/src/agent/index.ts +1 -0
  566. package/src/constraint-runtime/src/assistant/index.ts +15 -0
  567. package/src/constraint-runtime/src/bootstrap/index.ts +15 -0
  568. package/src/constraint-runtime/src/bootstrap_graph.ts +17 -0
  569. package/src/constraint-runtime/src/bridge/index.ts +15 -0
  570. package/src/constraint-runtime/src/buddy/index.ts +15 -0
  571. package/src/constraint-runtime/src/cli/index.ts +15 -0
  572. package/src/constraint-runtime/src/command_graph.ts +20 -0
  573. package/src/constraint-runtime/src/commands.ts +83 -0
  574. package/src/constraint-runtime/src/components/index.ts +15 -0
  575. package/src/constraint-runtime/src/constants/index.ts +15 -0
  576. package/src/constraint-runtime/src/constraint/budget.ts +25 -0
  577. package/src/constraint-runtime/src/constraint/index.ts +3 -0
  578. package/src/constraint-runtime/src/constraint/permission.ts +28 -0
  579. package/src/constraint-runtime/src/context.ts +45 -0
  580. package/src/constraint-runtime/src/coordinator/index.ts +15 -0
  581. package/src/constraint-runtime/src/cost_hook.ts +6 -0
  582. package/src/constraint-runtime/src/cost_tracker.ts +9 -0
  583. package/src/constraint-runtime/src/deferred_init.ts +18 -0
  584. package/src/constraint-runtime/src/direct_modes.ts +13 -0
  585. package/src/constraint-runtime/src/dynamic-tool-loader.ts +115 -0
  586. package/src/constraint-runtime/src/entrypoints/index.ts +15 -0
  587. package/src/constraint-runtime/src/execution_registry.ts +41 -0
  588. package/src/constraint-runtime/src/history.ts +16 -0
  589. package/src/constraint-runtime/src/hooks/index.ts +15 -0
  590. package/src/constraint-runtime/src/index.ts +28 -0
  591. package/src/constraint-runtime/src/ink.ts +4 -0
  592. package/src/constraint-runtime/src/keybindings/index.ts +15 -0
  593. package/src/constraint-runtime/src/memdir/index.ts +15 -0
  594. package/src/constraint-runtime/src/migrations/index.ts +15 -0
  595. package/src/constraint-runtime/src/models.ts +49 -0
  596. package/src/constraint-runtime/src/moreright/index.ts +15 -0
  597. package/src/constraint-runtime/src/native_ts/index.ts +15 -0
  598. package/src/constraint-runtime/src/output_styles/index.ts +15 -0
  599. package/src/constraint-runtime/src/parity_audit.ts +23 -0
  600. package/src/constraint-runtime/src/plugins/index.ts +15 -0
  601. package/src/constraint-runtime/src/port_manifest.ts +20 -0
  602. package/src/constraint-runtime/src/prefetch.ts +17 -0
  603. package/src/constraint-runtime/src/query.ts +7 -0
  604. package/src/constraint-runtime/src/reference_data/archive_surface_snapshot.json +63 -0
  605. package/src/constraint-runtime/src/reference_data/commands_snapshot.json +1037 -0
  606. package/src/constraint-runtime/src/reference_data/subsystems/OpenCLI.json +10 -0
  607. package/src/constraint-runtime/src/reference_data/subsystems/PolymarketSDK.json +12 -0
  608. package/src/constraint-runtime/src/reference_data/subsystems/SafeSDK.json +14 -0
  609. package/src/constraint-runtime/src/reference_data/subsystems/assistant.json +8 -0
  610. package/src/constraint-runtime/src/reference_data/subsystems/bootstrap.json +8 -0
  611. package/src/constraint-runtime/src/reference_data/subsystems/bridge.json +32 -0
  612. package/src/constraint-runtime/src/reference_data/subsystems/buddy.json +13 -0
  613. package/src/constraint-runtime/src/reference_data/subsystems/cli.json +26 -0
  614. package/src/constraint-runtime/src/reference_data/subsystems/components.json +32 -0
  615. package/src/constraint-runtime/src/reference_data/subsystems/constants.json +28 -0
  616. package/src/constraint-runtime/src/reference_data/subsystems/coordinator.json +8 -0
  617. package/src/constraint-runtime/src/reference_data/subsystems/entrypoints.json +15 -0
  618. package/src/constraint-runtime/src/reference_data/subsystems/hooks.json +32 -0
  619. package/src/constraint-runtime/src/reference_data/subsystems/keybindings.json +21 -0
  620. package/src/constraint-runtime/src/reference_data/subsystems/memdir.json +15 -0
  621. package/src/constraint-runtime/src/reference_data/subsystems/migrations.json +18 -0
  622. package/src/constraint-runtime/src/reference_data/subsystems/moreright.json +8 -0
  623. package/src/constraint-runtime/src/reference_data/subsystems/native_ts.json +11 -0
  624. package/src/constraint-runtime/src/reference_data/subsystems/outputStyles.json +8 -0
  625. package/src/constraint-runtime/src/reference_data/subsystems/plugins.json +9 -0
  626. package/src/constraint-runtime/src/reference_data/subsystems/remote.json +11 -0
  627. package/src/constraint-runtime/src/reference_data/subsystems/schemas.json +8 -0
  628. package/src/constraint-runtime/src/reference_data/subsystems/screens.json +10 -0
  629. package/src/constraint-runtime/src/reference_data/subsystems/server.json +10 -0
  630. package/src/constraint-runtime/src/reference_data/subsystems/services.json +32 -0
  631. package/src/constraint-runtime/src/reference_data/subsystems/skills.json +27 -0
  632. package/src/constraint-runtime/src/reference_data/subsystems/state.json +13 -0
  633. package/src/constraint-runtime/src/reference_data/subsystems/types.json +18 -0
  634. package/src/constraint-runtime/src/reference_data/subsystems/upstreamproxy.json +9 -0
  635. package/src/constraint-runtime/src/reference_data/subsystems/utils.json +32 -0
  636. package/src/constraint-runtime/src/reference_data/subsystems/vim.json +12 -0
  637. package/src/constraint-runtime/src/reference_data/subsystems/voice.json +8 -0
  638. package/src/constraint-runtime/src/reference_data/tools_snapshot.json +1042 -0
  639. package/src/constraint-runtime/src/remote/index.ts +15 -0
  640. package/src/constraint-runtime/src/remote_runtime.ts +17 -0
  641. package/src/constraint-runtime/src/runtime/index.ts +1 -0
  642. package/src/constraint-runtime/src/runtime/session.ts +42 -0
  643. package/src/constraint-runtime/src/schemas/index.ts +15 -0
  644. package/src/constraint-runtime/src/screens/index.ts +15 -0
  645. package/src/constraint-runtime/src/server/index.ts +15 -0
  646. package/src/constraint-runtime/src/services/index.ts +15 -0
  647. package/src/constraint-runtime/src/session_store.ts +32 -0
  648. package/src/constraint-runtime/src/setup.ts +50 -0
  649. package/src/constraint-runtime/src/skills/index.ts +1 -0
  650. package/src/constraint-runtime/src/skills/skill-registry.ts +40 -0
  651. package/src/constraint-runtime/src/state/index.ts +15 -0
  652. package/src/constraint-runtime/src/system_init.ts +21 -0
  653. package/src/constraint-runtime/src/thinking/engine.ts +61 -0
  654. package/src/constraint-runtime/src/thinking/index.ts +1 -0
  655. package/src/constraint-runtime/src/tool_pool.ts +20 -0
  656. package/src/constraint-runtime/src/tools/OpenCLI/execAdapter.ts +12 -0
  657. package/src/constraint-runtime/src/tools/OpenCLI/listAdapters.ts +12 -0
  658. package/src/constraint-runtime/src/tools/OpenCLI/runCommand.ts +13 -0
  659. package/src/constraint-runtime/src/tools/PolymarketSDK/cancelOrder.ts +10 -0
  660. package/src/constraint-runtime/src/tools/PolymarketSDK/createOrder.ts +13 -0
  661. package/src/constraint-runtime/src/tools/PolymarketSDK/getMarket.ts +14 -0
  662. package/src/constraint-runtime/src/tools/PolymarketSDK/getOrders.ts +10 -0
  663. package/src/constraint-runtime/src/tools/PolymarketSDK/listMarkets.ts +24 -0
  664. package/src/constraint-runtime/src/tools/SafeSDK/confirmTransaction.ts +13 -0
  665. package/src/constraint-runtime/src/tools/SafeSDK/createTransaction.ts +23 -0
  666. package/src/constraint-runtime/src/tools/SafeSDK/deploySafe.ts +12 -0
  667. package/src/constraint-runtime/src/tools/SafeSDK/executeTransaction.ts +12 -0
  668. package/src/constraint-runtime/src/tools/SafeSDK/getBalance.ts +10 -0
  669. package/src/constraint-runtime/src/tools/SafeSDK/getPendingTransactions.ts +10 -0
  670. package/src/constraint-runtime/src/tools/SafeSDK/proposeTransaction.ts +14 -0
  671. package/src/constraint-runtime/src/tools/WalletTools/autoPay.ts +58 -0
  672. package/src/constraint-runtime/src/tools/WalletTools/createWallet.ts +19 -0
  673. package/src/constraint-runtime/src/tools/WalletTools/getBalance.ts +28 -0
  674. package/src/constraint-runtime/src/tools/WalletTools/importWallet.ts +34 -0
  675. package/src/constraint-runtime/src/tools/WalletTools/sendTransaction.ts +50 -0
  676. package/src/constraint-runtime/src/tools/WalletTools/signMessage.ts +23 -0
  677. package/src/constraint-runtime/src/tools/WalletTools/transferToken.ts +49 -0
  678. package/src/constraint-runtime/src/tools.ts +100 -0
  679. package/src/constraint-runtime/src/transcript.ts +23 -0
  680. package/src/constraint-runtime/src/types/index.ts +15 -0
  681. package/src/constraint-runtime/src/upstream_proxy/index.ts +15 -0
  682. package/src/constraint-runtime/src/utils/index.ts +15 -0
  683. package/src/constraint-runtime/src/vim/index.ts +15 -0
  684. package/src/constraint-runtime/src/voice/index.ts +15 -0
  685. package/src/constraint-runtime/tests/agent.test.ts +20 -0
  686. package/src/constraint-runtime/tests/constraint.test.ts +47 -0
  687. package/src/constraint-runtime/tests/skill.test.ts +23 -0
  688. package/src/constraint-runtime/tests/thinking.test.ts +28 -0
  689. package/src/constraint-runtime/tsconfig.json +13 -0
  690. package/src/constraints/index.ts +6 -0
  691. package/src/documents/reader.ts +80 -0
  692. package/src/electron-preload.ts +25 -0
  693. package/src/electron.ts +200 -0
  694. package/src/index.ts +1606 -0
  695. package/src/llm/config-store.ts +312 -0
  696. package/src/llm/llm-judgment-client.ts +470 -0
  697. package/src/llm/pi-ai.ts +522 -0
  698. package/src/network/agent-network.ts +800 -0
  699. package/src/network/hybrid-messenger.ts +199 -0
  700. package/src/network/iroh-bootstrap.ts +57 -0
  701. package/src/network/iroh-discovery.ts +207 -0
  702. package/src/network/iroh-integration.ts +158 -0
  703. package/src/network/iroh-transport.ts +594 -0
  704. package/src/network/p2p.ts +963 -0
  705. package/src/network/storage/adapters/json-adapter.ts +439 -0
  706. package/src/network/storage/index.ts +187 -0
  707. package/src/network/storage/types.ts +166 -0
  708. package/src/pi-ecosystem/index.ts +453 -0
  709. package/src/pi-ecosystem-colony/index.ts +482 -0
  710. package/src/pi-ecosystem-goals/index.ts +585 -0
  711. package/src/pi-ecosystem-judgment/decision.ts +431 -0
  712. package/src/pi-ecosystem-judgment/distillation.ts +398 -0
  713. package/src/pi-ecosystem-judgment/human-value-store.ts +580 -0
  714. package/src/pi-ecosystem-judgment/index.ts +678 -0
  715. package/src/pi-ecosystem-judgment/value-injection.ts +744 -0
  716. package/src/pi-ecosystem-mcp/index.ts +427 -0
  717. package/src/pi-ecosystem-subagents/index.ts +408 -0
  718. package/src/social/ant-colony/AdaptiveHeartbeat.ts +131 -0
  719. package/src/social/ant-colony/PheromoneEngine.ts +302 -0
  720. package/src/social/ant-colony/index.ts +18 -0
  721. package/src/social/ant-colony/types.ts +94 -0
  722. package/src/social/channels/ChannelManager.ts +485 -0
  723. package/src/social/channels/DiapChannelBridge.ts +501 -0
  724. package/src/social/channels/InterestMatcher.ts +189 -0
  725. package/src/social/channels/agent-workflow-config.json +214 -0
  726. package/src/social/channels/agent-workflow-config.yaml +334 -0
  727. package/src/social/channels/channel-agent-session.ts +407 -0
  728. package/src/social/channels/channel-heartbeat-agent.ts +622 -0
  729. package/src/social/channels/diap-doc-parser.ts +282 -0
  730. package/src/social/channels/harness-workflow-integrator.ts +594 -0
  731. package/src/social/channels/index.ts +22 -0
  732. package/src/social/channels/types.ts +115 -0
  733. package/src/social/global-shared-context.ts +506 -0
  734. package/src/social/heartbeat.ts +1041 -0
  735. package/src/social/persona/enhanced-persona.ts +359 -0
  736. package/src/test/ai-judgment-test.ts +92 -0
  737. package/src/test/bollharness-integration.test.ts +398 -0
  738. package/src/test/channel-agent-multi-dialogue.ts +265 -0
  739. package/src/test/channel-heartbeat-agent-test.ts +244 -0
  740. package/src/test/constraint-layer.test.ts +191 -0
  741. package/src/test/diap-identity-test.ts +222 -0
  742. package/src/test/diap-quick-test.ts +73 -0
  743. package/src/test/global-shared-context.test.ts +393 -0
  744. package/src/test/harness-judgment-injection.test.ts +353 -0
  745. package/src/test/harness-workflow-integrator-test.ts +285 -0
  746. package/src/test/human-value-store.test.ts +316 -0
  747. package/src/test/hybrid-integration-test.ts +126 -0
  748. package/src/test/hybrid-messenger-verify.ts +68 -0
  749. package/src/test/iroh-bistream-debug.ts +50 -0
  750. package/src/test/iroh-communication.test.ts +81 -0
  751. package/src/test/iroh-debug-test.ts +69 -0
  752. package/src/test/iroh-diap-test.ts +90 -0
  753. package/src/test/iroh-direct-connect.ts +65 -0
  754. package/src/test/iroh-e2e-fixed.ts +106 -0
  755. package/src/test/iroh-e2e-same-process.ts +83 -0
  756. package/src/test/iroh-e2e.ts +83 -0
  757. package/src/test/iroh-final-e2e.ts +84 -0
  758. package/src/test/iroh-relay-test.ts +46 -0
  759. package/src/test/iroh-simple-test.ts +49 -0
  760. package/src/test/iroh-transport-verify.ts +60 -0
  761. package/src/test/iroh-transport.test.ts +47 -0
  762. package/src/test/iroh-two-nodes.ts +87 -0
  763. package/src/test/iroh-verify.ts +55 -0
  764. package/src/test/judgment-decision.test.ts +373 -0
  765. package/src/test/llm-judgment-integration.test.ts +257 -0
  766. package/src/test/p2p-agent-complex-dialogue.ts +490 -0
  767. package/src/test/p2p-agent-dialogue.ts +423 -0
  768. package/src/test/p2p-agent-full-bidirectional.ts +686 -0
  769. package/src/test/p2p-agent-harness-flow.ts +562 -0
  770. package/src/test/p2p-agent-harness-single.ts +175 -0
  771. package/src/test/p2p-ai-dialogue-test.ts +374 -0
  772. package/src/test/p2p-cid-connect-test.ts +245 -0
  773. package/src/test/p2p-connect-receiver.ts +85 -0
  774. package/src/test/p2p-iroh-test.ts +214 -0
  775. package/src/test/p2p-minimal-test.ts +264 -0
  776. package/src/test/p2p-node-1.ts +172 -0
  777. package/src/test/p2p-node-2.ts +172 -0
  778. package/src/test/p2p-server.ts +335 -0
  779. package/src/test/p2p-two-nodes-test.ts +542 -0
  780. package/src/test/pi-sdk.test.ts +47 -0
  781. package/src/test/set-persona.ts +56 -0
  782. package/src/test/simple.test.ts +11 -0
  783. package/src/test/storage-integration.test.ts +191 -0
  784. package/src/test/subagent-manager.test.ts +392 -0
  785. package/src/test/test-gate-flow.test.ts +92 -0
  786. package/src/test/workflow-engine.test.ts +101 -0
  787. package/src/web/api-config.html +265 -0
  788. package/src/web/client.js +1401 -0
  789. package/src/web/components/p2p/P2PModal.tsx +328 -0
  790. package/src/web/components/p2p/index.ts +404 -0
  791. package/src/web/components/p2p/index.tsx +297 -0
  792. package/src/web/components/p2p/p2p-connection.ts +186 -0
  793. package/src/web/components/p2p/p2p-identity.ts +92 -0
  794. package/src/web/components/p2p/p2p-manager.ts +109 -0
  795. package/src/web/components/p2p/p2p-messages.ts +219 -0
  796. package/src/web/components/p2p/p2p-modal.ts +685 -0
  797. package/src/web/components/p2p/p2p-store-memory.ts +162 -0
  798. package/src/web/components/p2p/types.ts +112 -0
  799. package/src/web/design.md +99 -0
  800. package/src/web/index.html +192 -0
  801. package/src/web/server.ts +1878 -0
  802. package/src/web/style.css +3260 -0
  803. package/src/workflows/collaboration.ts +455 -0
  804. package/src/workflows/index.ts +64 -0
  805. package/tsconfig.electron.json +20 -0
  806. package/tsconfig.json +19 -0
  807. package/vitest.config.ts +12 -0
  808. package//346/203/263/346/263/225.md +79 -0
@@ -0,0 +1,1401 @@
1
+ const messagesEl = document.getElementById('messages');
2
+ const input = document.getElementById('input');
3
+ const sendBtn = document.getElementById('send');
4
+ const sidebar = document.getElementById('sidebar');
5
+ const sidebarToggle = document.getElementById('sidebar-toggle');
6
+ const themeToggle = document.getElementById('theme-toggle');
7
+ const channelList = document.getElementById('channel-list');
8
+ const newChannelBtn = document.getElementById('new-channel-btn');
9
+ const newChannelInput = document.getElementById('new-channel-input');
10
+ const channelNameEl = document.getElementById('channel-name');
11
+ const loadSessionBtn = document.getElementById('load-session-btn');
12
+ const sessionFileInput = document.getElementById('session-file-input');
13
+
14
+ let eventSource = null;
15
+ let currentChannelId = null;
16
+ let currentAgentId = '';
17
+ let channels = [];
18
+ let isSidebarCollapsed = false;
19
+ let reconnectAttempts = 0;
20
+ let reconnectTimer = null; // 改进: 跟踪重连定时器
21
+ let lastUserCommand = ''; // 防止用户消息重复显示
22
+ let lastAiContent = ''; // 防止 AI 消息重复显示
23
+
24
+ function generateId() {
25
+ return crypto.randomUUID();
26
+ }
27
+
28
+ function applyTheme(theme) {
29
+ document.documentElement.setAttribute('data-theme', theme);
30
+ }
31
+
32
+ async function loadTheme() {
33
+ try {
34
+ const res = await fetch('/theme');
35
+ const data = await res.json();
36
+ applyTheme(data.theme);
37
+ if (data.agentId) {
38
+ currentAgentId = data.agentId;
39
+ }
40
+ return data;
41
+ } catch {
42
+ applyTheme('dark');
43
+ return { theme: 'dark', agentId: '' };
44
+ }
45
+ }
46
+
47
+ async function saveTheme(theme, agentId) {
48
+ try {
49
+ await fetch('/theme', {
50
+ method: 'POST',
51
+ headers: { 'Content-Type': 'application/json' },
52
+ body: JSON.stringify({ theme, agentId })
53
+ });
54
+ } catch (err) {
55
+ console.error('Failed to save theme:', err);
56
+ }
57
+ }
58
+
59
+ function toggleTheme() {
60
+ const current = document.documentElement.getAttribute('data-theme') || 'dark';
61
+ const next = current === 'dark' ? 'light' : 'dark';
62
+ applyTheme(next);
63
+ saveTheme(next, currentAgentId);
64
+ }
65
+
66
+ function toggleSidebar() {
67
+ isSidebarCollapsed = !isSidebarCollapsed;
68
+
69
+ if (isSidebarCollapsed) {
70
+ sidebar.classList.add('collapsed');
71
+ } else {
72
+ sidebar.classList.remove('collapsed');
73
+ }
74
+ }
75
+
76
+ function expandSidebar() {
77
+ isSidebarCollapsed = false;
78
+ sidebar.classList.remove('collapsed');
79
+ }
80
+
81
+ async function loadChannels() {
82
+ try {
83
+ const res = await fetch('/channels');
84
+ channels = await res.json();
85
+ console.log('[加载频道] 从服务器获取到', channels.length, '个频道');
86
+ channels.forEach((ch, i) => {
87
+ console.log(` [${i}] ${ch.name} - did: "${ch.did}"`);
88
+ });
89
+ renderChannels();
90
+ } catch (err) {
91
+ console.error('[加载频道] 失败:', err);
92
+ }
93
+ }
94
+
95
+ async function createChannel(name) {
96
+ if (!name.trim()) return;
97
+ try {
98
+ const res = await fetch('/channels', {
99
+ method: 'POST',
100
+ headers: { 'Content-Type': 'application/json' },
101
+ body: JSON.stringify({ name: name.trim(), agentId: currentAgentId })
102
+ });
103
+ const channel = await res.json();
104
+ console.log('[创建频道] 服务器返回:', channel);
105
+ console.log('[创建频道] DID:', channel.did, 'CID:', channel.cid);
106
+
107
+ // 立即添加频道并切换(不等待 DID)
108
+ channels.push(channel);
109
+ renderChannels();
110
+ selectChannel(channel.id);
111
+ newChannelInput.value = '';
112
+
113
+ // 后台更新 DID(如果还没有的话)
114
+ if (!channel.did || channel.did === 'undefined') {
115
+ console.log('[创建频道] 后台生成 DID...');
116
+ // 触发后台刷新,DID 会在下次请求时更新
117
+ setTimeout(() => {
118
+ fetch('/channels').then(res => res.json()).then(allChannels => {
119
+ channels = allChannels;
120
+ const updated = channels.find(c => c.id === channel.id);
121
+ if (updated && updated.did && updated.did !== 'undefined') {
122
+ console.log('[创建频道] DID 生成完成:', updated.did);
123
+ }
124
+ });
125
+ }, 2000);
126
+ }
127
+ } catch (err) {
128
+ console.error('Failed to create channel:', err);
129
+ }
130
+ }
131
+
132
+ async function deleteChannel(channelId, e) {
133
+ e.stopPropagation();
134
+ try {
135
+ await fetch(`/channels/${channelId}`, { method: 'DELETE' });
136
+ channels = channels.filter(c => c.id !== channelId);
137
+ if (currentChannelId === channelId) {
138
+ currentChannelId = channels[0]?.id || null;
139
+ if (currentChannelId) {
140
+ await loadSession(currentChannelId);
141
+ } else {
142
+ messagesEl.innerHTML = '';
143
+ }
144
+ }
145
+ renderChannels();
146
+ renderCollapsedChannels();
147
+ } catch (err) {
148
+ console.error('Failed to delete channel:', err);
149
+ }
150
+ }
151
+
152
+ function renderChannels() {
153
+ if (!channelList) return;
154
+ channelList.innerHTML = '';
155
+
156
+ // 使用 DocumentFragment 减少 DOM 操作
157
+ const fragment = document.createDocumentFragment();
158
+
159
+ channels.forEach(ch => {
160
+ const li = document.createElement('li');
161
+ li.className = `channel-item ${ch.id === currentChannelId ? 'active' : ''}`;
162
+ li.onclick = () => {
163
+ expandSidebar();
164
+ selectChannel(ch.id);
165
+ };
166
+ li.innerHTML = `
167
+ <div class="channel-icon">💬</div>
168
+ <span class="channel-name">${ch.name}</span>
169
+ <button class="channel-delete" data-id="${ch.id}">×</button>
170
+ `;
171
+
172
+ const deleteBtn = li.querySelector('.channel-delete');
173
+ if (deleteBtn) {
174
+ deleteBtn.onclick = (e) => deleteChannel(ch.id, e);
175
+ }
176
+
177
+ fragment.appendChild(li);
178
+ });
179
+
180
+ channelList.appendChild(fragment);
181
+ }
182
+
183
+ function renderCollapsedChannels() {
184
+ return;
185
+ }
186
+
187
+ async function selectChannel(channelId) {
188
+ console.log('[selectChannel] 开始切换到:', channelId);
189
+
190
+ // 立即更新当前频道 ID
191
+ currentChannelId = channelId;
192
+ reconnectAttempts = 0;
193
+
194
+ // 找到当前频道
195
+ const channel = channels.find(c => c.id === channelId);
196
+ if (channel && channelNameEl) {
197
+ channelNameEl.textContent = channel.name;
198
+ console.log('[selectChannel] 频道:', channel.name, 'DID:', channel.did || '无');
199
+ }
200
+
201
+ renderChannels();
202
+
203
+ // 关闭旧 SSE 连接
204
+ if (eventSource) {
205
+ console.log('[selectChannel] 关闭旧 SSE 连接');
206
+ eventSource.close();
207
+ eventSource = null;
208
+ }
209
+
210
+ // 清空消息区域
211
+ messagesEl.innerHTML = '';
212
+
213
+ // 加载 session
214
+ try {
215
+ const res = await fetch(`/sessions/${channelId}`);
216
+ const session = await res.json();
217
+ if (session.messages && session.messages.length > 0) {
218
+ session.messages.forEach(msg => {
219
+ addMessage(msg.content, msg.type, false);
220
+ });
221
+ } else {
222
+ addMessage('你好!我是 Bolloon Agent。有什么我可以帮你的吗?', 'ai', false);
223
+ }
224
+ } catch (err) {
225
+ console.error('[selectChannel] 加载 session 失败:', err);
226
+ addMessage('你好!我是 Bolloon Agent。有什么我可以帮你的吗?', 'ai', false);
227
+ }
228
+
229
+ // 建立新 SSE 连接
230
+ console.log('[selectChannel] 建立 SSE 连接');
231
+ connect();
232
+ }
233
+
234
+ async function loadSession(channelId) {
235
+ try {
236
+ const res = await fetch(`/sessions/${channelId}`);
237
+ const session = await res.json();
238
+ messagesEl.innerHTML = '';
239
+ if (session.messages && session.messages.length > 0) {
240
+ session.messages.forEach(msg => {
241
+ addMessage(msg.content, msg.type, false);
242
+ });
243
+ } else {
244
+ addMessage('你好!我是 Bolloon Agent。有什么我可以帮你的吗?', 'ai', false);
245
+ }
246
+ } catch (err) {
247
+ console.error('Failed to load session:', err);
248
+ messagesEl.innerHTML = '';
249
+ addMessage('你好!我是 Bolloon Agent。有什么我可以帮你的吗?', 'ai', false);
250
+ }
251
+ }
252
+
253
+ function addMessage(content, type, save = true) {
254
+ // 去重:只有 save=true 时(来自 SSE)才去重,save=false 时(来自 session 加载)直接显示
255
+ if (save) {
256
+ const lastContent = type === 'user' ? lastUserCommand : lastAiContent;
257
+ if (lastContent && content === lastContent) {
258
+ console.log(`[addMessage] 跳过重复的 ${type} 消息`);
259
+ return;
260
+ }
261
+ if (type === 'user') {
262
+ lastUserCommand = content;
263
+ } else {
264
+ lastAiContent = content;
265
+ }
266
+ }
267
+
268
+ const div = document.createElement('div');
269
+ div.className = `message message-${type}`;
270
+
271
+ // 清理工具结果容器(当新的 AI 消息到达时)
272
+ if (type === 'ai' && toolResultContainer) {
273
+ toolResultContainer.remove();
274
+ toolResultContainer = null;
275
+ }
276
+
277
+ // 清理内容:移除 tool call 标记和其他不应该显示的内容
278
+ let cleanContent = content
279
+ .replace(/\[TOOL_CALL\][\s\S]*?\[\/TOOL_CALL\]/g, '')
280
+ .replace(/TOOL_CALL[\s\S]*?\/TOOL_CALL/g, '')
281
+ .replace(/<tool_call>[\s\S]*?<\/tool_call>/gi, '')
282
+ .replace(/{\s*"tool":[\s\S]*?}/g, '')
283
+ .replace(/\[Function[^\]]*\]\s*/g, '')
284
+ .trim();
285
+
286
+ // 处理思维链内容(</think> 标签)- 折叠显示
287
+ const thinkMatch = cleanContent.match(/<think>([\s\S]*?)<\/think>/);
288
+ let mainContent = cleanContent;
289
+ let thinkContainer = null;
290
+
291
+ if (thinkMatch) {
292
+ const thinkContent = thinkMatch[1].trim();
293
+ mainContent = cleanContent.replace(/<think>[\s\S]*?<\/think>/, '').trim();
294
+
295
+ // 创建思维链折叠区域
296
+ thinkContainer = document.createElement('div');
297
+ thinkContainer.className = 'think-container';
298
+
299
+ const thinkToggle = document.createElement('div');
300
+ thinkToggle.className = 'think-toggle';
301
+ thinkToggle.innerHTML = '💭 思考过程 <span class="think-arrow">▸</span>';
302
+ thinkToggle.onclick = function() {
303
+ const details = thinkContainer.querySelector('.think-content');
304
+ const arrow = thinkToggle.querySelector('.think-arrow');
305
+ if (details.style.display === 'none') {
306
+ details.style.display = 'block';
307
+ arrow.textContent = '▾';
308
+ } else {
309
+ details.style.display = 'none';
310
+ arrow.textContent = '▸';
311
+ }
312
+ };
313
+
314
+ const thinkDiv = document.createElement('div');
315
+ thinkDiv.className = 'think-content';
316
+ thinkDiv.style.display = 'none'; // 默认折叠
317
+ thinkDiv.innerHTML = `<pre>${thinkContent.replace(/</g, '&lt;').replace(/>/g, '&gt;')}</pre>`;
318
+
319
+ thinkContainer.appendChild(thinkToggle);
320
+ thinkContainer.appendChild(thinkDiv);
321
+ }
322
+
323
+ const envMatch = mainContent.match(/^(.+?)\n<environment_details>([\s\S]*?)<\/environment_details>\n([\s\S]*)$/);
324
+
325
+ if (envMatch) {
326
+ const identity = envMatch[1].trim();
327
+ const envDetails = envMatch[2].trim();
328
+ const messageBody = envMatch[3].trim();
329
+
330
+ const header = document.createElement('div');
331
+ header.className = 'message-header';
332
+ header.textContent = identity;
333
+ div.appendChild(header);
334
+
335
+ // 环境信息区域(可折叠,默认折叠)
336
+ const envContainer = document.createElement('div');
337
+ envContainer.className = 'env-container';
338
+
339
+ const envToggle = document.createElement('div');
340
+ envToggle.className = 'env-toggle';
341
+ envToggle.innerHTML = '⚙️ 环境信息 <span class="env-arrow">▸</span>';
342
+ envToggle.onclick = function() {
343
+ const details = envContainer.querySelector('.environment-details');
344
+ const arrow = envToggle.querySelector('.env-arrow');
345
+ if (details.style.display === 'none') {
346
+ details.style.display = 'block';
347
+ arrow.textContent = '▾';
348
+ } else {
349
+ details.style.display = 'none';
350
+ arrow.textContent = '▸';
351
+ }
352
+ };
353
+
354
+ const envDiv = document.createElement('div');
355
+ envDiv.className = 'environment-details';
356
+ envDiv.style.display = 'none'; // 默认折叠
357
+ envDiv.innerHTML = `<pre>${envDetails}</pre>`;
358
+
359
+ envContainer.appendChild(envToggle);
360
+ envContainer.appendChild(envDiv);
361
+ div.appendChild(envContainer);
362
+
363
+ // 思考在环境信息下面
364
+ if (thinkContainer) {
365
+ div.appendChild(thinkContainer);
366
+ }
367
+
368
+ if (messageBody) {
369
+ const bubble = document.createElement('div');
370
+ bubble.className = `bubble bubble-${type}`;
371
+ bubble.innerHTML = marked.parse(messageBody);
372
+ div.appendChild(bubble);
373
+ }
374
+ } else if (cleanContent) {
375
+ // 没有环境信息时,思考放在消息之前
376
+ if (thinkContainer) {
377
+ div.appendChild(thinkContainer);
378
+ }
379
+ const bubble = document.createElement('div');
380
+ bubble.className = `bubble bubble-${type}`;
381
+ bubble.innerHTML = marked.parse(cleanContent);
382
+ div.appendChild(bubble);
383
+ } else {
384
+ return; // 没有有效内容,不显示空消息
385
+ }
386
+
387
+ // 提取纯文本内容用于复制(去除 HTML 标签)
388
+ const rawContent = cleanContent
389
+ .replace(/<[^>]+>/g, '')
390
+ .replace(/&lt;/g, '<')
391
+ .replace(/&gt;/g, '>')
392
+ .replace(/&amp;/g, '&')
393
+ .replace(/&nbsp;/g, ' ');
394
+
395
+ const time = document.createElement('div');
396
+ time.className = 'time';
397
+ time.textContent = new Date().toLocaleTimeString('zh-CN', { hour: '2-digit', minute: '2-digit' });
398
+
399
+ // AI 消息添加操作按钮
400
+ if (type === 'ai') {
401
+ const actionsDiv = document.createElement('div');
402
+ actionsDiv.className = 'message-actions';
403
+
404
+ const copyBtn = document.createElement('button');
405
+ copyBtn.className = 'action-btn copy-btn';
406
+ copyBtn.innerHTML = `<svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><rect x="9" y="9" width="13" height="13" rx="2" ry="2"></rect><path d="M5 15H4a2 2 0 0 1-2-2V4a2 2 0 0 1 2-2h9a2 2 0 0 1 2 2v1"></path></svg> 复制`;
407
+ copyBtn.title = '复制消息';
408
+ copyBtn.onclick = function() {
409
+ navigator.clipboard.writeText(rawContent).then(() => {
410
+ copyBtn.innerHTML = `<svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><polyline points="20 6 9 17 4 12"></polyline></svg> 已复制`;
411
+ setTimeout(() => {
412
+ copyBtn.innerHTML = `<svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><rect x="9" y="9" width="13" height="13" rx="2" ry="2"></rect><path d="M5 15H4a2 2 0 0 1-2-2V4a2 2 0 0 1 2-2h9a2 2 0 0 1 2 2v1"></path></svg> 复制`;
413
+ }, 2000);
414
+ });
415
+ };
416
+
417
+ const regenerateBtn = document.createElement('button');
418
+ regenerateBtn.className = 'action-btn regenerate-btn';
419
+ regenerateBtn.innerHTML = `<svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="M21 2v6h-6M3 12a9 9 0 0 1 15-6.7L21 8M3 22v-6h6M21 12a9 9 0 0 1-15 6.7L3 16"></path></svg> 重新回答`;
420
+ regenerateBtn.title = '重新生成回复';
421
+ regenerateBtn.onclick = function() {
422
+ regenerateBtn.innerHTML = `<svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" class="spin"><path d="M21 2v6h-6M3 12a9 9 0 0 1 15-6.7L21 8M3 22v-6h6M21 12a9 9 0 0 1-15 6.7L3 16"></path></svg> 生成中...`;
423
+ regenerateBtn.disabled = true;
424
+ // 获取当前频道对应的用户消息
425
+ const messages = div.parentElement.querySelectorAll('.message');
426
+ let lastUserMsg = '';
427
+ for (let i = messages.length - 1; i >= 0; i--) {
428
+ const msg = messages[i];
429
+ if (msg.classList.contains('message-user')) {
430
+ const bubble = msg.querySelector('.bubble');
431
+ if (bubble) {
432
+ lastUserMsg = bubble.textContent || bubble.innerText || '';
433
+ break;
434
+ }
435
+ }
436
+ }
437
+ // 调用重新生成 API
438
+ fetch('/regenerate', {
439
+ method: 'POST',
440
+ headers: { 'Content-Type': 'application/json' },
441
+ body: JSON.stringify({ channelId: currentChannelId, userMessage: lastUserMsg })
442
+ }).then(res => {
443
+ if (!res.ok) {
444
+ throw new Error('regenerate failed');
445
+ }
446
+ }).catch(err => {
447
+ console.error('重新生成失败:', err);
448
+ regenerateBtn.innerHTML = `<svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="M21 2v6h-6M3 12a9 9 0 0 1 15-6.7L21 8M3 22v-6h6M21 12a9 9 0 0 1-15 6.7L3 16"></path></svg> 失败`;
449
+ setTimeout(() => {
450
+ regenerateBtn.innerHTML = `<svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="M21 2v6h-6M3 12a9 9 0 0 1 15-6.7L21 8M3 22v-6h6M21 12a9 9 0 0 1-15 6.7L3 16"></path></svg> 重新回答`;
451
+ regenerateBtn.disabled = false;
452
+ }, 2000);
453
+ });
454
+ };
455
+
456
+ actionsDiv.appendChild(copyBtn);
457
+ actionsDiv.appendChild(regenerateBtn);
458
+ div.appendChild(actionsDiv);
459
+ }
460
+
461
+ div.appendChild(time);
462
+ messagesEl.appendChild(div);
463
+
464
+ messagesEl.scrollTop = messagesEl.scrollHeight;
465
+ }
466
+
467
+ function showTyping() {
468
+ hideTyping();
469
+ const div = document.createElement('div');
470
+ div.className = 'message message-ai';
471
+ div.id = 'typing';
472
+ div.innerHTML = '<div class="typing"><div class="typing-spinner"></div><span class="typing-text">思考中...</span></div>';
473
+ messagesEl.appendChild(div);
474
+ messagesEl.scrollTop = messagesEl.scrollHeight;
475
+ }
476
+
477
+ function hideTyping() {
478
+ const typing = document.getElementById('typing');
479
+ if (typing) typing.remove();
480
+ hideStreaming();
481
+ }
482
+
483
+ let streamingMessageEl = null;
484
+
485
+ function showStreaming() {
486
+ hideStreaming();
487
+ streamingMessageEl = document.createElement('div');
488
+ streamingMessageEl.className = 'message message-ai';
489
+ streamingMessageEl.id = 'streaming';
490
+ const bubble = document.createElement('div');
491
+ bubble.className = 'bubble bubble-ai streaming-content';
492
+ streamingMessageEl.appendChild(bubble);
493
+ messagesEl.appendChild(streamingMessageEl);
494
+ messagesEl.scrollTop = messagesEl.scrollHeight;
495
+ }
496
+
497
+ function hideStreaming() {
498
+ if (streamingMessageEl) {
499
+ streamingMessageEl.remove();
500
+ streamingMessageEl = null;
501
+ }
502
+ }
503
+
504
+ function updateStreamingContent(content) {
505
+ if (streamingMessageEl) {
506
+ const bubble = streamingMessageEl.querySelector('.streaming-content');
507
+ if (bubble) {
508
+ bubble.textContent = content;
509
+ messagesEl.scrollTop = messagesEl.scrollHeight;
510
+ }
511
+ }
512
+ }
513
+
514
+ function handleStreamEvent(data) {
515
+ if (data.streamType === 'thinking') {
516
+ showStreaming();
517
+ updateStreamingContent(data.content || '思考中...');
518
+ } else if (data.streamType === 'token') {
519
+ showStreaming();
520
+ const current = streamingMessageEl?.querySelector('.streaming-content')?.textContent || '';
521
+ updateStreamingContent(current + data.content);
522
+ }
523
+ }
524
+
525
+ function handleStatusEvent(data) {
526
+ // 检查是否是工具调用结果
527
+ const content = data.content || '';
528
+ const isJsonResult = content.startsWith('{') && content.includes('"success"');
529
+
530
+ if (isJsonResult) {
531
+ // 工具结果:折叠显示
532
+ showToolResult(data.tool, content);
533
+ } else {
534
+ // 普通状态:流式显示
535
+ showStreaming();
536
+ const icon = data.tool ? `🔧 ${data.tool}: ` : '';
537
+ updateStreamingContent(icon + data.content);
538
+ }
539
+ }
540
+
541
+ // 显示工具调用结果(折叠)
542
+ let toolResultContainer = null;
543
+
544
+ function showToolResult(toolName, resultJson) {
545
+ // 清理之前的流式显示
546
+ hideStreaming();
547
+
548
+ // 获取或创建工具结果容器
549
+ if (!toolResultContainer) {
550
+ toolResultContainer = document.createElement('div');
551
+ toolResultContainer.className = 'tool-results-container';
552
+ messagesEl.appendChild(toolResultContainer);
553
+ }
554
+
555
+ // 尝试解析并格式化 JSON
556
+ let formattedResult = resultJson;
557
+ try {
558
+ const parsed = JSON.parse(resultJson);
559
+ formattedResult = formatToolResult(parsed);
560
+ } catch {}
561
+
562
+ // 创建折叠项
563
+ const resultEl = document.createElement('div');
564
+ resultEl.className = 'tool-result-item collapsed';
565
+
566
+ const toolDisplayName = toolName || '工具结果';
567
+ const headerEl = document.createElement('div');
568
+ headerEl.className = 'tool-result-header';
569
+ headerEl.innerHTML = `
570
+ <span class="tool-result-icon">🔧</span>
571
+ <span class="tool-result-name">${toolDisplayName}</span>
572
+ <span class="tool-result-toggle">▸</span>
573
+ `;
574
+ // 绑定事件处理器(避免内联 onclick)
575
+ headerEl.addEventListener('click', () => {
576
+ resultEl.classList.toggle('collapsed');
577
+ resultEl.classList.toggle('expanded');
578
+ });
579
+
580
+ const contentEl = document.createElement('div');
581
+ contentEl.className = 'tool-result-content';
582
+ contentEl.innerHTML = `<pre>${formattedResult}</pre>`;
583
+
584
+ resultEl.appendChild(headerEl);
585
+ resultEl.appendChild(contentEl);
586
+ toolResultContainer.appendChild(resultEl);
587
+ messagesEl.scrollTop = messagesEl.scrollHeight;
588
+ }
589
+
590
+ // 格式化工具结果为易读格式
591
+ function formatToolResult(obj, indent = 0) {
592
+ const spaces = ' '.repeat(indent);
593
+
594
+ if (obj === null || obj === undefined) {
595
+ return 'null';
596
+ }
597
+
598
+ if (typeof obj === 'object') {
599
+ if (Array.isArray(obj)) {
600
+ if (obj.length === 0) return '[]';
601
+ return obj.map(item => spaces + '- ' + formatToolResult(item, indent + 1)).join('\n');
602
+ }
603
+
604
+ const keys = Object.keys(obj);
605
+ if (keys.length === 0) return '{}';
606
+
607
+ return keys.map(key => {
608
+ const value = obj[key];
609
+ if (typeof value === 'object') {
610
+ return `${spaces}${key}:\n${formatToolResult(value, indent + 1)}`;
611
+ }
612
+ return `${spaces}${key}: ${value}`;
613
+ }).join('\n');
614
+ }
615
+
616
+ return String(obj);
617
+ }
618
+
619
+ // 工作流状态显示
620
+ let workflowDisplayEl = null;
621
+
622
+ function createWorkflowDisplay() {
623
+ const container = document.createElement('div');
624
+ container.id = 'workflow-display';
625
+ container.className = 'workflow-display';
626
+ container.innerHTML = `
627
+ <div class="workflow-header">
628
+ <span class="workflow-icon">🔄</span>
629
+ <span class="workflow-title">工作流执行中</span>
630
+ <span class="workflow-loop-count"></span>
631
+ </div>
632
+ <div class="workflow-steps-list"></div>
633
+ <div class="workflow-streams"></div>
634
+ `;
635
+ return container;
636
+ }
637
+
638
+ function handleTaskStatusEvent(data) {
639
+ console.log('[工作流] 任务状态:', data);
640
+
641
+ // 获取或创建工作流显示区域
642
+ if (!workflowDisplayEl) {
643
+ workflowDisplayEl = createWorkflowDisplay();
644
+ messagesEl.appendChild(workflowDisplayEl);
645
+ }
646
+
647
+ const stepsList = workflowDisplayEl.querySelector('.workflow-steps-list');
648
+ const header = workflowDisplayEl.querySelector('.workflow-header');
649
+
650
+ // 更新标题
651
+ if (data.taskId) {
652
+ header.querySelector('.workflow-title').textContent = `任务: ${data.taskId.substring(0, 12)}...`;
653
+ }
654
+
655
+ // 更新状态
656
+ if (data.status) {
657
+ const statusText = header.querySelector('.workflow-title');
658
+ statusText.textContent = `任务 ${data.status === 'running' ? '执行中' : data.status}`;
659
+ }
660
+
661
+ // 更新进度
662
+ if (data.progress !== undefined) {
663
+ header.querySelector('.workflow-title').textContent = `进度: ${data.progress}%`;
664
+ }
665
+
666
+ // 更新步骤
667
+ if (data.currentStep !== undefined && data.totalSteps) {
668
+ const stepEl = document.createElement('div');
669
+ stepEl.className = 'workflow-step-item';
670
+ stepEl.innerHTML = `
671
+ <span class="step-running">⟳</span>
672
+ <span>步骤 ${data.currentStep + 1}/${data.totalSteps}</span>
673
+ `;
674
+ stepsList.appendChild(stepEl);
675
+ }
676
+
677
+ // 完成时移除显示
678
+ if (data.status === 'completed' || data.status === 'failed') {
679
+ setTimeout(() => {
680
+ if (workflowDisplayEl) {
681
+ workflowDisplayEl.remove();
682
+ workflowDisplayEl = null;
683
+ }
684
+ }, 3000);
685
+ }
686
+ }
687
+
688
+ function handleWorkflowStepEvent(data) {
689
+ console.log('[工作流] 步骤:', data);
690
+
691
+ if (!workflowDisplayEl) {
692
+ workflowDisplayEl = createWorkflowDisplay();
693
+ messagesEl.appendChild(workflowDisplayEl);
694
+ }
695
+
696
+ const streamsDiv = workflowDisplayEl.querySelector('.workflow-streams');
697
+
698
+ // 如果有步骤标签,显示步骤信息
699
+ if (data.step) {
700
+ const stepEl = document.createElement('div');
701
+ stepEl.className = 'workflow-step-stream';
702
+ stepEl.innerHTML = `
703
+ <div class="step-label">🔧 ${data.step}</div>
704
+ <div class="step-content">${data.content || ''}</div>
705
+ `;
706
+ streamsDiv.appendChild(stepEl);
707
+ messagesEl.scrollTop = messagesEl.scrollHeight;
708
+ }
709
+ }
710
+
711
+ function handleWorkflowLoopEvent(data) {
712
+ console.log('[工作流] 循环:', data);
713
+
714
+ if (!workflowDisplayEl) {
715
+ workflowDisplayEl = createWorkflowDisplay();
716
+ messagesEl.appendChild(workflowDisplayEl);
717
+ }
718
+
719
+ const loopCount = workflowDisplayEl.querySelector('.workflow-loop-count');
720
+ const streamsDiv = workflowDisplayEl.querySelector('.workflow-streams');
721
+
722
+ // 更新循环次数
723
+ if (data.loopCount !== undefined) {
724
+ loopCount.textContent = `循环 #${data.loopCount}`;
725
+ loopCount.style.display = 'inline';
726
+ }
727
+
728
+ // 显示循环信息
729
+ if (data.content) {
730
+ const loopEl = document.createElement('div');
731
+ loopEl.className = 'workflow-loop-item';
732
+ loopEl.innerHTML = `
733
+ <div class="loop-header">
734
+ <span class="loop-icon">🔁</span>
735
+ <span>循环 ${data.loopCount || '?'}</span>
736
+ <span class="loop-status">${data.status || ''}</span>
737
+ </div>
738
+ <div class="loop-content">${data.content}</div>
739
+ `;
740
+ streamsDiv.appendChild(loopEl);
741
+ messagesEl.scrollTop = messagesEl.scrollHeight;
742
+ }
743
+ }
744
+
745
+ // 用户命令可视化 - 当用户发送命令时调用
746
+ let userCommandDisplayEl = null;
747
+
748
+ function showUserCommand(command) {
749
+ // 先移除之前的消息中的 user bubble(如果有重复的话)
750
+ const existingUserBubbles = messagesEl.querySelectorAll('.message-user');
751
+ existingUserBubbles.forEach(el => el.remove());
752
+
753
+ // 移除之前的命令显示
754
+ if (userCommandDisplayEl) {
755
+ userCommandDisplayEl.remove();
756
+ }
757
+
758
+ // 创建美化版本的命令显示
759
+ userCommandDisplayEl = document.createElement('div');
760
+ userCommandDisplayEl.className = 'message message-user';
761
+ userCommandDisplayEl.innerHTML = `
762
+ <div class="user-command-display">
763
+ <div class="command-prompt">
764
+ <span class="prompt-icon">›</span>
765
+ <span class="prompt-text">${command}</span>
766
+ </div>
767
+ </div>
768
+ `;
769
+
770
+ // 在工作流显示之前插入
771
+ if (workflowDisplayEl) {
772
+ messagesEl.insertBefore(userCommandDisplayEl, workflowDisplayEl);
773
+ } else {
774
+ messagesEl.appendChild(userCommandDisplayEl);
775
+ }
776
+
777
+ messagesEl.scrollTop = messagesEl.scrollHeight;
778
+ }
779
+
780
+ function connect() {
781
+ // 清除旧的重连定时器
782
+ if (reconnectTimer) {
783
+ clearTimeout(reconnectTimer);
784
+ reconnectTimer = null;
785
+ }
786
+
787
+ // 关闭已有连接
788
+ if (eventSource) {
789
+ eventSource.close();
790
+ eventSource = null;
791
+ }
792
+
793
+ const sseUrl = currentChannelId ? `/events?channelId=${encodeURIComponent(currentChannelId)}` : '/events';
794
+ console.log('[connect] 创建 SSE 连接:', sseUrl);
795
+
796
+ eventSource = new EventSource(sseUrl);
797
+
798
+ eventSource.onopen = () => {
799
+ console.log('[SSE] 已连接');
800
+ reconnectAttempts = 0;
801
+ };
802
+
803
+ eventSource.onerror = () => {
804
+ console.error('[SSE] 连接错误');
805
+ eventSource.close();
806
+ eventSource = null;
807
+ reconnectAttempts++;
808
+ // 清除旧的重连定时器
809
+ if (reconnectTimer) clearTimeout(reconnectTimer);
810
+ reconnectTimer = setTimeout(connect, Math.min(5000 * reconnectAttempts, 30000));
811
+ };
812
+
813
+ eventSource.onmessage = (e) => {
814
+ try {
815
+ const data = JSON.parse(e.data);
816
+ console.log('[SSE] 收到消息:', data.type, 'channelId:', data.channelId);
817
+
818
+ // 只处理当前频道的消息
819
+ if (data.channelId && data.channelId !== currentChannelId) {
820
+ console.log('[SSE] 忽略非当前频道消息');
821
+ return;
822
+ }
823
+
824
+ if (data.type === 'user') {
825
+ showUserCommand(data.content); // 用户命令可视化(已包含 message-user)
826
+ // 不再调用 addMessage,避免重复
827
+ } else if (data.type === 'ai') {
828
+ addMessage(data.content, 'ai');
829
+ } else if (data.type === 'stream') {
830
+ handleStreamEvent(data);
831
+ } else if (data.type === 'regenerating') {
832
+ // 开始重新生成,清除最后一个 AI 消息
833
+ const messages = messagesEl.querySelectorAll('.message-ai');
834
+ if (messages.length > 0) {
835
+ const lastAiMsg = messages[messages.length - 1];
836
+ lastAiMsg.remove();
837
+ }
838
+ showTyping();
839
+ } else if (data.type === 'status') {
840
+ handleStatusEvent(data);
841
+ } else if (data.type === 'done') {
842
+ hideTyping();
843
+ } else if (data.type === 'renamed') {
844
+ const channel = channels.find(c => c.id === data.channelId);
845
+ if (channel) {
846
+ channel.name = data.newName;
847
+ renderChannels();
848
+ if (currentChannelId === data.channelId && channelNameEl) {
849
+ channelNameEl.textContent = data.newName;
850
+ }
851
+ }
852
+ } else if (data.type === 'error') {
853
+ hideTyping();
854
+ addMessage('错误: ' + data.content, 'ai');
855
+ } else if (data.type === 'task_status') {
856
+ handleTaskStatusEvent(data);
857
+ } else if (data.type === 'workflow_step') {
858
+ handleWorkflowStepEvent(data);
859
+ } else if (data.type === 'workflow_loop') {
860
+ handleWorkflowLoopEvent(data);
861
+ }
862
+ } catch (parseErr) {
863
+ console.error('[SSE] 解析错误', parseErr);
864
+ }
865
+ };
866
+ }
867
+
868
+ async function sendMessage() {
869
+ const text = input.value.trim();
870
+ if (!text) return;
871
+
872
+ input.value = '';
873
+ showTyping();
874
+
875
+ // 获取当前频道的 DID
876
+ const channel = channels.find(c => c.id === currentChannelId);
877
+ const channelDid = channel?.did || '';
878
+ console.log('[发送消息] 频道 DID:', channelDid);
879
+
880
+ try {
881
+ const res = await fetch('/message', {
882
+ method: 'POST',
883
+ headers: { 'Content-Type': 'application/json' },
884
+ body: JSON.stringify({
885
+ text,
886
+ channelId: currentChannelId,
887
+ channelDid
888
+ })
889
+ });
890
+
891
+ if (!res.ok) {
892
+ hideTyping();
893
+ addMessage('发送失败', 'ai');
894
+ }
895
+ } catch (err) {
896
+ hideTyping();
897
+ addMessage('连接错误', 'ai');
898
+ console.error('Send error', err);
899
+ }
900
+ }
901
+
902
+ sendBtn.addEventListener('click', sendMessage);
903
+ input.addEventListener('keydown', (e) => {
904
+ if (e.key === 'Enter' && !e.shiftKey) {
905
+ e.preventDefault();
906
+ sendMessage();
907
+ }
908
+ });
909
+
910
+ if (themeToggle) {
911
+ themeToggle.addEventListener('click', toggleTheme);
912
+ }
913
+
914
+ const apiConfigBtn = document.getElementById('api-config-btn');
915
+ if (apiConfigBtn) {
916
+ apiConfigBtn.addEventListener('click', () => {
917
+ window.location.href = '/api-config';
918
+ });
919
+ }
920
+
921
+ if (sidebarToggle) {
922
+ sidebarToggle.addEventListener('click', toggleSidebar);
923
+ }
924
+
925
+ if (loadSessionBtn && sessionFileInput) {
926
+ loadSessionBtn.addEventListener('click', () => {
927
+ sessionFileInput.click();
928
+ });
929
+
930
+ sessionFileInput.addEventListener('change', async (e) => {
931
+ const file = e.target.files?.[0];
932
+ if (!file) return;
933
+
934
+ try {
935
+ const text = await file.text();
936
+ const session = JSON.parse(text);
937
+
938
+ if (session.messages && Array.isArray(session.messages)) {
939
+ messagesEl.innerHTML = '';
940
+ session.messages.forEach(msg => {
941
+ addMessage(msg.content, msg.type, false);
942
+ });
943
+
944
+ const channelName = session.channelId || file.name.replace('.json', '');
945
+ if (channelNameEl) {
946
+ channelNameEl.textContent = channelName;
947
+ }
948
+
949
+ addMessage(`已加载 session: ${file.name}`, 'ai', false);
950
+ } else {
951
+ addMessage('无效的 session 文件格式', 'ai', false);
952
+ }
953
+ } catch (err) {
954
+ console.error('Failed to load session:', err);
955
+ addMessage('加载 session 失败: ' + err.message, 'ai', false);
956
+ }
957
+
958
+ sessionFileInput.value = '';
959
+ });
960
+ }
961
+
962
+ if (newChannelBtn) {
963
+ newChannelBtn.addEventListener('click', () => {
964
+ createChannel('智能体');
965
+ });
966
+ }
967
+
968
+ if (newChannelInput) {
969
+ newChannelInput.addEventListener('keydown', (e) => {
970
+ if (e.key === 'Enter') {
971
+ e.preventDefault();
972
+ createChannel(newChannelInput.value);
973
+ }
974
+ });
975
+ }
976
+
977
+ async function checkApiConfig() {
978
+ try {
979
+ const res = await fetch('/api/llm-config');
980
+ const config = await res.json();
981
+
982
+ // 检查是否有供应商已配置
983
+ const hasConfigured = Object.values(config.providers).some(p => p.enabled && p.apiKey);
984
+
985
+ if (!hasConfigured) {
986
+ // 显示 API 配置提示
987
+ const hint = document.createElement('div');
988
+ hint.className = 'api-config-hint';
989
+ hint.innerHTML = `
990
+ <div class="hint-icon">⚠️</div>
991
+ <div class="hint-text">
992
+ <strong>API 未配置</strong><br>
993
+ 请先配置 AI 模型才能开始对话
994
+ </div>
995
+ <button class="hint-btn" id="api-config-hint-btn">前往配置</button>
996
+ `;
997
+ document.body.appendChild(hint);
998
+
999
+ // 绑定事件处理器(避免内联 onclick)
1000
+ const hintBtn = document.getElementById('api-config-hint-btn');
1001
+ if (hintBtn) {
1002
+ hintBtn.addEventListener('click', () => {
1003
+ window.location.href = '/api-config';
1004
+ });
1005
+ }
1006
+ }
1007
+ } catch (err) {
1008
+ console.error('Failed to check API config:', err);
1009
+ }
1010
+ }
1011
+
1012
+ async function init() {
1013
+ const themeData = await loadTheme();
1014
+ currentAgentId = themeData.agentId || `agent_${generateId().substring(0, 8)}`;
1015
+
1016
+ if (!themeData.agentId) {
1017
+ await saveTheme(themeData.theme, currentAgentId);
1018
+ }
1019
+
1020
+ await loadChannels();
1021
+ await checkApiConfig();
1022
+
1023
+ if (channels.length > 0) {
1024
+ await selectChannel(channels[0].id);
1025
+ } else {
1026
+ await createChannel('默认会话');
1027
+ }
1028
+ }
1029
+
1030
+ // P2P Network Modal - React Version
1031
+ const p2pNetworkBtn = document.getElementById('p2p-network-btn');
1032
+
1033
+ // 打开 P2P Modal (使用 React)
1034
+ if (p2pNetworkBtn) {
1035
+ p2pNetworkBtn.addEventListener('click', () => {
1036
+ if (typeof window.showP2PModal === 'function') {
1037
+ window.showP2PModal();
1038
+ }
1039
+ });
1040
+ }
1041
+
1042
+ // Task Queue
1043
+ const taskModal = document.getElementById('task-modal');
1044
+ const taskQueueBtn = document.getElementById('task-queue-btn');
1045
+ const taskModalClose = document.getElementById('task-modal-close');
1046
+ const createTaskModal = document.getElementById('create-task-modal');
1047
+ const createTaskModalClose = document.getElementById('create-task-modal-close');
1048
+ const taskList = document.getElementById('task-list');
1049
+ const taskAddBtn = document.getElementById('task-add-btn');
1050
+ const taskExecuteNextBtn = document.getElementById('task-execute-next-btn');
1051
+ const taskTypeSelect = document.getElementById('task-type');
1052
+ const taskTitleInput = document.getElementById('task-title');
1053
+ const taskDescInput = document.getElementById('task-desc');
1054
+ const taskStepsInput = document.getElementById('task-steps');
1055
+ const taskCreateBtn = document.getElementById('task-create-btn');
1056
+ const taskCancelBtn = document.getElementById('task-cancel-btn');
1057
+ const taskBadge = document.getElementById('task-badge');
1058
+
1059
+ let tasks = [];
1060
+
1061
+ async function loadTasks() {
1062
+ try {
1063
+ const res = await fetch('/api/tasks');
1064
+ if (res.ok) {
1065
+ tasks = await res.json();
1066
+ renderTasks();
1067
+ updateTaskBadge();
1068
+ }
1069
+ } catch {
1070
+ tasks = [];
1071
+ renderTasks();
1072
+ }
1073
+ }
1074
+
1075
+ function renderTasks() {
1076
+ if (!taskList) return;
1077
+
1078
+ if (tasks.length === 0) {
1079
+ taskList.innerHTML = '<div class="task-empty">暂无任务,点击上方按钮创建</div>';
1080
+ return;
1081
+ }
1082
+
1083
+ // 使用 DocumentFragment 优化性能
1084
+ const fragment = document.createDocumentFragment();
1085
+
1086
+ tasks.forEach(task => {
1087
+ const div = document.createElement('div');
1088
+ div.className = `task-item ${task.status}`;
1089
+ div.dataset.id = task.id;
1090
+
1091
+ div.innerHTML = `
1092
+ <div class="task-item-header">
1093
+ <div class="task-item-title">
1094
+ <span>${getTaskIcon(task.type)}</span>
1095
+ <span>${task.title}</span>
1096
+ </div>
1097
+ <span class="task-item-status ${task.status}">${getTaskStatusText(task.status)}</span>
1098
+ </div>
1099
+ ${task.description ? `<div class="task-item-desc">${task.description.substring(0, 100)}${task.description.length > 100 ? '...' : ''}</div>` : ''}
1100
+ <div class="task-item-progress">
1101
+ <div class="task-item-progress-bar" style="width: ${task.progress}%"></div>
1102
+ </div>
1103
+ ${task.steps && task.steps.length > 0 ? `
1104
+ <div class="task-item-steps">
1105
+ ${task.steps.map((step, i) => `
1106
+ <div class="task-item-step ${step.status}">
1107
+ ${step.status === 'completed' ? '✓' : step.status === 'running' ? '⟳' : step.status === 'failed' ? '✗' : '○'} ${i + 1}. ${step.name}
1108
+ </div>
1109
+ `).join('')}
1110
+ </div>
1111
+ ` : ''}
1112
+ <div class="task-item-actions">
1113
+ ${task.status === 'pending' ? `<button class="btn-sm btn-primary" data-action="execute">▶ 执行</button>` : ''}
1114
+ ${task.status === 'running' ? `<button class="btn-sm" disabled>执行中...</button>` : ''}
1115
+ ${task.status === 'completed' ? `<button class="btn-sm" data-action="delete">删除</button>` : ''}
1116
+ ${task.status === 'failed' ? `<button class="btn-sm btn-primary" data-action="retry">重试</button>` : ''}
1117
+ </div>
1118
+ `;
1119
+
1120
+ fragment.appendChild(div);
1121
+ });
1122
+
1123
+ taskList.innerHTML = '';
1124
+ taskList.appendChild(fragment);
1125
+
1126
+ // 绑定事件处理器(避免内联 onclick)
1127
+ taskList.querySelectorAll('[data-action]').forEach(btn => {
1128
+ btn.addEventListener('click', (e) => {
1129
+ e.stopPropagation();
1130
+ const item = e.currentTarget.closest('.task-item');
1131
+ const taskId = item?.dataset.id;
1132
+ if (!taskId) return;
1133
+
1134
+ const action = e.currentTarget.dataset.action;
1135
+ switch (action) {
1136
+ case 'execute': executeTask(taskId); break;
1137
+ case 'delete': deleteTask(taskId); break;
1138
+ case 'retry': retryTask(taskId); break;
1139
+ }
1140
+ });
1141
+ });
1142
+ }
1143
+
1144
+ function getTaskIcon(type) {
1145
+ switch (type) {
1146
+ case 'chat': return '💬';
1147
+ case 'read': return '📄';
1148
+ case 'summarize': return '📝';
1149
+ case 'improve': return '✏️';
1150
+ case 'workflow': return '🔄';
1151
+ default: return '📋';
1152
+ }
1153
+ }
1154
+
1155
+ function getTaskStatusText(status) {
1156
+ switch (status) {
1157
+ case 'pending': return '待执行';
1158
+ case 'running': return '执行中';
1159
+ case 'completed': return '已完成';
1160
+ case 'failed': return '失败';
1161
+ case 'paused': return '已暂停';
1162
+ default: return status;
1163
+ }
1164
+ }
1165
+
1166
+ function updateTaskBadge() {
1167
+ if (!taskBadge) return;
1168
+ const pending = tasks.filter(t => t.status === 'pending').length;
1169
+ if (pending > 0) {
1170
+ taskBadge.textContent = pending.toString();
1171
+ taskBadge.style.display = 'block';
1172
+ } else {
1173
+ taskBadge.style.display = 'none';
1174
+ }
1175
+ }
1176
+
1177
+ function showTaskModal() {
1178
+ if (taskModal) {
1179
+ taskModal.classList.add('active');
1180
+ loadTasks();
1181
+ }
1182
+ }
1183
+
1184
+ function hideTaskModal() {
1185
+ if (taskModal) {
1186
+ taskModal.classList.remove('active');
1187
+ }
1188
+ }
1189
+
1190
+ function showCreateTaskModal() {
1191
+ if (createTaskModal) {
1192
+ createTaskModal.classList.add('active');
1193
+ if (taskTitleInput) taskTitleInput.value = '';
1194
+ if (taskDescInput) taskDescInput.value = '';
1195
+ if (taskStepsInput) taskStepsInput.value = '';
1196
+ }
1197
+ }
1198
+
1199
+ function hideCreateTaskModal() {
1200
+ if (createTaskModal) {
1201
+ createTaskModal.classList.remove('active');
1202
+ }
1203
+ }
1204
+
1205
+ async function createTask() {
1206
+ const type = taskTypeSelect?.value || 'chat';
1207
+ const title = taskTitleInput?.value?.trim();
1208
+ const description = taskDescInput?.value?.trim();
1209
+
1210
+ if (!title) {
1211
+ alert('请输入任务标题');
1212
+ return;
1213
+ }
1214
+
1215
+ const taskData = {
1216
+ type,
1217
+ title,
1218
+ description
1219
+ };
1220
+
1221
+ if (type === 'workflow' && taskStepsInput?.value) {
1222
+ taskData.steps = taskStepsInput.value.split('\n').filter(s => s.trim());
1223
+ }
1224
+
1225
+ try {
1226
+ const res = await fetch('/api/tasks', {
1227
+ method: 'POST',
1228
+ headers: { 'Content-Type': 'application/json' },
1229
+ body: JSON.stringify(taskData)
1230
+ });
1231
+
1232
+ if (res.ok) {
1233
+ const task = await res.json();
1234
+
1235
+ // 自动执行任务
1236
+ if (currentChannelId) {
1237
+ await fetch(`/api/tasks/${task.id}/execute`, {
1238
+ method: 'POST',
1239
+ headers: { 'Content-Type': 'application/json' },
1240
+ body: JSON.stringify({ channelId: currentChannelId })
1241
+ });
1242
+ }
1243
+
1244
+ hideCreateTaskModal();
1245
+ await loadTasks();
1246
+ }
1247
+ } catch (err) {
1248
+ console.error('Failed to create task:', err);
1249
+ }
1250
+ }
1251
+
1252
+ async function executeTask(taskId) {
1253
+ if (!currentChannelId) {
1254
+ alert('请先选择一个频道');
1255
+ return;
1256
+ }
1257
+
1258
+ try {
1259
+ await fetch(`/api/tasks/${taskId}/execute`, {
1260
+ method: 'POST',
1261
+ headers: { 'Content-Type': 'application/json' },
1262
+ body: JSON.stringify({ channelId: currentChannelId })
1263
+ });
1264
+ await loadTasks();
1265
+ } catch (err) {
1266
+ console.error('Failed to execute task:', err);
1267
+ }
1268
+ }
1269
+
1270
+ async function retryTask(taskId) {
1271
+ const tasks = await (await fetch('/api/tasks')).json();
1272
+ const task = tasks.find(t => t.id === taskId);
1273
+ if (task) {
1274
+ task.status = 'pending';
1275
+ task.error = undefined;
1276
+ await fetch(`/api/tasks/${taskId}`, {
1277
+ method: 'PATCH',
1278
+ headers: { 'Content-Type': 'application/json' },
1279
+ body: JSON.stringify({ status: 'pending' })
1280
+ });
1281
+ await executeTask(taskId);
1282
+ }
1283
+ }
1284
+
1285
+ async function deleteTask(taskId) {
1286
+ try {
1287
+ await fetch(`/api/tasks/${taskId}`, { method: 'DELETE' });
1288
+ await loadTasks();
1289
+ } catch (err) {
1290
+ console.error('Failed to delete task:', err);
1291
+ }
1292
+ }
1293
+
1294
+ async function executeNextTask() {
1295
+ if (!currentChannelId) {
1296
+ alert('请先选择一个频道');
1297
+ return;
1298
+ }
1299
+
1300
+ try {
1301
+ const res = await fetch('/api/tasks/execute-next', {
1302
+ method: 'POST',
1303
+ headers: { 'Content-Type': 'application/json' },
1304
+ body: JSON.stringify({ channelId: currentChannelId })
1305
+ });
1306
+
1307
+ if (res.ok) {
1308
+ const data = await res.json();
1309
+ if (!data.ok) {
1310
+ addMessage(data.message || '没有待执行的任务', 'ai');
1311
+ }
1312
+ await loadTasks();
1313
+ }
1314
+ } catch (err) {
1315
+ console.error('Failed to execute next task:', err);
1316
+ }
1317
+ }
1318
+
1319
+ // Task modal events
1320
+ if (taskQueueBtn) {
1321
+ taskQueueBtn.addEventListener('click', showTaskModal);
1322
+ }
1323
+
1324
+ if (taskModalClose) {
1325
+ taskModalClose.addEventListener('click', hideTaskModal);
1326
+ }
1327
+
1328
+ if (taskModal) {
1329
+ taskModal.addEventListener('click', (e) => {
1330
+ if (e.target === taskModal) {
1331
+ hideTaskModal();
1332
+ }
1333
+ });
1334
+ }
1335
+
1336
+ if (taskAddBtn) {
1337
+ taskAddBtn.addEventListener('click', showCreateTaskModal);
1338
+ }
1339
+
1340
+ if (taskExecuteNextBtn) {
1341
+ taskExecuteNextBtn.addEventListener('click', executeNextTask);
1342
+ }
1343
+
1344
+ if (taskCancelBtn) {
1345
+ taskCancelBtn.addEventListener('click', hideCreateTaskModal);
1346
+ }
1347
+
1348
+ if (createTaskModalClose) {
1349
+ createTaskModalClose.addEventListener('click', hideCreateTaskModal);
1350
+ }
1351
+
1352
+ if (createTaskModal) {
1353
+ createTaskModal.addEventListener('click', (e) => {
1354
+ if (e.target === createTaskModal) {
1355
+ hideCreateTaskModal();
1356
+ }
1357
+ });
1358
+ }
1359
+
1360
+ if (taskCreateBtn) {
1361
+ taskCreateBtn.addEventListener('click', createTask);
1362
+ }
1363
+
1364
+ if (taskTypeSelect) {
1365
+ taskTypeSelect.addEventListener('change', () => {
1366
+ const workflowSteps = document.querySelector('.workflow-steps');
1367
+ if (workflowSteps) {
1368
+ workflowSteps.style.display = taskTypeSelect.value === 'workflow' ? 'block' : 'none';
1369
+ }
1370
+ });
1371
+ }
1372
+
1373
+ // Handle SSE task status updates
1374
+ const originalOnMessage = window.addEventListener ? null : null;
1375
+
1376
+ // Extend SSE handler for task updates
1377
+ const originalConnect = connect;
1378
+ connect = async function() {
1379
+ // Call original connect
1380
+ await originalConnect();
1381
+
1382
+ // Reconnect SSE for task updates
1383
+ const taskEventSource = new EventSource('/events');
1384
+
1385
+ taskEventSource.onmessage = (e) => {
1386
+ try {
1387
+ const data = JSON.parse(e.data);
1388
+ if (data.type === 'task_status') {
1389
+ loadTasks();
1390
+ }
1391
+ } catch {}
1392
+ };
1393
+
1394
+ taskEventSource.onerror = () => {
1395
+ taskEventSource.close();
1396
+ };
1397
+ };
1398
+
1399
+ // 启动应用
1400
+ init();
1401
+