@agentunion/kite 1.5.0 → 1.6.1

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 (574) hide show
  1. package/.claude/skills/kite/checklists/feature-checklist.md +496 -0
  2. package/.claude/skills/kite/references/event-patterns.md +180 -0
  3. package/.claude/skills/kite/references/health-check.md +202 -0
  4. package/.claude/skills/kite/references/http-service.md +199 -0
  5. package/.claude/skills/kite/references/module-md-spec.md +172 -0
  6. package/.claude/skills/kite/references/multi-connection.md +147 -0
  7. package/.claude/skills/kite/references/rpc-patterns.md +199 -0
  8. package/.claude/skills/kite/references/shutdown-sequence.md +146 -0
  9. package/.claude/skills/kite/references/stdin-protocol.md +147 -0
  10. package/.claude/skills/kite/references/test-center-integration.md +178 -0
  11. package/.claude/skills/kite/references/ws-lifecycle.md +301 -0
  12. package/.claude/skills/kite/skill.md +272 -0
  13. package/.claude/skills/kite/templates/go/README.md +20 -0
  14. package/.claude/skills/kite/templates/node/entry.js +134 -0
  15. package/.claude/skills/kite/templates/node/module.md +16 -0
  16. package/.claude/skills/kite/templates/node/server.js +351 -0
  17. package/.claude/skills/kite/templates/node/server_http.js +90 -0
  18. package/.claude/skills/kite/templates/python/entry.py +425 -0
  19. package/.claude/skills/kite/templates/python/module.md +26 -0
  20. package/.claude/skills/kite/templates/python/server.py +447 -0
  21. package/.claude/skills/kite/templates/python/server_http.py +433 -0
  22. package/cli.js +38 -4
  23. package/core/env_checker.py +96 -0
  24. package/docs/05-/347/237/255/344/277/241/350/256/244/350/257/201/344/270/216/347/224/250/346/210/267/344/277/241/346/201/257/346/216/245/345/217/243/346/226/207/346/241/243.md +507 -0
  25. package/docs/ACP/345/215/217/350/256/256/345/205/274/345/256/271/346/226/271/346/241/210.md +138 -0
  26. package/docs/CI/344/270/216AI/350/207/252/345/212/250/345/214/226/346/265/213/350/257/225/346/226/271/346/241/210.md +75 -0
  27. package/docs/CLI/345/274/200/345/217/221/350/256/241/345/210/222.md +595 -0
  28. package/docs/ClaudeCode/350/277/234/347/250/213/345/215/217/344/275/234/347/263/273/347/273/237-/346/212/200/346/234/257/350/257/204/344/274/260.md +535 -0
  29. package/docs/ClaudeCode/350/277/234/347/250/213/345/215/217/344/275/234/347/263/273/347/273/237/350/256/276/350/256/241.md +631 -0
  30. package/docs/Evol-App/344/275/277/347/224/250KernelClient/346/224/271/351/200/240/345/256/214/346/210/220.md +342 -0
  31. package/docs/Evol/346/216/247/345/210/266/345/217/260/346/217/222/344/273/266/345/214/226/346/236/266/346/236/204/346/246/202/350/246/201.md +604 -0
  32. package/docs/Evol/346/216/247/345/210/266/345/217/260/346/217/222/344/273/266/345/214/226/346/236/266/346/236/204/350/256/276/350/256/241.md +1708 -0
  33. package/docs/Evol/346/250/241/345/235/227/350/256/276/350/256/241/346/226/271/346/241/210.md +1154 -0
  34. package/docs/Evol/351/241/265/351/235/242/346/217/222/344/273/266/345/214/226-Evol/346/250/241/345/235/227/345/256/236/346/226/275/346/214/207/345/215/227.md +403 -0
  35. package/docs/Evol/351/241/265/351/235/242/346/217/222/344/273/266/345/214/226-/345/244/226/351/203/250/346/250/241/345/235/227/346/216/245/345/205/245/346/214/207/345/215/227.md +468 -0
  36. package/docs/HTTP-RPC/350/277/201/347/247/273/345/210/260WebSocket/350/256/241/345/210/222.md +318 -0
  37. package/docs/INDEX.md +388 -0
  38. package/docs/KITE_DOCS_GUIDE.md +33 -0
  39. package/docs/Kernel-Client-Kite-Token/346/224/257/346/214/201/345/256/236/346/226/275/345/256/214/346/210/220.md +330 -0
  40. package/docs/Kernel/344/270/273/345/212/250Ping/346/234/272/345/210/266-/346/255/243/347/241/256/345/256/236/347/216/260.md +235 -0
  41. package/docs/Kernel/344/270/273/345/212/250Ping/346/234/272/345/210/266/345/256/236/346/226/275/346/200/273/347/273/223.md +204 -0
  42. package/docs/Kite/345/256/211/350/243/205/351/227/256/351/242/230/350/247/243/345/206/263/346/226/271/346/241/210.md +362 -0
  43. package/docs/Kite/346/216/247/345/210/266/345/217/260/346/217/222/344/273/266/345/214/226/346/236/266/346/236/204/350/256/276/350/256/241-/347/273/210/346/236/201/347/233/256/346/240/207.md +721 -0
  44. package/docs/Kite/346/216/247/345/210/266/345/217/260/347/273/237/344/270/200WebSocket/346/224/271/351/200/240/346/226/271/346/241/210.md +821 -0
  45. package/docs/Kite/346/241/206/346/236/266/350/256/276/350/256/241/01-/346/241/206/346/236/266/345/256/232/344/275/215.md +12 -0
  46. package/docs/Kite/346/241/206/346/236/266/350/256/276/350/256/241/02-/346/240/270/345/277/203/346/246/202/345/277/265.md +341 -0
  47. package/docs/Kite/346/241/206/346/236/266/350/256/276/350/256/241/03-/347/263/273/347/273/237/346/236/266/346/236/204.md +257 -0
  48. package/docs/Kite/346/241/206/346/236/266/350/256/276/350/256/241/04-/346/250/241/345/235/227/350/247/204/350/214/203.md +263 -0
  49. package/docs/Kite/346/241/206/346/236/266/350/256/276/350/256/241/05-/346/240/270/345/277/203/346/265/201/347/250/213-/346/226/260/347/211/210.md +267 -0
  50. package/docs/Kite/346/241/206/346/236/266/350/256/276/350/256/241/05-/346/240/270/345/277/203/346/265/201/347/250/213.md +149 -0
  51. package/docs/Kite/346/241/206/346/236/266/350/256/276/350/256/241/06-/347/233/256/345/275/225/347/273/223/346/236/204.md +231 -0
  52. package/docs/Kite/346/241/206/346/236/266/350/256/276/350/256/241/07-/346/225/260/346/215/256/346/250/241/345/236/213.md +68 -0
  53. package/docs/Kite/346/241/206/346/236/266/350/256/276/350/256/241/08-/346/211/251/345/261/225/346/200/247.md +34 -0
  54. package/docs/Kite/346/241/206/346/236/266/350/256/276/350/256/241/09-/344/270/216/345/205/267/344/275/223/345/272/224/347/224/250/347/232/204/345/205/263/347/263/273.md +22 -0
  55. package/docs/Kite/346/241/206/346/236/266/350/256/276/350/256/241/README.md +46 -0
  56. package/docs/Kite/347/263/273/347/273/237/345/220/257/345/212/250/346/265/201/347/250/213.md +567 -0
  57. package/docs/Launcher/345/220/257/345/212/250/345/231/250/346/226/207/346/241/243.md +745 -0
  58. package/docs/Polyglot/350/277/220/350/241/214/346/227/266/344/270/216Clawdbot/345/205/274/345/256/271/346/200/247/350/256/276/350/256/241.md +321 -0
  59. package/docs/Redis/344/270/216/346/250/241/345/235/227/345/244/232/345/256/236/344/276/213/346/226/271/346/241/210.md +438 -0
  60. package/docs/Relay-Kite-Token/350/256/244/350/257/201/345/256/236/346/226/275/345/256/214/346/210/220.md +178 -0
  61. package/docs/Relay-Token/346/235/203/351/231/220/351/205/215/347/275/256/351/252/214/350/257/201.md +113 -0
  62. package/docs/Watchdog/345/201/245/345/272/267/346/243/200/346/237/245/344/270/216WebSocket-Ping/346/234/272/345/210/266/345/210/206/346/236/220.md +367 -0
  63. package/docs/Watchdog/350/265/204/346/272/220/347/233/221/346/216/247/347/255/226/347/225/245.md +92 -0
  64. package/docs/WebSocket/346/216/245/346/224/266/345/276/252/347/216/257/346/255/273/351/224/201/351/230/262/350/214/203/350/247/204/350/214/203.md +357 -0
  65. package/docs/WebSocket/350/277/236/346/216/245/351/237/247/346/200/247/344/270/216/351/207/215/350/277/236/346/234/272/345/210/266/345/256/214/346/225/264/346/226/271/346/241/210.md +531 -0
  66. package/docs/WebSocket/350/277/236/346/216/245/351/237/247/346/200/247/346/226/271/346/241/210.md +169 -0
  67. package/docs/WebSocket/351/207/215/350/277/236/346/234/272/345/210/266/346/265/213/350/257/225/346/212/245/345/221/212.md +169 -0
  68. package/docs/WebSocket/351/207/215/350/277/236/351/200/200/351/201/277/346/234/272/345/210/266/346/226/271/346/241/210.md +394 -0
  69. package/docs/Web/346/250/241/345/235/227/344/270/216Evol/346/250/241/345/235/227/351/207/215/346/236/204/345/210/206/346/236/220.md +521 -0
  70. package/docs/audit-api-guide.md +68 -0
  71. package/docs/audit-module-design.md +315 -0
  72. package/docs/audit-module-implementation-summary.md +149 -0
  73. package/docs/llm-context-design.md +52 -0
  74. package/docs/llm-test-enhancement-plan.md +970 -0
  75. package/docs/logs-api-guide.md +42 -0
  76. package/docs/npm/345/214/205Python/347/216/257/345/242/203/347/256/241/347/220/206/346/226/271/346/241/210.md +302 -0
  77. package/docs/npm/345/217/221/345/270/203/344/270/216CLI/344/275/277/347/224/250/346/214/207/345/215/227.md +245 -0
  78. package/docs/stdio/344/270/216/347/253/257/345/217/243/345/217/221/347/216/260/351/207/215/346/236/204.md +480 -0
  79. package/docs/web/346/250/241/345/235/227/344/270/255/350/275/254/346/234/215/345/212/241/350/256/276/350/256/241/346/226/271/346/241/210.md +449 -0
  80. package/docs//344/272/213/344/273/266/345/244/204/347/220/206/346/234/272/345/210/266.md +388 -0
  81. package/docs//344/272/213/344/273/266/345/244/204/347/220/206/350/247/204/350/214/203.md +113 -0
  82. package/docs//344/272/213/344/273/266/350/256/242/351/230/205/351/200/232/351/205/215/347/254/246/350/247/204/350/214/203.md +256 -0
  83. package/docs//344/272/213/344/273/266/351/230/237/345/210/227/345/274/271/346/200/247/347/256/241/347/220/206.md +449 -0
  84. package/docs//344/272/244/344/272/222/345/274/217/347/273/210/347/253/257/346/216/247/345/210/266/346/226/271/346/241/210.md +301 -0
  85. package/docs//344/273/243/347/220/206/345/220/257/345/212/250/345/231/250/344/270/216/345/256/271/345/231/250/345/214/226.md +140 -0
  86. package/docs//344/273/243/347/240/201/347/273/237/350/256/241/345/267/245/345/205/267/344/275/277/347/224/250/350/257/264/346/230/216.md +217 -0
  87. package/docs//344/274/230/351/233/205/351/200/200/345/207/272/350/247/204/350/214/203.md +362 -0
  88. package/docs//344/276/235/350/265/226/347/256/241/347/220/206/350/257/264/346/230/216.md +141 -0
  89. package/docs//344/277/256/345/244/215/346/235/203/351/231/220/351/227/256/351/242/230-evol-RPC/346/235/203/351/231/220.md +268 -0
  90. package/docs//345/210/240/351/231/244kernel-client-example/345/256/214/346/210/220.md +309 -0
  91. package/docs//345/210/240/351/231/244ws-management/345/256/214/346/210/220.md +418 -0
  92. package/docs//345/220/257/345/212/250/344/274/230/345/214/226/346/226/271/346/241/210.md +522 -0
  93. package/docs//345/220/257/345/212/250/344/276/235/350/265/226/344/270/216/346/216/222/345/272/217.md +105 -0
  94. package/docs//345/256/211/350/243/205/350/204/232/346/234/254/345/274/200/345/217/221/346/226/207/346/241/243.md +643 -0
  95. package/docs//345/256/214/346/225/264/345/220/257/345/212/250/346/265/201/347/250/213/350/256/276/350/256/241.md +452 -0
  96. package/docs//345/256/236/347/216/260/350/247/204/345/210/222.md +195 -0
  97. package/docs//345/277/203/350/267/263/346/234/272/345/210/266/351/207/215/346/236/204/346/200/273/347/273/223.md +166 -0
  98. package/docs//346/217/241/346/211/213/350/256/244/350/257/201/346/226/271/346/241/210-/345/256/211/345/205/250/345/256/241/346/237/245.md +176 -0
  99. package/docs//346/217/241/346/211/213/350/256/244/350/257/201/346/226/271/346/241/210.md +908 -0
  100. package/docs//346/226/207/346/241/243/346/233/264/346/226/260/346/270/205/345/215/225.md +83 -0
  101. package/docs//346/227/245/345/277/227/344/270/216/345/274/202/345/270/270/345/244/204/347/220/206/350/247/204/350/214/203.md +829 -0
  102. package/docs//346/227/245/345/277/227/350/260/203/350/257/225/345/256/236/346/210/230/346/214/207/345/215/227.md +25 -0
  103. package/docs//346/236/266/346/236/204/345/200/237/351/211/264/346/214/207/345/215/227.md +977 -0
  104. package/docs//346/236/266/346/236/204/346/224/271/351/200/240-/345/256/214/346/210/220/346/200/273/347/273/223.md +440 -0
  105. package/docs//346/236/266/346/236/204/347/216/260/347/212/266/344/270/216/347/273/210/346/236/201/347/233/256/346/240/207/345/257/271/346/257/224/345/210/206/346/236/220.md +508 -0
  106. package/docs//346/250/241/345/235/227/345/244/232/350/277/236/346/216/245/346/216/247/345/210/266/347/255/226/347/225/245.md +220 -0
  107. package/docs//346/250/241/345/235/227/345/256/211/350/243/205/346/234/272/345/210/266/350/256/276/350/256/241.md +500 -0
  108. package/docs//346/250/241/345/235/227/345/274/200/345/217/221/346/214/207/345/215/227.md +1824 -0
  109. package/docs//346/250/241/345/235/227/347/203/255/346/233/264/346/226/260.md +89 -0
  110. package/docs//346/250/241/345/235/227/350/277/234/347/250/213/351/203/250/347/275/262/345/274/200/345/217/221/350/247/204/350/214/203.md +460 -0
  111. package/docs//346/250/241/345/235/227/351/200/200/345/207/272/346/234/272/345/210/266/345/256/214/346/225/264/346/226/271/346/241/210.md +303 -0
  112. package/docs//346/250/241/345/235/227/351/205/215/347/275/256/345/212/240/350/275/275/344/270/216/347/203/255/351/207/215/350/275/275/350/247/204/350/214/203.md +369 -0
  113. package/docs//346/265/213/350/257/225/344/270/255/345/277/203/346/267/273/345/212/240/346/250/241/345/235/227/346/265/213/350/257/225/346/214/207/345/215/227.md +147 -0
  114. package/docs//347/211/210/346/234/254/351/224/201/345/256/232/347/216/257/345/242/203/347/256/241/347/220/206/346/226/271/346/241/210.md +331 -0
  115. package/docs//347/216/257/345/242/203/345/217/230/351/207/217/344/270/216/350/277/220/350/241/214/346/227/266/347/233/256/345/275/225/350/256/276/350/256/241.md +499 -0
  116. package/docs//347/216/257/345/242/203/347/256/241/347/220/206/345/256/214/346/225/264/346/226/271/346/241/210.md +334 -0
  117. package/docs//350/231/232/346/213/237/346/250/241/345/235/227/344/270/255/350/275/254/346/234/215/345/212/241/345/256/214/346/225/264/350/256/276/350/256/241.md +1496 -0
  118. package/docs//350/231/232/346/213/237/347/216/257/345/242/203/345/267/245/344/275/234/345/216/237/347/220/206.md +163 -0
  119. package/docs//350/256/241/345/210/222/347/256/241/347/220/206/345/231/250/344/275/277/347/224/250/346/214/207/345/215/227.md +196 -0
  120. package/docs//350/256/244/350/257/201/346/250/241/345/235/227/344/270/216Gateway/350/256/276/350/256/241/346/226/271/346/241/210.md +765 -0
  121. package/docs//350/277/234/347/250/213/346/250/241/345/235/227/350/256/276/350/256/241-/346/227/247/347/211/210.md +1117 -0
  122. package/docs//350/277/234/347/250/213/346/250/241/345/235/227/350/256/276/350/256/241.md +451 -0
  123. package/docs//351/207/215/346/236/204/346/234/272/345/210/266/346/270/205/345/215/225.md +192 -0
  124. package/docs//351/223/276/350/267/257/350/277/275/350/270/252/346/226/271/346/241/210.md +242 -0
  125. package/docs//351/231/215/347/272/247/347/255/226/347/225/245/350/256/276/350/256/241/346/226/271/346/241/210.md +618 -0
  126. package/extensions/agents/assistant/entry.py +113 -14
  127. package/extensions/agents/assistant/module.md +27 -22
  128. package/extensions/agents/assistant/server.py +291 -105
  129. package/extensions/channels/acp_channel/entry.py +114 -16
  130. package/extensions/channels/acp_channel/module.md +4 -0
  131. package/extensions/channels/acp_channel/server.py +396 -105
  132. package/extensions/channels/phone_channel/__init__.py +1 -0
  133. package/extensions/channels/phone_channel/entry.py +503 -0
  134. package/extensions/channels/phone_channel/module.md +31 -0
  135. package/extensions/channels/phone_channel/server.py +686 -0
  136. package/extensions/event_hub_bench/entry.py +55 -12
  137. package/extensions/event_hub_bench/module.md +27 -27
  138. package/extensions/services/audit/README.md +134 -0
  139. package/extensions/services/audit/collector.py +73 -0
  140. package/extensions/services/audit/entry.py +444 -0
  141. package/extensions/services/audit/module.md +66 -0
  142. package/extensions/services/audit/query_audit.py +111 -0
  143. package/extensions/services/audit/routes/__init__.py +1 -0
  144. package/extensions/services/audit/routes/routes_audit.py +113 -0
  145. package/extensions/services/audit/schemas/__init__.py +5 -0
  146. package/extensions/services/audit/schemas/audit_event.py +92 -0
  147. package/extensions/services/audit/server.py +542 -0
  148. package/extensions/services/audit/storage.py +95 -0
  149. package/extensions/services/auth/entry.py +1054 -0
  150. package/extensions/services/auth/module.md +31 -0
  151. package/extensions/services/auth/token_store.py +185 -0
  152. package/extensions/services/auth/verifiers/evol_account.py +101 -0
  153. package/extensions/services/auth/verifiers/kite_token.py +38 -0
  154. package/extensions/services/auth/verifiers/pairing_code.py +71 -0
  155. package/extensions/services/backup/entry.py +494 -197
  156. package/extensions/services/backup/module.md +4 -2
  157. package/extensions/services/dataclaw/api/__init__.py +0 -0
  158. package/extensions/services/dataclaw/api/admin.py +367 -0
  159. package/extensions/services/dataclaw/api/copyright.py +175 -0
  160. package/extensions/services/dataclaw/api/credits.py +177 -0
  161. package/extensions/services/dataclaw/api/data.py +179 -0
  162. package/extensions/services/dataclaw/api/demands.py +269 -0
  163. package/extensions/services/dataclaw/api/feeds.py +262 -0
  164. package/extensions/services/dataclaw/api/identity.py +505 -0
  165. package/extensions/services/dataclaw/api/notifications.py +104 -0
  166. package/extensions/services/dataclaw/api/reviews.py +138 -0
  167. package/extensions/services/dataclaw/api/search.py +153 -0
  168. package/extensions/services/dataclaw/api/subscriptions.py +157 -0
  169. package/extensions/services/dataclaw/config.json5 +96 -0
  170. package/extensions/services/dataclaw/core/__init__.py +0 -0
  171. package/extensions/services/dataclaw/core/auth.py +95 -0
  172. package/extensions/services/dataclaw/core/config.py +50 -0
  173. package/extensions/services/dataclaw/core/database.py +70 -0
  174. package/extensions/services/dataclaw/entry.py +416 -0
  175. package/extensions/services/dataclaw/gofeed/351/241/271/347/233/256/346/211/200/346/234/211/346/235/203/350/275/254/347/247/273/346/265/201/347/250/213/350/257/264/346/230/216.md +309 -0
  176. package/extensions/services/dataclaw/migrate.py +283 -0
  177. package/extensions/services/dataclaw/models/__init__.py +0 -0
  178. package/extensions/services/dataclaw/module.md +49 -0
  179. package/extensions/services/dataclaw/requirements.txt +18 -0
  180. package/extensions/services/dataclaw/server.py +759 -0
  181. package/extensions/services/dataclaw/services/__init__.py +0 -0
  182. package/extensions/services/dataclaw/services/agent_service.py +132 -0
  183. package/extensions/services/dataclaw/services/credit_service.py +235 -0
  184. package/extensions/services/dataclaw/services/email_service.py +140 -0
  185. package/extensions/services/dataclaw/services/feed_service.py +259 -0
  186. package/extensions/services/dataclaw/services/notification_service.py +209 -0
  187. package/extensions/services/dataclaw/services/oauth_service.py +275 -0
  188. package/extensions/services/dataclaw/services/pricing.py +102 -0
  189. package/extensions/services/dataclaw/services/quality.py +79 -0
  190. package/extensions/services/dataclaw/services/reputation.py +142 -0
  191. package/extensions/services/dataclaw/services/sms_service.py +174 -0
  192. package/extensions/services/dataclaw/static/css/common.css +853 -0
  193. package/extensions/services/dataclaw/static/css/themes/blue.css +42 -0
  194. package/extensions/services/dataclaw/static/css/themes/dark.css +42 -0
  195. package/extensions/services/dataclaw/static/css/themes/light.css +35 -0
  196. package/extensions/services/dataclaw/static/js/api.js +103 -0
  197. package/extensions/services/dataclaw/static/js/common.js +321 -0
  198. package/extensions/services/dataclaw/static/js/i18n.js +95 -0
  199. package/extensions/services/dataclaw/static/js/pages/admin.js +152 -0
  200. package/extensions/services/dataclaw/static/js/pages/dashboard.js +82 -0
  201. package/extensions/services/dataclaw/static/js/pages/feed-detail.js +180 -0
  202. package/extensions/services/dataclaw/static/js/pages/feed-manage.js +158 -0
  203. package/extensions/services/dataclaw/static/js/theme.js +46 -0
  204. package/extensions/services/dataclaw/static/locales/en-US.json +464 -0
  205. package/extensions/services/dataclaw/static/locales/ja-JP.json +464 -0
  206. package/extensions/services/dataclaw/static/locales/zh-CN.json +464 -0
  207. package/extensions/services/dataclaw/templates/admin/index.html +90 -0
  208. package/extensions/services/dataclaw/templates/base.html +136 -0
  209. package/extensions/services/dataclaw/templates/credits/balance.html +106 -0
  210. package/extensions/services/dataclaw/templates/credits/deposit.html +164 -0
  211. package/extensions/services/dataclaw/templates/credits/history.html +90 -0
  212. package/extensions/services/dataclaw/templates/dashboard.html +52 -0
  213. package/extensions/services/dataclaw/templates/demands/create.html +78 -0
  214. package/extensions/services/dataclaw/templates/demands/detail.html +136 -0
  215. package/extensions/services/dataclaw/templates/demands/list.html +94 -0
  216. package/extensions/services/dataclaw/templates/feeds/create.html +95 -0
  217. package/extensions/services/dataclaw/templates/feeds/detail.html +110 -0
  218. package/extensions/services/dataclaw/templates/feeds/list.html +110 -0
  219. package/extensions/services/dataclaw/templates/feeds/manage.html +88 -0
  220. package/extensions/services/dataclaw/templates/index.html +185 -0
  221. package/extensions/services/dataclaw/templates/login.html +246 -0
  222. package/extensions/services/dataclaw/templates/register.html +164 -0
  223. package/extensions/services/dataclaw/templates/settings/notifications.html +96 -0
  224. package/extensions/services/dataclaw/templates/settings/profile.html +167 -0
  225. package/extensions/services/dataclaw/templates/subscriptions/list.html +64 -0
  226. package/extensions/services/dataclaw/tests/__init__.py +0 -0
  227. package/extensions/services/dataclaw/tests/conftest.py +68 -0
  228. package/extensions/services/dataclaw/tests/integration/__init__.py +0 -0
  229. package/extensions/services/dataclaw/tests/integration/test_workflows.py +239 -0
  230. package/extensions/services/dataclaw/tests/unit/__init__.py +0 -0
  231. package/extensions/services/dataclaw/tests/unit/test_admin.py +70 -0
  232. package/extensions/services/dataclaw/tests/unit/test_copyright.py +63 -0
  233. package/extensions/services/dataclaw/tests/unit/test_credits.py +80 -0
  234. package/extensions/services/dataclaw/tests/unit/test_data.py +98 -0
  235. package/extensions/services/dataclaw/tests/unit/test_demands.py +106 -0
  236. package/extensions/services/dataclaw/tests/unit/test_feeds.py +98 -0
  237. package/extensions/services/dataclaw/tests/unit/test_identity.py +88 -0
  238. package/extensions/services/dataclaw/tests/unit/test_notifications.py +36 -0
  239. package/extensions/services/dataclaw/tests/unit/test_reviews.py +68 -0
  240. package/extensions/services/dataclaw/tests/unit/test_search.py +64 -0
  241. package/extensions/services/dataclaw/tests/unit/test_subscriptions.py +65 -0
  242. package/extensions/services/dataclaw/tests/unit/test_system.py +106 -0
  243. package/extensions/services/dataclaw/utils/__init__.py +0 -0
  244. package/extensions/services/dataclaw/utils/crypto.py +38 -0
  245. package/extensions/services/dataclaw/utils/id_generator.py +52 -0
  246. package/extensions/services/dataclaw/ws/__init__.py +0 -0
  247. package/extensions/services/dataclaw/ws/handler.py +163 -0
  248. package/extensions/services/dataclaw//345/215/217/350/256/2561-/351/241/271/347/233/256/346/235/241/344/273/266/346/216/210/346/235/203/344/270/216/350/202/241/346/235/203/345/257/271/344/273/267/345/215/217/350/256/256.md +243 -0
  249. package/extensions/services/dataclaw//345/215/217/350/256/2562-/351/241/271/347/233/256/350/264/255/344/271/260/346/235/203/344/270/216/345/244/226/345/214/205/345/247/224/346/211/230/345/274/200/345/217/221/345/215/217/350/256/256.md +434 -0
  250. package/extensions/services/evol/__init__.py +1 -0
  251. package/extensions/services/evol/async_http.py +551 -0
  252. package/extensions/services/evol/auth_manager.py +602 -443
  253. package/extensions/services/evol/config.json5 +16 -0
  254. package/extensions/services/evol/entry.py +568 -406
  255. package/extensions/services/evol/evol_api.py +969 -173
  256. package/extensions/services/evol/mfa_totp.py +77 -0
  257. package/extensions/services/evol/module.md +150 -32
  258. package/extensions/services/evol/nonce_pool.py +113 -0
  259. package/extensions/services/evol/oauth_manager.py +223 -0
  260. package/extensions/services/evol/pairing.py +3 -2
  261. package/extensions/services/evol/pairing_codes.jsonl +1 -0
  262. package/extensions/services/evol/relay.py +1031 -682
  263. package/extensions/services/evol/relay_config.json5 +85 -67
  264. package/extensions/services/evol/routes/routes_llm.py +231 -0
  265. package/extensions/services/evol/routes/routes_rpc.py +90 -89
  266. package/extensions/services/evol/routes/routes_test.py +11 -4
  267. package/extensions/services/evol/server.py +2426 -875
  268. package/extensions/services/evol/static/assets/CommissionView-Cs_ys6Gm.js +1 -0
  269. package/extensions/services/evol/static/assets/CommissionView-DACet_Oo.css +1 -0
  270. package/extensions/services/evol/static/assets/IframePage-DbO11U9G.js +1 -0
  271. package/extensions/services/evol/static/assets/IframePage-c572lT8i.css +1 -0
  272. package/extensions/services/evol/static/assets/TeamDetailView-DULrGD7k.css +1 -0
  273. package/extensions/services/evol/static/assets/TeamDetailView-gy_MBEqG.js +139 -0
  274. package/extensions/services/evol/static/assets/element-plus-Bd7pZkkM.js +63 -0
  275. package/extensions/services/evol/static/assets/index-CmMONKzG.css +1 -0
  276. package/extensions/services/evol/static/assets/index-D44bBe__.js +2 -0
  277. package/extensions/services/evol/static/assets/vue-vendor-DtF-__I4.js +29 -0
  278. package/extensions/services/evol/static/index.html +16 -781
  279. package/extensions/services/evol/static/logo.png +0 -0
  280. package/extensions/services/evol/stats_manager.py +243 -240
  281. package/extensions/services/evol/web/README.md +89 -0
  282. package/extensions/services/evol/web/build.bat +44 -0
  283. package/extensions/services/evol/web/index.html +13 -0
  284. package/extensions/services/evol/web/package-lock.json +1718 -0
  285. package/extensions/services/evol/web/package.json +26 -0
  286. package/extensions/services/evol/web/public/logo.png +0 -0
  287. package/extensions/services/evol/web/src/App.vue +7 -0
  288. package/extensions/services/evol/web/src/components/layout/AppHeader.vue +202 -0
  289. package/extensions/services/evol/web/src/components/layout/AppLayout.vue +61 -0
  290. package/extensions/services/evol/web/src/components/layout/AppSidebar.vue +115 -0
  291. package/extensions/services/evol/web/src/components/login/LoginPage.vue +271 -0
  292. package/extensions/services/evol/web/src/components/team/AddMemberModal.vue +181 -0
  293. package/extensions/services/evol/web/src/components/team/GroupTreeNode.vue +156 -0
  294. package/extensions/services/evol/web/src/components/team/TeamAlertConfig.vue +221 -0
  295. package/extensions/services/evol/web/src/components/team/TeamBillModal.vue +165 -0
  296. package/extensions/services/evol/web/src/components/team/TeamMembersAndGroups.vue +499 -0
  297. package/extensions/services/evol/web/src/components/team/TeamStatsPanel.vue +907 -0
  298. package/extensions/services/evol/web/src/components/team/TreeNode.vue +331 -0
  299. package/extensions/services/evol/web/src/components/team/stats/StatsExportProgress.vue +44 -0
  300. package/extensions/services/evol/web/src/components/team/stats/StatsHeader.vue +89 -0
  301. package/extensions/services/evol/web/src/components/team/stats/StatsMemberDetail.vue +415 -0
  302. package/extensions/services/evol/web/src/components/team/stats/StatsSummary.vue +42 -0
  303. package/extensions/services/evol/web/src/components/team/stats/helpers.ts +195 -0
  304. package/extensions/services/evol/web/src/components/team/stats/stats.css +741 -0
  305. package/extensions/services/evol/web/src/components/team/stats/useStatsApi.ts +114 -0
  306. package/extensions/services/evol/web/src/components/team/stats/useStatsCharts.ts +242 -0
  307. package/extensions/services/evol/web/src/components/team/stats/useStatsExport.ts +232 -0
  308. package/extensions/services/evol/web/src/composables/useFormatters.ts +42 -0
  309. package/extensions/services/evol/web/src/composables/useTheme.ts +52 -0
  310. package/extensions/services/evol/web/src/env.d.ts +7 -0
  311. package/extensions/services/evol/web/src/i18n/en.ts +361 -0
  312. package/extensions/services/evol/web/src/i18n/index.ts +36 -0
  313. package/extensions/services/evol/web/src/i18n/zh.ts +379 -0
  314. package/extensions/services/evol/web/src/main.ts +21 -0
  315. package/extensions/services/evol/web/src/router/index.ts +81 -0
  316. package/extensions/services/evol/web/src/services/kernel-client.ts +406 -0
  317. package/extensions/services/evol/web/src/stores/auth.ts +189 -0
  318. package/extensions/services/evol/web/src/stores/connection.ts +134 -0
  319. package/extensions/services/evol/web/src/stores/pages.ts +79 -0
  320. package/extensions/services/evol/web/src/styles/base.css +213 -0
  321. package/extensions/services/evol/web/src/styles/variables.css +138 -0
  322. package/extensions/services/evol/web/src/types/rpc.ts +35 -0
  323. package/extensions/services/evol/web/src/types/token.ts +87 -0
  324. package/extensions/services/evol/web/src/views/AccountView.vue +1532 -0
  325. package/extensions/services/evol/web/src/views/AiServiceView.vue +219 -0
  326. package/extensions/services/evol/web/src/views/CommissionView.vue +1220 -0
  327. package/extensions/services/evol/web/src/views/CreditsView.vue +131 -0
  328. package/extensions/services/evol/web/src/views/EndpointView.vue +163 -0
  329. package/extensions/services/evol/web/src/views/IframePage.vue +120 -0
  330. package/extensions/services/evol/web/src/views/TeamDetailView.vue +473 -0
  331. package/extensions/services/evol/web/src/views/TeamView.vue +332 -0
  332. package/extensions/services/evol/web/tsconfig.json +31 -0
  333. package/extensions/services/evol/web/tsconfig.node.json +10 -0
  334. package/extensions/services/evol/web/vite.config.ts +49 -0
  335. package/extensions/services/evolmem/__init__.py +0 -0
  336. package/extensions/services/evolmem/entry.py +387 -0
  337. package/extensions/services/evolmem/hooks/__init__.py +0 -0
  338. package/extensions/services/evolmem/hooks/assistant_stop.py +228 -0
  339. package/extensions/services/evolmem/hooks/common.py +76 -0
  340. package/extensions/services/evolmem/hooks/pre_tool_use.py +56 -0
  341. package/extensions/services/evolmem/hooks/session_end.py +133 -0
  342. package/extensions/services/evolmem/hooks/session_start.py +229 -0
  343. package/extensions/services/evolmem/hooks/user_prompt.py +122 -0
  344. package/extensions/services/evolmem/module.md +48 -0
  345. package/extensions/services/evolmem/prompts/00-server-info.md +28 -0
  346. package/extensions/services/evolmem/prompts/01-behavior.md +46 -0
  347. package/extensions/services/evolmem/prompts/02-summary-format.md +112 -0
  348. package/extensions/services/evolmem/prompts/03-file-query.md +92 -0
  349. package/extensions/services/evolmem/prompts/04-topic-stats.md +11 -0
  350. package/extensions/services/evolmem/prompts/05-recent-topics.md +84 -0
  351. package/extensions/services/evolmem/scripts/__init__.py +0 -0
  352. package/extensions/services/evolmem/scripts/extract_keywords.py +40 -0
  353. package/extensions/services/evolmem/scripts/search_topics.py +91 -0
  354. package/extensions/services/evolmem/server.py +641 -0
  355. package/extensions/services/gateway/entry.py +964 -0
  356. package/extensions/services/gateway/module.md +29 -0
  357. package/extensions/services/gateway/nonce_pool.py +65 -0
  358. package/extensions/services/gateway/relay.py +133 -0
  359. package/extensions/services/gateway/ws_server.py +285 -0
  360. package/extensions/services/kite_console/auth_manager.py +603 -0
  361. package/extensions/services/kite_console/config.json5 +19 -0
  362. package/extensions/services/kite_console/config_loader.py +117 -0
  363. package/extensions/services/kite_console/entry.py +528 -0
  364. package/extensions/services/kite_console/evol_api.py +179 -0
  365. package/extensions/services/kite_console/evol_config.json5 +29 -0
  366. package/extensions/services/kite_console/mfa_totp.py +77 -0
  367. package/extensions/services/kite_console/migrate_tokens.py +122 -0
  368. package/extensions/services/kite_console/module.md +37 -0
  369. package/extensions/services/kite_console/nonce_pool.py +113 -0
  370. package/extensions/services/kite_console/oauth_manager.py +223 -0
  371. package/extensions/services/kite_console/pairing.py +280 -0
  372. package/extensions/services/kite_console/pairing_codes.jsonl +2 -0
  373. package/extensions/services/kite_console/relay.py +1350 -0
  374. package/extensions/services/kite_console/relay_config.json5 +96 -0
  375. package/extensions/services/kite_console/routes/__init__.py +1 -0
  376. package/extensions/services/kite_console/routes/routes_llm.py +231 -0
  377. package/extensions/services/kite_console/routes/routes_proxy.py +115 -0
  378. package/extensions/services/kite_console/routes/routes_rpc.py +89 -0
  379. package/extensions/services/kite_console/routes/routes_test.py +68 -0
  380. package/extensions/services/kite_console/server.py +1742 -0
  381. package/extensions/services/{evol → kite_console}/static/css/style.css +656 -2
  382. package/extensions/services/kite_console/static/index.html +1524 -0
  383. package/extensions/services/{evol → kite_console}/static/js/dialog.js +11 -4
  384. package/extensions/services/kite_console/static/js/evol-app.js +7740 -0
  385. package/extensions/services/{evol/static/js/evol-app.js → kite_console/static/js/evol-app.js.backup} +2777 -1949
  386. package/extensions/services/kite_console/static/js/kernel-client.js +560 -0
  387. package/extensions/services/{evol/static/js/kernel-client.js → kite_console/static/js/kernel-client.js.backup} +41 -3
  388. package/extensions/services/{evol → kite_console}/static/js/registry-tests.js +7 -0
  389. package/extensions/services/kite_console/static/js/tests/ARCHITECTURE.md +67 -0
  390. package/extensions/services/kite_console/static/js/tests/README.md +140 -0
  391. package/extensions/services/kite_console/static/js/tests/index.js +161 -0
  392. package/extensions/services/kite_console/static/js/tests/integration/auth.js +120 -0
  393. package/extensions/services/kite_console/static/js/tests/integration/channel-interaction.js +188 -0
  394. package/extensions/services/kite_console/static/js/tests/integration/elastic-connection.js +115 -0
  395. package/extensions/services/kite_console/static/js/tests/integration/full-workflow.js +43 -0
  396. package/extensions/services/kite_console/static/js/tests/integration/multi-instance.js +304 -0
  397. package/extensions/services/kite_console/static/js/tests/integration/nested-rpc.js +266 -0
  398. package/extensions/services/kite_console/static/js/tests/integration/pingpong.js +25 -0
  399. package/extensions/services/kite_console/static/js/tests/integration/redis.js +227 -0
  400. package/extensions/services/kite_console/static/js/tests/integration/registry-core.js +52 -0
  401. package/extensions/services/kite_console/static/js/tests/integration/remote-deploy.js +85 -0
  402. package/extensions/services/kite_console/static/js/tests/integration/require-init.js +96 -0
  403. package/extensions/services/kite_console/static/js/tests/integration/scaling-control.js +193 -0
  404. package/extensions/services/kite_console/static/js/tests/integration/trace.js +109 -0
  405. package/extensions/services/kite_console/static/js/tests/modules/acp_channel.js +339 -0
  406. package/extensions/services/kite_console/static/js/tests/modules/auth.js +96 -0
  407. package/extensions/services/kite_console/static/js/tests/modules/backup.js +49 -0
  408. package/extensions/services/kite_console/static/js/tests/modules/gateway.js +41 -0
  409. package/extensions/services/kite_console/static/js/tests/modules/kernel.js +90 -0
  410. package/extensions/services/kite_console/static/js/tests/modules/launcher.js +75 -0
  411. package/extensions/services/kite_console/static/js/tests/modules/multi_instance.js +129 -0
  412. package/extensions/services/kite_console/static/js/tests/modules/phone_channel.js +364 -0
  413. package/extensions/services/kite_console/static/js/tests/modules/redis.js +178 -0
  414. package/extensions/services/kite_console/static/js/tests/modules/watchdog.js +60 -0
  415. package/extensions/services/kite_console/static/js/tests/modules/web.js +70 -0
  416. package/extensions/services/kite_console/static/js/tests/test-runner.js +123 -0
  417. package/extensions/services/kite_console/static/js/virtual-list.js +200 -0
  418. package/extensions/services/kite_console/static/test_kernel_client_token.html +352 -0
  419. package/extensions/services/kite_console/stats_manager.py +247 -0
  420. package/extensions/services/logs/README.md +215 -0
  421. package/extensions/services/logs/api_logger.py +37 -0
  422. package/extensions/services/logs/baseline.py +121 -0
  423. package/extensions/services/logs/cleaner.py +76 -0
  424. package/extensions/services/logs/entry.py +449 -0
  425. package/extensions/services/logs/formatter.py +129 -0
  426. package/extensions/services/logs/module.md +38 -0
  427. package/extensions/services/logs/quick_diagnostic.py +128 -0
  428. package/extensions/services/logs/routes/__init__.py +1 -0
  429. package/extensions/services/logs/routes/routes_logs.py +218 -0
  430. package/extensions/services/logs/routes/routes_logs.py.backup +173 -0
  431. package/extensions/services/logs/scanner.py +100 -0
  432. package/extensions/services/logs/searcher.py +263 -0
  433. package/extensions/services/logs/server.py +553 -0
  434. package/extensions/services/logs.zip +0 -0
  435. package/extensions/services/model_service/config.json5 +30 -0
  436. package/extensions/services/model_service/entry.py +620 -171
  437. package/extensions/services/model_service/module.md +11 -2
  438. package/extensions/services/proxy/__init__.py +0 -0
  439. package/extensions/services/proxy/aid_manager.py +419 -0
  440. package/extensions/services/proxy/auth_bridge.py +182 -0
  441. package/extensions/services/proxy/config_store.py +79 -0
  442. package/extensions/services/proxy/entry.py +528 -0
  443. package/extensions/services/proxy/evol/presenter/agentIdPresenter.py +2 -2
  444. package/extensions/services/proxy/evol/presenter/apikeyPresenter.py +18 -28
  445. package/extensions/services/proxy/evol/presenter/configPresenter.py +80 -1127
  446. package/extensions/services/proxy/evol/presenter/userPresenter.py +71 -477
  447. package/extensions/services/proxy/evol/server/claude_proxy_async.py +11 -7
  448. package/extensions/services/proxy/module.md +151 -0
  449. package/extensions/services/proxy/server.py +952 -271
  450. package/extensions/services/redis/ALIGNMENT_CHECKLIST.md +121 -0
  451. package/extensions/services/redis/ALIGNMENT_STATUS.md +548 -0
  452. package/extensions/services/redis/config.json5 +8 -0
  453. package/extensions/services/redis/entry.py +1509 -0
  454. package/extensions/services/redis/entry.py.backup +405 -0
  455. package/extensions/services/redis/module.md +48 -0
  456. package/extensions/services/redis/redis_builtin.py +332 -0
  457. package/extensions/services/redis/redis_external.py +164 -0
  458. package/extensions/services/testUi/entry.py +446 -0
  459. package/extensions/services/testUi/module.md +18 -0
  460. package/extensions/services/testUi/ui/cards.html +131 -0
  461. package/extensions/services/testUi/ui/index.html +22 -0
  462. package/extensions/services/testUi/ui/particles.html +143 -0
  463. package/extensions/services/watchdog/entry.py +1258 -793
  464. package/extensions/services/watchdog/module.md +2 -0
  465. package/extensions/services/watchdog/monitor.py +465 -87
  466. package/extensions/services/web/auth_manager.py +602 -0
  467. package/extensions/services/web/config.json5 +11 -0
  468. package/extensions/services/web/entry.py +598 -478
  469. package/extensions/services/web/mfa_totp.py +77 -0
  470. package/extensions/services/web/module.md +16 -13
  471. package/extensions/services/web/nonce_pool.py +113 -0
  472. package/extensions/services/web/oauth_manager.py +223 -0
  473. package/extensions/services/web/pairing.py +3 -2
  474. package/extensions/services/web/pairing_codes.jsonl +1 -0
  475. package/extensions/services/web/relay.py +442 -63
  476. package/extensions/services/web/relay_config.json5 +1 -2
  477. package/extensions/services/web/routes/routes_rpc.py +6 -6
  478. package/extensions/services/web/server.py +360 -173
  479. package/extensions/services/web/static/index.html +1752 -1738
  480. package/extensions/services/web/static/js/app.js +32 -0
  481. package/extensions/services/web/static/js/kernel-client.js +48 -9
  482. package/extensions/services/web/vendor/bluetooth/audio.py +1 -1
  483. package/extensions/services/web/vendor/config.py +2 -2
  484. package/extensions/services/web/vendor/storage/identity.py +1 -1
  485. package/kernel/entry.py +77 -23
  486. package/kernel/event_hub.py +1122 -74
  487. package/kernel/module.md +2 -1
  488. package/kernel/registry_store.py +208 -11
  489. package/kernel/rpc_router.py +1400 -491
  490. package/kernel/server.py +1021 -134
  491. package/kite_cli/__init__.py +9 -1
  492. package/kite_cli/builders/__init__.py +4 -0
  493. package/kite_cli/builders/base.py +67 -0
  494. package/kite_cli/builders/custom.py +31 -0
  495. package/kite_cli/builders/detector.py +56 -0
  496. package/kite_cli/builders/go.py +34 -0
  497. package/kite_cli/builders/gradle.py +41 -0
  498. package/kite_cli/builders/maven.py +36 -0
  499. package/kite_cli/builders/npm.py +44 -0
  500. package/kite_cli/builders/python.py +37 -0
  501. package/kite_cli/commands/BUILD_GUIDE.md +109 -0
  502. package/kite_cli/commands/build.py +142 -0
  503. package/kite_cli/commands/check.py +60 -0
  504. package/kite_cli/commands/config.py +156 -0
  505. package/kite_cli/commands/deps.py +58 -0
  506. package/kite_cli/commands/deps_install.py +7 -7
  507. package/kite_cli/commands/disable.py +162 -0
  508. package/kite_cli/commands/enable.py +162 -0
  509. package/kite_cli/commands/export.py +96 -0
  510. package/kite_cli/commands/import_cmd.py +110 -0
  511. package/kite_cli/commands/install.py +50 -23
  512. package/kite_cli/commands/install_skill.py +107 -0
  513. package/kite_cli/commands/list.py +128 -31
  514. package/kite_cli/commands/outdated.py +202 -0
  515. package/kite_cli/commands/search.py +33 -17
  516. package/kite_cli/commands/update.py +115 -2
  517. package/kite_cli/commands/venv_setup.py +6 -6
  518. package/kite_cli/commands/why.py +48 -0
  519. package/kite_cli/core/config_manager.py +145 -0
  520. package/kite_cli/core/downloader.py +32 -2
  521. package/kite_cli/main.py +153 -7
  522. package/kite_cli/utils/colors.py +153 -0
  523. package/kite_cli/utils/dependency_graph.py +209 -0
  524. package/kite_cli/utils/process.py +55 -0
  525. package/kite_cli/utils/progress.py +207 -0
  526. package/kite_cli/utils/table.py +101 -0
  527. package/launcher/count_lines.py +192 -43
  528. package/launcher/entry.py +4543 -2802
  529. package/launcher/logging_setup.py +54 -1
  530. package/launcher/module.md +32 -6
  531. package/launcher/module_scanner.py +93 -20
  532. package/launcher/process_manager.py +355 -76
  533. package/main.py +6 -0
  534. package/package.json +4 -1
  535. package/requirements.txt +41 -38
  536. package/scripts/auto-fix-deps.py +128 -0
  537. package/scripts/env-manager.js +25 -2
  538. package/scripts/final-test.js +78 -0
  539. package/scripts/setup-python-env.js +700 -191
  540. package/scripts/test-alluser.js +48 -0
  541. package/scripts/test-different-version.js +86 -0
  542. package/scripts/test-direct.js +63 -0
  543. package/scripts/test-extract-installer.js +28 -0
  544. package/scripts/test-install-log.js +54 -0
  545. package/scripts/test-installer.js +39 -0
  546. package/scripts/test-integration.js +250 -0
  547. package/scripts/test-real-install.js +210 -0
  548. package/scripts/test-targetdir.js +49 -0
  549. package/scripts/test-venv-real.js +47 -0
  550. package/scripts/test-venv-simple.js +57 -0
  551. package/scripts/test-wait.js +49 -0
  552. package/scripts/test-with-log.js +63 -0
  553. package/extensions/services/evol/config.yaml +0 -149
  554. package/extensions/services/evol/routes/routes_management_ws.py +0 -127
  555. package/extensions/services/evol/static/index_evol.html +0 -14
  556. package/extensions/services/evol/static/js/app.js +0 -6304
  557. package/extensions/services/evol/static/js/auth.js +0 -326
  558. package/extensions/services/evol/static/js/evol-app-fixed.js +0 -50
  559. package/extensions/services/evol/static/js/evol-app.js.bak +0 -1800
  560. package/extensions/services/evol/static/js/kernel-client-example.js +0 -228
  561. package/extensions/services/evol/static/js/main.js +0 -141
  562. package/extensions/services/evol/static/js/stats.js +0 -217
  563. package/extensions/services/evol/static/js/token-manager.js +0 -175
  564. package/extensions/services/proxy/CHANGELOG_20260308.md +0 -258
  565. package/extensions/services/proxy/_fix_prints.py +0 -133
  566. package/extensions/services/proxy/_fix_prints2.py +0 -87
  567. package/extensions/services/proxy/console_auth.py +0 -109
  568. package/extensions/services/proxy/logs/websocket.log +0 -260
  569. package/extensions/services/proxy/main.py +0 -240
  570. package/extensions/services/proxy/requirements.txt +0 -13
  571. package/extensions/services/web/config.yaml +0 -149
  572. /package/extensions/services/{evol → kite_console}/static/pairing.html +0 -0
  573. /package/extensions/services/{evol → kite_console}/static/test_registry.html +0 -0
  574. /package/extensions/services/{evol → kite_console}/static/test_relay.html +0 -0
@@ -1,406 +1,568 @@
1
- """
2
- Evol module entry point.
3
- Reads boot_info from stdin, starts HTTP server for Evol account management.
4
- """
5
-
6
- import builtins
7
- import json
8
- import os
9
- import re
10
- import socket
11
- import sys
12
- import threading
13
- import time
14
- import traceback
15
- from datetime import datetime, timezone
16
-
17
- import uvicorn
18
-
19
-
20
- # ── Module configuration ──
21
-
22
- def _load_module_config() -> dict:
23
- """Load module configuration from module.md frontmatter."""
24
- _this_dir = os.path.dirname(os.path.abspath(__file__))
25
- module_md = os.path.join(_this_dir, "module.md")
26
-
27
- project_root = os.environ.get("KITE_PROJECT", "")
28
- if project_root and _this_dir.startswith(project_root):
29
- rel_path = os.path.relpath(_this_dir, project_root)
30
- else:
31
- rel_path = _this_dir
32
-
33
- result = {
34
- "name": "",
35
- "preferred_port": 0,
36
- "advertise_ip": "0.0.0.0"
37
- }
38
-
39
- if not os.path.exists(module_md):
40
- print(f"[{rel_path}] ERROR: Invalid module configuration in module.md")
41
- print(f" Path: {rel_path}/module.md")
42
- print(f" Reason: File not found")
43
- sys.exit(1)
44
-
45
- try:
46
- with open(module_md, encoding="utf-8") as f:
47
- text = f.read()
48
-
49
- m = re.match(r'^---\s*\n(.*?)\n---', text, re.DOTALL)
50
- if not m:
51
- print(f"[{rel_path}] ERROR: Invalid module configuration in module.md")
52
- print(f" Path: {rel_path}/module.md")
53
- print(f" Reason: Missing YAML frontmatter")
54
- sys.exit(1)
55
-
56
- import yaml
57
- fm = yaml.safe_load(m.group(1)) or {}
58
-
59
- if "name" not in fm:
60
- print(f"[{rel_path}] ERROR: Invalid module configuration in module.md")
61
- print(f" Path: {rel_path}/module.md")
62
- print(f" Reason: Missing 'name' field")
63
- sys.exit(1)
64
-
65
- raw_name = str(fm["name"]).strip()
66
-
67
- if not raw_name:
68
- print(f"[{rel_path}] ERROR: Invalid module configuration in module.md")
69
- print(f" Path: {rel_path}/module.md")
70
- print(f" Reason: Empty module name")
71
- sys.exit(1)
72
-
73
- sanitized = re.sub(r'[^a-zA-Z0-9_\-]', '', raw_name)
74
-
75
- if sanitized != raw_name:
76
- invalid_chars = ''.join(sorted(set(c for c in raw_name if c not in sanitized)))
77
- print(f"[{rel_path}] ERROR: Invalid module configuration in module.md")
78
- print(f" Path: {rel_path}/module.md")
79
- print(f" Reason: Invalid characters in name '{raw_name}': {repr(invalid_chars)}")
80
- sys.exit(1)
81
-
82
- result["name"] = sanitized
83
-
84
- if "preferred_port" in fm:
85
- try:
86
- result["preferred_port"] = int(fm["preferred_port"])
87
- except (ValueError, TypeError):
88
- pass
89
-
90
- if "advertise_ip" in fm:
91
- result["advertise_ip"] = str(fm["advertise_ip"])
92
-
93
- except SystemExit:
94
- raise
95
- except Exception as e:
96
- print(f"[{rel_path}] ERROR: Failed to read module.md: {e}")
97
- sys.exit(1)
98
-
99
- return result
100
-
101
- _module_config = _load_module_config()
102
- MODULE_NAME = _module_config["name"]
103
-
104
-
105
- # ── Safe stdout/stderr ──
106
-
107
- class _SafeWriter:
108
- def __init__(self, stream):
109
- self._stream = stream
110
-
111
- def write(self, s):
112
- try:
113
- self._stream.write(s)
114
- except (BrokenPipeError, OSError):
115
- pass
116
-
117
- def flush(self):
118
- try:
119
- self._stream.flush()
120
- except (BrokenPipeError, OSError):
121
- pass
122
-
123
- def __getattr__(self, name):
124
- return getattr(self._stream, name)
125
-
126
- sys.stdout = _SafeWriter(sys.stdout)
127
- sys.stderr = _SafeWriter(sys.stderr)
128
-
129
-
130
- # ── Timestamped print + log file writer ──
131
-
132
- _builtin_print = builtins.print
133
- _start_ts = time.monotonic()
134
- _last_ts = time.monotonic()
135
- _ANSI_RE = re.compile(r"\033\[[0-9;]*m")
136
- _log_lock = threading.Lock()
137
- _log_latest_path = None
138
- _log_daily_path = None
139
- _log_daily_date = ""
140
- _log_dir = None
141
- _crash_log_path = None
142
-
143
- def _strip_ansi(s: str) -> str:
144
- return _ANSI_RE.sub("", s)
145
-
146
- def _resolve_daily_log_path():
147
- """Resolve daily log path based on current date."""
148
- global _log_daily_path, _log_daily_date
149
- if not _log_dir:
150
- return
151
- today = datetime.now().strftime("%Y-%m-%d")
152
- if today == _log_daily_date and _log_daily_path:
153
- return
154
- month_dir = os.path.join(_log_dir, today[:7])
155
- os.makedirs(month_dir, exist_ok=True)
156
- _log_daily_path = os.path.join(month_dir, f"{today}.log")
157
- _log_daily_date = today
158
-
159
- def _write_log(plain_line: str):
160
- """Write a plain-text line to both latest.log and daily log."""
161
- with _log_lock:
162
- if _log_latest_path:
163
- try:
164
- with open(_log_latest_path, "a", encoding="utf-8") as f:
165
- f.write(plain_line)
166
- except Exception:
167
- pass
168
- _resolve_daily_log_path()
169
- if _log_daily_path:
170
- try:
171
- with open(_log_daily_path, "a", encoding="utf-8") as f:
172
- f.write(plain_line)
173
- except Exception:
174
- pass
175
-
176
- def _write_crash(exc_type, exc_value, exc_tb, thread_name=None, severity="critical", handled=False):
177
- """Write crash record to crashes.jsonl + daily crash archive."""
178
- record = {
179
- "timestamp": datetime.now(timezone.utc).isoformat(),
180
- "module": MODULE_NAME,
181
- "thread": thread_name or threading.current_thread().name,
182
- "exception_type": exc_type.__name__ if exc_type else "Unknown",
183
- "exception_message": str(exc_value),
184
- "traceback": "".join(traceback.format_exception(exc_type, exc_value, exc_tb)),
185
- "severity": severity,
186
- "handled": handled,
187
- "process_id": os.getpid(),
188
- "platform": sys.platform,
189
- "runtime_version": f"Python {sys.version.split()[0]}",
190
- }
191
-
192
- if exc_tb:
193
- tb_entries = traceback.extract_tb(exc_tb)
194
- if tb_entries:
195
- last = tb_entries[-1]
196
- record["context"] = {
197
- "function": last.name,
198
- "file": os.path.basename(last.filename),
199
- "line": last.lineno,
200
- }
201
-
202
- line = json.dumps(record, ensure_ascii=False) + "\n"
203
-
204
- if _crash_log_path:
205
- try:
206
- with open(_crash_log_path, "a", encoding="utf-8") as f:
207
- f.write(line)
208
- except Exception:
209
- pass
210
-
211
- if _log_dir:
212
- try:
213
- today = datetime.now().strftime("%Y-%m-%d")
214
- archive_dir = os.path.join(_log_dir, "crashes", today[:7])
215
- os.makedirs(archive_dir, exist_ok=True)
216
- archive_path = os.path.join(archive_dir, f"{today}.jsonl")
217
- with open(archive_path, "a", encoding="utf-8") as f:
218
- f.write(line)
219
- except Exception:
220
- pass
221
-
222
- def _setup_exception_hooks():
223
- """Set up global exception hooks."""
224
- _orig_excepthook = sys.excepthook
225
-
226
- def _excepthook(exc_type, exc_value, exc_tb):
227
- _write_crash(exc_type, exc_value, exc_tb, severity="critical", handled=False)
228
- _orig_excepthook(exc_type, exc_value, exc_tb)
229
-
230
- sys.excepthook = _excepthook
231
-
232
- if hasattr(threading, "excepthook"):
233
- def _thread_excepthook(args):
234
- _write_crash(args.exc_type, args.exc_value, args.exc_traceback,
235
- thread_name=args.thread.name if args.thread else "unknown",
236
- severity="error", handled=False)
237
- threading.excepthook = _thread_excepthook
238
-
239
- def _tprint(*args, **kwargs):
240
- global _last_ts
241
- now = time.monotonic()
242
- elapsed = now - _start_ts
243
- delta = now - _last_ts
244
- _last_ts = now
245
-
246
- if elapsed < 1:
247
- elapsed_str = f"{elapsed * 1000:.0f}ms"
248
- elif elapsed < 100:
249
- elapsed_str = f"{elapsed:.1f}s"
250
- else:
251
- elapsed_str = f"{elapsed:.0f}s"
252
-
253
- if delta < 0.001:
254
- delta_str = ""
255
- elif delta < 1:
256
- delta_str = f"+{delta * 1000:.0f}ms"
257
- elif delta < 100:
258
- delta_str = f"+{delta:.1f}s"
259
- else:
260
- delta_str = f"+{delta:.0f}s"
261
-
262
- ts = datetime.now().strftime("%H:%M:%S.%f")[:-3]
263
-
264
- _builtin_print(*args, **kwargs)
265
-
266
- if _log_latest_path or _log_daily_path:
267
- sep = kwargs.get("sep", " ")
268
- end = kwargs.get("end", "\n")
269
- text = sep.join(str(a) for a in args)
270
- prefix = f"[{elapsed_str:>6}] {ts} {delta_str:>8} [{MODULE_NAME}] "
271
- _write_log(prefix + _strip_ansi(text) + end)
272
-
273
- builtins.print = _tprint
274
-
275
-
276
- # ── Path setup ──
277
-
278
- _this_dir = os.path.dirname(os.path.abspath(__file__))
279
- _project_root = os.environ.get("KITE_PROJECT") or os.path.dirname(os.path.dirname(os.path.dirname(_this_dir)))
280
- if _project_root not in sys.path:
281
- sys.path.insert(0, _project_root)
282
-
283
-
284
- # ── Import server ──
285
-
286
- from extensions.services.evol.server import EvolServer
287
-
288
-
289
- def _fmt_elapsed(t0: float) -> str:
290
- d = time.monotonic() - t0
291
- if d < 1:
292
- return f"{d * 1000:.0f}ms"
293
- if d < 10:
294
- return f"{d:.1f}s"
295
- return f"{d:.0f}s"
296
-
297
-
298
- def _bind_port(preferred: int, host: str, max_attempts: int = 10) -> int | None:
299
- """Try to bind to preferred port, then port+1, port+2, ..."""
300
- if not preferred:
301
- with socket.socket(socket.AF_INET, socket.SOCK_STREAM) as s:
302
- s.bind((host, 0))
303
- return s.getsockname()[1]
304
-
305
- for attempt in range(max_attempts):
306
- port = preferred + attempt
307
- try:
308
- with socket.socket(socket.AF_INET, socket.SOCK_STREAM) as s:
309
- s.bind((host, port))
310
- if attempt > 0:
311
- print(f"Bound to port {port} (preferred {preferred} was occupied)")
312
- return port
313
- except OSError:
314
- if attempt < max_attempts - 1:
315
- continue
316
- else:
317
- print(f"ERROR: Failed to bind port after {max_attempts} attempts")
318
- return None
319
-
320
- return None
321
-
322
-
323
- def main():
324
- # Initialize log file paths
325
- global _log_dir, _log_latest_path, _crash_log_path
326
- module_data = os.environ.get("KITE_MODULE_DATA")
327
- if module_data:
328
- _log_dir = os.path.join(module_data, "log")
329
- os.makedirs(_log_dir, exist_ok=True)
330
- suffix = os.environ.get("KITE_INSTANCE_SUFFIX", "")
331
-
332
- _log_latest_path = os.path.join(_log_dir, f"latest{suffix}.log")
333
- try:
334
- with open(_log_latest_path, "w", encoding="utf-8") as f:
335
- pass
336
- except Exception:
337
- _log_latest_path = None
338
-
339
- _crash_log_path = os.path.join(_log_dir, f"crashes{suffix}.jsonl")
340
- try:
341
- with open(_crash_log_path, "w", encoding="utf-8") as f:
342
- pass
343
- except Exception:
344
- _crash_log_path = None
345
-
346
- _resolve_daily_log_path()
347
-
348
- _setup_exception_hooks()
349
-
350
- _t0 = time.monotonic()
351
-
352
- # Read boot_info from stdin
353
- token = ""
354
- try:
355
- line = sys.stdin.readline().strip()
356
- if line:
357
- boot_info = json.loads(line)
358
- token = boot_info.get("token", "")
359
- except Exception:
360
- pass
361
-
362
- kernel_port = int(os.environ.get("KITE_KERNEL_PORT", "0"))
363
-
364
- if not token or not kernel_port:
365
- print("ERROR: Missing token or KITE_KERNEL_PORT")
366
- sys.exit(1)
367
-
368
- print(f"Token received ({len(token)} chars), kernel port: {kernel_port} ({_fmt_elapsed(_t0)})")
369
-
370
- host = _module_config["advertise_ip"]
371
- port = _bind_port(_module_config["preferred_port"], host)
372
-
373
- if port is None:
374
- print("ERROR: Cannot bind to any port, exiting")
375
- sys.exit(1)
376
-
377
- server = EvolServer(
378
- token=token,
379
- kernel_port=kernel_port,
380
- host=host,
381
- port=port,
382
- boot_t0=_t0,
383
- )
384
-
385
- display_host = "localhost" if host == "0.0.0.0" else host
386
- url = f"http://{display_host}:{port}"
387
- print(f"Starting on {host}:{port} ({_fmt_elapsed(_t0)})")
388
- print(f"\033[32m✓ Evol UI ready: {url}\033[0m")
389
-
390
- try:
391
- config = uvicorn.Config(server.app, host=host, port=port, log_level="warning")
392
- uvi_server = uvicorn.Server(config)
393
- server._uvicorn_server = uvi_server
394
- uvi_server.run()
395
- except Exception as e:
396
- print(f"ERROR: {e}")
397
- traceback.print_exc()
398
- sys.exit(1)
399
-
400
- # Check if server requested exit with non-zero code
401
- if server._exit_code != 0:
402
- sys.exit(server._exit_code)
403
-
404
-
405
- if __name__ == "__main__":
406
- main()
1
+ """
2
+ Evol module entry point.
3
+ Reads boot_info from stdin, starts HTTP server for Evol account management.
4
+ """
5
+
6
+ import builtins
7
+ import json
8
+ import os
9
+ import re
10
+ import socket
11
+ import sys
12
+ import threading
13
+ import time
14
+ import traceback
15
+ from datetime import datetime, timezone
16
+
17
+ import uvicorn
18
+
19
+
20
+ # ── Module configuration ──
21
+
22
+ def _load_module_config() -> dict:
23
+ """Load module configuration from module.md frontmatter."""
24
+ _this_dir = os.path.dirname(os.path.abspath(__file__))
25
+ module_md = os.path.join(_this_dir, "module.md")
26
+
27
+ project_root = os.environ.get("KITE_PROJECT", "")
28
+ if project_root and _this_dir.startswith(project_root):
29
+ rel_path = os.path.relpath(_this_dir, project_root)
30
+ else:
31
+ rel_path = _this_dir
32
+
33
+ result = {
34
+ "name": "",
35
+ "preferred_port": 0,
36
+ "advertise_ip": "0.0.0.0"
37
+ }
38
+
39
+ if not os.path.exists(module_md):
40
+ print(f"[{rel_path}] ERROR: Invalid module configuration in module.md")
41
+ print(f" Path: {rel_path}/module.md")
42
+ print(f" Reason: File not found")
43
+ sys.exit(1)
44
+
45
+ try:
46
+ with open(module_md, encoding="utf-8") as f:
47
+ text = f.read()
48
+
49
+ m = re.match(r'^---\s*\n(.*?)\n---', text, re.DOTALL)
50
+ if not m:
51
+ print(f"[{rel_path}] ERROR: Invalid module configuration in module.md")
52
+ print(f" Path: {rel_path}/module.md")
53
+ print(f" Reason: Missing YAML frontmatter")
54
+ sys.exit(1)
55
+
56
+ import yaml
57
+ fm = yaml.safe_load(m.group(1)) or {}
58
+
59
+ if "name" not in fm:
60
+ print(f"[{rel_path}] ERROR: Invalid module configuration in module.md")
61
+ print(f" Path: {rel_path}/module.md")
62
+ print(f" Reason: Missing 'name' field")
63
+ sys.exit(1)
64
+
65
+ raw_name = str(fm["name"]).strip()
66
+
67
+ if not raw_name:
68
+ print(f"[{rel_path}] ERROR: Invalid module configuration in module.md")
69
+ print(f" Path: {rel_path}/module.md")
70
+ print(f" Reason: Empty module name")
71
+ sys.exit(1)
72
+
73
+ sanitized = re.sub(r'[^a-zA-Z0-9_\-]', '', raw_name)
74
+
75
+ if sanitized != raw_name:
76
+ invalid_chars = ''.join(sorted(set(c for c in raw_name if c not in sanitized)))
77
+ print(f"[{rel_path}] ERROR: Invalid module configuration in module.md")
78
+ print(f" Path: {rel_path}/module.md")
79
+ print(f" Reason: Invalid characters in name '{raw_name}': {repr(invalid_chars)}")
80
+ sys.exit(1)
81
+
82
+ result["name"] = sanitized
83
+
84
+ if "preferred_port" in fm:
85
+ try:
86
+ result["preferred_port"] = int(fm["preferred_port"])
87
+ except (ValueError, TypeError):
88
+ pass
89
+
90
+ if "advertise_ip" in fm:
91
+ result["advertise_ip"] = str(fm["advertise_ip"])
92
+
93
+ # max_connections(弹性连接上限)
94
+ try:
95
+ result["max_connections"] = max(1, min(10, int(fm.get("max_connections", 1))))
96
+ except (ValueError, TypeError):
97
+ result["max_connections"] = 1
98
+ # business_config(从 businesses 数组取第一个 config_file)
99
+ businesses = fm.get("businesses")
100
+ if isinstance(businesses, list) and businesses:
101
+ cfg_file = businesses[0].get("config_file") if isinstance(businesses[0], dict) else None
102
+ if cfg_file:
103
+ result["business_config"] = str(cfg_file).strip()
104
+
105
+ except SystemExit:
106
+ raise
107
+ except Exception as e:
108
+ print(f"[{rel_path}] ERROR: Failed to read module.md: {e}")
109
+ sys.exit(1)
110
+
111
+ return result
112
+
113
+ _module_config = _load_module_config()
114
+ MODULE_NAME = _module_config["name"]
115
+
116
+
117
+ # ── Safe stdout/stderr ──
118
+
119
+ class _SafeWriter:
120
+ def __init__(self, stream):
121
+ self._stream = stream
122
+
123
+ def write(self, s):
124
+ try:
125
+ self._stream.write(s)
126
+ except (BrokenPipeError, OSError):
127
+ pass
128
+
129
+ def flush(self):
130
+ try:
131
+ self._stream.flush()
132
+ except (BrokenPipeError, OSError):
133
+ pass
134
+
135
+ def __getattr__(self, name):
136
+ return getattr(self._stream, name)
137
+
138
+ sys.stdout = _SafeWriter(sys.stdout)
139
+ sys.stderr = _SafeWriter(sys.stderr)
140
+
141
+
142
+ # ── Timestamped print + log file writer ──
143
+
144
+ _builtin_print = builtins.print
145
+ _start_ts = time.monotonic()
146
+ _last_ts = time.monotonic()
147
+ _ANSI_RE = re.compile(r"\033\[[0-9;]*m")
148
+ _log_lock = threading.Lock()
149
+ _log_latest_path = None
150
+ _log_daily_path = None
151
+ _log_daily_date = ""
152
+ _log_dir = None
153
+ _crash_log_path = None
154
+
155
+ def _strip_ansi(s: str) -> str:
156
+ return _ANSI_RE.sub("", s)
157
+
158
+ def _resolve_daily_log_path():
159
+ """Resolve daily log path based on current date."""
160
+ global _log_daily_path, _log_daily_date
161
+ if not _log_dir:
162
+ return
163
+ today = datetime.now().strftime("%Y-%m-%d")
164
+ if today == _log_daily_date and _log_daily_path:
165
+ return
166
+ month_dir = os.path.join(_log_dir, today[:7])
167
+ os.makedirs(month_dir, exist_ok=True)
168
+ _log_daily_path = os.path.join(month_dir, f"{today}.log")
169
+ _log_daily_date = today
170
+
171
+ def _write_log(plain_line: str):
172
+ """Write a plain-text line to both latest.log and daily log."""
173
+ with _log_lock:
174
+ if _log_latest_path:
175
+ try:
176
+ with open(_log_latest_path, "a", encoding="utf-8") as f:
177
+ f.write(plain_line)
178
+ except Exception:
179
+ pass
180
+ _resolve_daily_log_path()
181
+ if _log_daily_path:
182
+ try:
183
+ with open(_log_daily_path, "a", encoding="utf-8") as f:
184
+ f.write(plain_line)
185
+ except Exception:
186
+ pass
187
+
188
+ def _write_crash(exc_type, exc_value, exc_tb, thread_name=None, severity="critical", handled=False):
189
+ """Write crash record to crashes.jsonl + daily crash archive."""
190
+ record = {
191
+ "timestamp": datetime.now(timezone.utc).isoformat(),
192
+ "module": MODULE_NAME,
193
+ "thread": thread_name or threading.current_thread().name,
194
+ "exception_type": exc_type.__name__ if exc_type else "Unknown",
195
+ "exception_message": str(exc_value),
196
+ "traceback": "".join(traceback.format_exception(exc_type, exc_value, exc_tb)),
197
+ "severity": severity,
198
+ "handled": handled,
199
+ "process_id": os.getpid(),
200
+ "platform": sys.platform,
201
+ "runtime_version": f"Python {sys.version.split()[0]}",
202
+ }
203
+
204
+ if exc_tb:
205
+ tb_entries = traceback.extract_tb(exc_tb)
206
+ if tb_entries:
207
+ last = tb_entries[-1]
208
+ record["context"] = {
209
+ "function": last.name,
210
+ "file": os.path.basename(last.filename),
211
+ "line": last.lineno,
212
+ }
213
+
214
+ line = json.dumps(record, ensure_ascii=False) + "\n"
215
+
216
+ if _crash_log_path:
217
+ try:
218
+ with open(_crash_log_path, "a", encoding="utf-8") as f:
219
+ f.write(line)
220
+ except Exception:
221
+ pass
222
+
223
+ if _log_dir:
224
+ try:
225
+ today = datetime.now().strftime("%Y-%m-%d")
226
+ archive_dir = os.path.join(_log_dir, "crashes", today[:7])
227
+ os.makedirs(archive_dir, exist_ok=True)
228
+ archive_path = os.path.join(archive_dir, f"{today}.jsonl")
229
+ with open(archive_path, "a", encoding="utf-8") as f:
230
+ f.write(line)
231
+ except Exception:
232
+ pass
233
+
234
+ def _setup_exception_hooks():
235
+ """Set up global exception hooks."""
236
+ _orig_excepthook = sys.excepthook
237
+
238
+ def _excepthook(exc_type, exc_value, exc_tb):
239
+ _write_crash(exc_type, exc_value, exc_tb, severity="critical", handled=False)
240
+ _orig_excepthook(exc_type, exc_value, exc_tb)
241
+
242
+ sys.excepthook = _excepthook
243
+
244
+ if hasattr(threading, "excepthook"):
245
+ def _thread_excepthook(args):
246
+ _write_crash(args.exc_type, args.exc_value, args.exc_traceback,
247
+ thread_name=args.thread.name if args.thread else "unknown",
248
+ severity="error", handled=False)
249
+ threading.excepthook = _thread_excepthook
250
+
251
+ def _tprint(*args, **kwargs):
252
+ global _last_ts
253
+ now = time.monotonic()
254
+ elapsed = now - _start_ts
255
+ delta = now - _last_ts
256
+ _last_ts = now
257
+
258
+ if elapsed < 1:
259
+ elapsed_str = f"{elapsed * 1000:.0f}ms"
260
+ elif elapsed < 100:
261
+ elapsed_str = f"{elapsed:.1f}s"
262
+ else:
263
+ elapsed_str = f"{elapsed:.0f}s"
264
+
265
+ if delta < 0.001:
266
+ delta_str = ""
267
+ elif delta < 1:
268
+ delta_str = f"+{delta * 1000:.0f}ms"
269
+ elif delta < 100:
270
+ delta_str = f"+{delta:.1f}s"
271
+ else:
272
+ delta_str = f"+{delta:.0f}s"
273
+
274
+ ts = datetime.now().strftime("%H:%M:%S.%f")[:-3]
275
+
276
+ _builtin_print(*args, **kwargs)
277
+
278
+ if _log_latest_path or _log_daily_path:
279
+ sep = kwargs.get("sep", " ")
280
+ end = kwargs.get("end", "\n")
281
+ text = sep.join(str(a) for a in args)
282
+ prefix = f"[{elapsed_str:>6}] {ts} {delta_str:>8} [{MODULE_NAME}] "
283
+ _write_log(prefix + _strip_ansi(text) + end)
284
+
285
+ builtins.print = _tprint
286
+
287
+
288
+ # ── Path setup ──
289
+
290
+ _this_dir = os.path.dirname(os.path.abspath(__file__))
291
+ _project_root = os.environ.get("KITE_PROJECT") or os.path.dirname(os.path.dirname(os.path.dirname(_this_dir)))
292
+ if _project_root not in sys.path:
293
+ sys.path.insert(0, _project_root)
294
+
295
+
296
+ # ── Import server ──
297
+
298
+ from extensions.services.evol.server import EvolServer
299
+
300
+
301
+ def _fmt_elapsed(t0: float) -> str:
302
+ d = time.monotonic() - t0
303
+ if d < 1:
304
+ return f"{d * 1000:.0f}ms"
305
+ if d < 10:
306
+ return f"{d:.1f}s"
307
+ return f"{d:.0f}s"
308
+
309
+
310
+ def _bind_port(preferred: int, host: str, max_attempts: int = 3) -> int | None:
311
+ """Try to bind to preferred port, then port+1, port+2, ..."""
312
+ if not preferred:
313
+ with socket.socket(socket.AF_INET, socket.SOCK_STREAM) as s:
314
+ s.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
315
+ s.bind((host, 0))
316
+ return s.getsockname()[1]
317
+
318
+ for attempt in range(max_attempts):
319
+ port = preferred + attempt
320
+ try:
321
+ with socket.socket(socket.AF_INET, socket.SOCK_STREAM) as s:
322
+ s.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
323
+ s.bind((host, port))
324
+ if attempt > 0:
325
+ print(f"Bound to port {port} (preferred {preferred} was occupied)")
326
+ return port
327
+ except OSError as e:
328
+ if attempt < max_attempts - 1:
329
+ print(f"Port {port} occupied, trying next port...")
330
+ continue
331
+ else:
332
+ print(f"ERROR: Failed to bind port after {max_attempts} attempts ({preferred}-{preferred + max_attempts - 1})")
333
+ return None
334
+
335
+ return None
336
+
337
+
338
+ def _auto_build_frontend():
339
+ """Auto-build Vue frontend on every startup."""
340
+ _this_dir = os.path.dirname(os.path.abspath(__file__))
341
+ web_dir = os.path.join(_this_dir, "web")
342
+
343
+ if not os.path.exists(web_dir):
344
+ return
345
+
346
+ package_json = os.path.join(web_dir, "package.json")
347
+ if not os.path.exists(package_json):
348
+ return
349
+
350
+ print("[evol] Building frontend...")
351
+ import subprocess
352
+ import platform
353
+
354
+ npm_cmd = "npm.cmd" if platform.system() == "Windows" else "npm"
355
+
356
+ try:
357
+ node_modules = os.path.join(web_dir, "node_modules")
358
+ if not os.path.exists(node_modules):
359
+ print("[evol] Installing dependencies...")
360
+ subprocess.run([npm_cmd, "install"], cwd=web_dir, check=True, capture_output=True, shell=True)
361
+
362
+ subprocess.run([npm_cmd, "run", "build"], cwd=web_dir, check=True, capture_output=True, shell=True)
363
+ print("[evol] Frontend build complete")
364
+ except subprocess.CalledProcessError as e:
365
+ print(f"[evol] WARNING: Frontend build failed: {e}")
366
+ except FileNotFoundError:
367
+ print("[evol] WARNING: npm not found, skipping frontend build")
368
+
369
+ def _run_local_mode(kernel_port: int):
370
+ # Initialize log file paths
371
+ global _log_dir, _log_latest_path, _crash_log_path
372
+ module_data = os.environ.get("KITE_MODULE_DATA")
373
+ if module_data:
374
+ _log_dir = os.path.join(module_data, "log")
375
+ os.makedirs(_log_dir, exist_ok=True)
376
+ suffix = os.environ.get("KITE_INSTANCE_SUFFIX", "")
377
+
378
+ _log_latest_path = os.path.join(_log_dir, f"latest{suffix}.log")
379
+ try:
380
+ with open(_log_latest_path, "w", encoding="utf-8") as f:
381
+ pass
382
+ except Exception:
383
+ _log_latest_path = None
384
+
385
+ _crash_log_path = os.path.join(_log_dir, f"crashes{suffix}.jsonl")
386
+ try:
387
+ with open(_crash_log_path, "w", encoding="utf-8") as f:
388
+ pass
389
+ except Exception:
390
+ _crash_log_path = None
391
+
392
+ _resolve_daily_log_path()
393
+
394
+ _setup_exception_hooks()
395
+
396
+ _t0 = time.monotonic()
397
+
398
+ print(f"Kernel port: {kernel_port} ({_fmt_elapsed(_t0)})")
399
+
400
+ host = _module_config["advertise_ip"]
401
+ port = _bind_port(_module_config["preferred_port"], host)
402
+
403
+ if port is None:
404
+ print("ERROR: Cannot bind to any port, exiting")
405
+ sys.exit(1)
406
+
407
+ # Read token from stdin before creating server
408
+ line = sys.stdin.readline().strip()
409
+ if not line:
410
+ print(f"\033[31m[{MODULE_NAME}] ERROR: stdin closed\033[0m")
411
+ sys.exit(1)
412
+
413
+ try:
414
+ msg = json.loads(line)
415
+ except json.JSONDecodeError as e:
416
+ print(f"\033[31m[{MODULE_NAME}] ERROR: Invalid JSON from stdin: {e}\033[0m")
417
+ sys.exit(1)
418
+
419
+ if "error" in msg:
420
+ print(f"\033[31m[{MODULE_NAME}] 启动失败: {msg.get('message', 'unknown')}\033[0m")
421
+ sys.exit(1)
422
+
423
+ token = msg.get("token", "")
424
+ if not token:
425
+ print(f"\033[31m[{MODULE_NAME}] ERROR: No token in stdin message\033[0m")
426
+ sys.exit(1)
427
+
428
+ # Create server with token
429
+ server = EvolServer(
430
+ module_name=MODULE_NAME,
431
+ token=token,
432
+ kernel_port=kernel_port,
433
+ host=host,
434
+ port=port,
435
+ boot_t0=_t0,
436
+ max_connections=_module_config.get("max_connections", 1),
437
+ )
438
+
439
+ display_host = "localhost" if host == "0.0.0.0" else host
440
+ url = f"http://{display_host}:{port}"
441
+ print(f"Starting on {host}:{port} ({_fmt_elapsed(_t0)})")
442
+ print(f"\033[32m✓ Evol UI ready: {url}\033[0m")
443
+
444
+ # 前端编译在后台线程执行,不阻塞 module.ready
445
+ def _build_frontend_bg():
446
+ try:
447
+ _auto_build_frontend()
448
+ except Exception as e:
449
+ print(f"[evol] WARNING: Frontend build error: {e}")
450
+ _build_thread = threading.Thread(target=_build_frontend_bg, daemon=True, name="frontend-build")
451
+ _build_thread.start()
452
+
453
+ try:
454
+ config = uvicorn.Config(server.app, host=host, port=port, log_level="warning")
455
+ uvi_server = uvicorn.Server(config)
456
+ server._uvicorn_server = uvi_server
457
+ uvi_server.run()
458
+ except Exception as e:
459
+ print(f"ERROR: {e}")
460
+ traceback.print_exc()
461
+ sys.exit(1)
462
+
463
+ # Check if server requested exit with non-zero code
464
+ if server._exit_code != 0:
465
+ sys.exit(server._exit_code)
466
+
467
+
468
+ def _run_remote_mode(gateway_url: str):
469
+ global _log_dir, _log_latest_path, _crash_log_path
470
+ home = os.environ.get("HOME") or os.environ.get("USERPROFILE") or os.path.expanduser("~")
471
+ data_dir = os.path.join(home, ".kite", "remote", MODULE_NAME)
472
+ os.makedirs(data_dir, exist_ok=True)
473
+ _log_dir = os.path.join(data_dir, "log")
474
+ os.makedirs(_log_dir, exist_ok=True)
475
+ _log_latest_path = os.path.join(_log_dir, "latest.log")
476
+ try:
477
+ with open(_log_latest_path, "w", encoding="utf-8") as f: pass
478
+ except Exception: _log_latest_path = None
479
+ _crash_log_path = os.path.join(_log_dir, "crashes.jsonl")
480
+ try:
481
+ with open(_crash_log_path, "w", encoding="utf-8") as f: pass
482
+ except Exception: _crash_log_path = None
483
+ _resolve_daily_log_path()
484
+ _setup_exception_hooks()
485
+
486
+ _t0 = time.monotonic()
487
+ kite_token = _get_kite_token(MODULE_NAME, gateway_url)
488
+ print(f"[{MODULE_NAME}] Remote mode: gateway={gateway_url}")
489
+
490
+ host = _module_config["advertise_ip"]
491
+ port = _bind_port(_module_config["preferred_port"], host)
492
+ if port is None:
493
+ print(f"[{MODULE_NAME}] ERROR: Cannot bind to any port")
494
+ sys.exit(1)
495
+
496
+ server = EvolServer(
497
+ module_name=MODULE_NAME,
498
+ token="",
499
+ kernel_port=0,
500
+ host=host,
501
+ port=port,
502
+ boot_t0=_t0,
503
+ max_connections=_module_config.get("max_connections", 1),
504
+ gateway_url=gateway_url,
505
+ kite_token=kite_token,
506
+ )
507
+
508
+ try:
509
+ config = uvicorn.Config(server.app, host=host, port=port, log_level="warning")
510
+ uvi_server = uvicorn.Server(config)
511
+ server._uvicorn_server = uvi_server
512
+ uvi_server.run()
513
+ except Exception as e:
514
+ _write_crash(type(e), e, e.__traceback__, severity="critical", handled=True)
515
+ sys.exit(1)
516
+ if hasattr(server, '_auth_failed') and server._auth_failed:
517
+ _clear_token_cache(MODULE_NAME, gateway_url)
518
+ if server._exit_code != 0:
519
+ sys.exit(server._exit_code)
520
+
521
+
522
+ # ── Token management ──
523
+
524
+ def _gateway_to_filename(gateway_url: str) -> str:
525
+ try:
526
+ from urllib.parse import urlparse
527
+ parsed = urlparse(gateway_url)
528
+ host = parsed.hostname or "unknown"
529
+ port = parsed.port or (443 if parsed.scheme == "wss" else 80)
530
+ return f"{host}-{port}.json".replace(":", "-").replace("/", "-")
531
+ except Exception: return "default.json"
532
+
533
+ def _load_token_cache(module_name: str, gateway_url: str) -> dict | None:
534
+ home = os.environ.get("HOME") or os.environ.get("USERPROFILE") or os.path.expanduser("~")
535
+ token_file = os.path.join(home, ".kite", "remote", module_name, "tokens", _gateway_to_filename(gateway_url))
536
+ if not os.path.exists(token_file): return None
537
+ try:
538
+ with open(token_file, "r") as f: return json.load(f)
539
+ except Exception: return None
540
+
541
+ def _clear_token_cache(module_name: str, gateway_url: str):
542
+ home = os.environ.get("HOME") or os.environ.get("USERPROFILE") or os.path.expanduser("~")
543
+ token_file = os.path.join(home, ".kite", "remote", module_name, "tokens", _gateway_to_filename(gateway_url))
544
+ try: os.remove(token_file)
545
+ except Exception: pass
546
+
547
+ def _get_kite_token(module_name: str, gateway_url: str) -> str:
548
+ token = os.environ.get("KITE_TOKEN")
549
+ if token: return token
550
+ cache = _load_token_cache(module_name, gateway_url)
551
+ if cache: return cache["token"]
552
+ print(f"[{module_name}] No token for {gateway_url}")
553
+ print(f" export KITE_TOKEN=<token>")
554
+ sys.exit(1)
555
+
556
+
557
+ if __name__ == "__main__":
558
+ kernel_port = os.environ.get("KITE_KERNEL_PORT")
559
+ if kernel_port:
560
+ _run_local_mode(int(kernel_port))
561
+ else:
562
+ config = _load_module_config()
563
+ gateway_url = config.get("gateway_url") or os.environ.get("KITE_GATEWAY_URL")
564
+ if gateway_url:
565
+ _run_remote_mode(gateway_url)
566
+ else:
567
+ print(f"[{MODULE_NAME}] ERROR: No KITE_KERNEL_PORT and no gateway_url")
568
+ sys.exit(1)