@agentunion/kite 1.5.0 → 1.6.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 (573) 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/builders/__init__.py +4 -0
  492. package/kite_cli/builders/base.py +67 -0
  493. package/kite_cli/builders/custom.py +31 -0
  494. package/kite_cli/builders/detector.py +56 -0
  495. package/kite_cli/builders/go.py +34 -0
  496. package/kite_cli/builders/gradle.py +41 -0
  497. package/kite_cli/builders/maven.py +36 -0
  498. package/kite_cli/builders/npm.py +44 -0
  499. package/kite_cli/builders/python.py +37 -0
  500. package/kite_cli/commands/BUILD_GUIDE.md +109 -0
  501. package/kite_cli/commands/build.py +142 -0
  502. package/kite_cli/commands/check.py +60 -0
  503. package/kite_cli/commands/config.py +156 -0
  504. package/kite_cli/commands/deps.py +58 -0
  505. package/kite_cli/commands/deps_install.py +7 -7
  506. package/kite_cli/commands/disable.py +162 -0
  507. package/kite_cli/commands/enable.py +162 -0
  508. package/kite_cli/commands/export.py +96 -0
  509. package/kite_cli/commands/import_cmd.py +110 -0
  510. package/kite_cli/commands/install.py +50 -23
  511. package/kite_cli/commands/install_skill.py +107 -0
  512. package/kite_cli/commands/list.py +128 -31
  513. package/kite_cli/commands/outdated.py +202 -0
  514. package/kite_cli/commands/search.py +33 -17
  515. package/kite_cli/commands/update.py +115 -2
  516. package/kite_cli/commands/venv_setup.py +6 -6
  517. package/kite_cli/commands/why.py +48 -0
  518. package/kite_cli/core/config_manager.py +145 -0
  519. package/kite_cli/core/downloader.py +32 -2
  520. package/kite_cli/main.py +151 -5
  521. package/kite_cli/utils/colors.py +153 -0
  522. package/kite_cli/utils/dependency_graph.py +209 -0
  523. package/kite_cli/utils/process.py +55 -0
  524. package/kite_cli/utils/progress.py +207 -0
  525. package/kite_cli/utils/table.py +101 -0
  526. package/launcher/count_lines.py +192 -43
  527. package/launcher/entry.py +4543 -2802
  528. package/launcher/logging_setup.py +54 -1
  529. package/launcher/module.md +32 -6
  530. package/launcher/module_scanner.py +93 -20
  531. package/launcher/process_manager.py +355 -76
  532. package/main.py +6 -0
  533. package/package.json +4 -1
  534. package/requirements.txt +41 -38
  535. package/scripts/auto-fix-deps.py +128 -0
  536. package/scripts/env-manager.js +25 -2
  537. package/scripts/final-test.js +78 -0
  538. package/scripts/setup-python-env.js +700 -191
  539. package/scripts/test-alluser.js +48 -0
  540. package/scripts/test-different-version.js +86 -0
  541. package/scripts/test-direct.js +63 -0
  542. package/scripts/test-extract-installer.js +28 -0
  543. package/scripts/test-install-log.js +54 -0
  544. package/scripts/test-installer.js +39 -0
  545. package/scripts/test-integration.js +250 -0
  546. package/scripts/test-real-install.js +210 -0
  547. package/scripts/test-targetdir.js +49 -0
  548. package/scripts/test-venv-real.js +47 -0
  549. package/scripts/test-venv-simple.js +57 -0
  550. package/scripts/test-wait.js +49 -0
  551. package/scripts/test-with-log.js +63 -0
  552. package/extensions/services/evol/config.yaml +0 -149
  553. package/extensions/services/evol/routes/routes_management_ws.py +0 -127
  554. package/extensions/services/evol/static/index_evol.html +0 -14
  555. package/extensions/services/evol/static/js/app.js +0 -6304
  556. package/extensions/services/evol/static/js/auth.js +0 -326
  557. package/extensions/services/evol/static/js/evol-app-fixed.js +0 -50
  558. package/extensions/services/evol/static/js/evol-app.js.bak +0 -1800
  559. package/extensions/services/evol/static/js/kernel-client-example.js +0 -228
  560. package/extensions/services/evol/static/js/main.js +0 -141
  561. package/extensions/services/evol/static/js/stats.js +0 -217
  562. package/extensions/services/evol/static/js/token-manager.js +0 -175
  563. package/extensions/services/proxy/CHANGELOG_20260308.md +0 -258
  564. package/extensions/services/proxy/_fix_prints.py +0 -133
  565. package/extensions/services/proxy/_fix_prints2.py +0 -87
  566. package/extensions/services/proxy/console_auth.py +0 -109
  567. package/extensions/services/proxy/logs/websocket.log +0 -260
  568. package/extensions/services/proxy/main.py +0 -240
  569. package/extensions/services/proxy/requirements.txt +0 -13
  570. package/extensions/services/web/config.yaml +0 -149
  571. /package/extensions/services/{evol → kite_console}/static/pairing.html +0 -0
  572. /package/extensions/services/{evol → kite_console}/static/test_registry.html +0 -0
  573. /package/extensions/services/{evol → kite_console}/static/test_relay.html +0 -0
@@ -0,0 +1,505 @@
1
+ """
2
+ 身份认证 API
3
+ """
4
+ from datetime import datetime, timedelta, timezone
5
+ from typing import Optional
6
+
7
+ from fastapi import APIRouter, Depends, HTTPException
8
+ from pydantic import BaseModel
9
+ from sqlalchemy import text
10
+ from sqlalchemy.ext.asyncio import AsyncSession
11
+
12
+ from core.auth import create_access_token, get_current_agent, require_auth
13
+ from core.database import get_session
14
+ from utils.id_generator import gen_agent_id, gen_api_key, gen_key_id, gen_verify_code, gen_uuid
15
+ from utils.crypto import hash_api_key
16
+
17
+ router = APIRouter()
18
+ DISC = {"entry": "/.well-known/agent-data-exchange.json", "api_spec": "/v1/openapi.json"}
19
+
20
+
21
+ class RegisterRequest(BaseModel):
22
+ name: str
23
+ type: str = "custom"
24
+ description: str = ""
25
+ callback_url: str = ""
26
+ contacts: dict = {}
27
+ request_verification: bool = False
28
+
29
+
30
+ class VerifyRequest(BaseModel):
31
+ agent_id: str
32
+ code: str
33
+
34
+
35
+ class RecoverRequest(BaseModel):
36
+ contact_type: str
37
+ contact_value: str
38
+
39
+
40
+ class RecoverVerifyRequest(BaseModel):
41
+ agent_id: str
42
+ code: str
43
+
44
+
45
+ class UpdateProfileRequest(BaseModel):
46
+ name: Optional[str] = None
47
+ description: Optional[str] = None
48
+ callback_url: Optional[str] = None
49
+
50
+
51
+ class UpdateContactsRequest(BaseModel):
52
+ phone: Optional[str] = None
53
+ email: Optional[str] = None
54
+
55
+
56
+ class CreateKeyRequest(BaseModel):
57
+ name: str = ""
58
+ permissions: list = ["read", "write"]
59
+ expires_days: Optional[int] = None
60
+
61
+
62
+ @router.post("/register")
63
+ async def register(body: RegisterRequest, session: AsyncSession = Depends(get_session)):
64
+ row = await session.execute(
65
+ text("SELECT id FROM agents WHERE name=:n AND status!='deleted' LIMIT 1"), {"n": body.name}
66
+ )
67
+ if row.fetchone():
68
+ raise HTTPException(400, detail={"ok": False, "error": "NAME_TAKEN",
69
+ "message": f"名称 '{body.name}' 已被占用", "_discovery": DISC})
70
+ agent_id = gen_agent_id()
71
+ if body.request_verification and body.contacts.get("phone"):
72
+ await session.execute(
73
+ text("INSERT INTO agents (id, name, type, description, callback_url, verified, credits, status) "
74
+ "VALUES (:id, :name, :type, :desc, :cb, 0, 0.00, 'active')"),
75
+ {"id": agent_id, "name": body.name, "type": body.type, "desc": body.description, "cb": body.callback_url}
76
+ )
77
+ phone = body.contacts["phone"]
78
+ code = gen_verify_code()
79
+ expires = datetime.now(timezone.utc) + timedelta(minutes=10)
80
+ await session.execute(
81
+ text("INSERT INTO verification_codes (contact_type, contact_value, code, purpose, agent_id, expires_at) "
82
+ "VALUES ('phone', :phone, :code, 'register', :aid, :exp)"),
83
+ {"phone": phone, "code": code, "aid": agent_id, "exp": expires}
84
+ )
85
+ await session.commit()
86
+ return {"ok": True, "data": {"agent_id": agent_id, "status": "pending_verification",
87
+ "verification_sent_to": phone,
88
+ "message": "验证码已发送,请在 10 分钟内完成验证"}, "_discovery": DISC}
89
+ else:
90
+ api_key = gen_api_key()
91
+ key_id = gen_key_id()
92
+ initial_credits = 100.0
93
+ await session.execute(
94
+ text("INSERT INTO agents (id, name, type, description, callback_url, verified, credits, status) "
95
+ "VALUES (:id, :name, :type, :desc, :cb, 0, :credits, 'active')"),
96
+ {"id": agent_id, "name": body.name, "type": body.type, "desc": body.description,
97
+ "cb": body.callback_url, "credits": initial_credits}
98
+ )
99
+ await session.execute(
100
+ text("INSERT INTO api_keys (id, agent_id, key_hash, key_prefix, name, permissions, status) "
101
+ "VALUES (:kid, :aid, :hash, :prefix, 'default', '[\"read\",\"write\"]', 'active')"),
102
+ {"kid": key_id, "aid": agent_id, "hash": hash_api_key(api_key), "prefix": api_key[:10]}
103
+ )
104
+ tid = gen_uuid()[:32]
105
+ await session.execute(
106
+ text("INSERT INTO credit_transactions (id, agent_id, type, amount, balance_after, description) "
107
+ "VALUES (:tid, :aid, 'bonus', :amt, :bal, '注册奖励')"),
108
+ {"tid": tid, "aid": agent_id, "amt": initial_credits, "bal": initial_credits}
109
+ )
110
+ await session.commit()
111
+ return {"ok": True, "data": {"agent_id": agent_id, "api_key": api_key,
112
+ "initial_credits": initial_credits, "verified": False,
113
+ "message": f"注册成功,已赠送 {initial_credits} 积分"}, "_discovery": DISC}
114
+
115
+
116
+ @router.post("/verify")
117
+ async def verify(body: VerifyRequest, session: AsyncSession = Depends(get_session)):
118
+ row = (await session.execute(
119
+ text("SELECT id, expires_at, used FROM verification_codes "
120
+ "WHERE agent_id=:aid AND code=:code AND purpose='register' ORDER BY created_at DESC LIMIT 1"),
121
+ {"aid": body.agent_id, "code": body.code}
122
+ )).fetchone()
123
+ if not row:
124
+ raise HTTPException(400, detail={"ok": False, "error": "INVALID_CODE",
125
+ "message": "验证码无效", "_discovery": DISC})
126
+ if row.used:
127
+ raise HTTPException(400, detail={"ok": False, "error": "CODE_USED",
128
+ "message": "验证码已使用", "_discovery": DISC})
129
+ if datetime.now(timezone.utc) > row.expires_at.replace(tzinfo=timezone.utc):
130
+ raise HTTPException(400, detail={"ok": False, "error": "CODE_EXPIRED",
131
+ "message": "验证码已过期", "_discovery": DISC})
132
+ await session.execute(
133
+ text("UPDATE agents SET verified=1 WHERE id=:aid"), {"aid": body.agent_id}
134
+ )
135
+ await session.execute(
136
+ text("UPDATE verification_codes SET used=1 WHERE id=:id"), {"id": row.id}
137
+ )
138
+ api_key = gen_api_key()
139
+ key_id = gen_key_id()
140
+ await session.execute(
141
+ text("INSERT INTO api_keys (id, agent_id, key_hash, key_prefix, name, permissions, status) "
142
+ "VALUES (:kid, :aid, :hash, :prefix, 'default', '[\"read\",\"write\"]', 'active')"),
143
+ {"kid": key_id, "aid": body.agent_id, "hash": hash_api_key(api_key), "prefix": api_key[:10]}
144
+ )
145
+ await session.commit()
146
+ return {"ok": True, "data": {"agent_id": body.agent_id, "api_key": api_key,
147
+ "verified": True, "message": "验证成功"}, "_discovery": DISC}
148
+
149
+
150
+ @router.post("/recover")
151
+ async def recover(body: RecoverRequest, session: AsyncSession = Depends(get_session)):
152
+ row = (await session.execute(
153
+ text("SELECT a.id FROM agents a "
154
+ "JOIN agent_contacts c ON c.agent_id=a.id "
155
+ "WHERE c.contact_type=:ct AND c.contact_value=:cv AND a.status='active' LIMIT 1"),
156
+ {"ct": body.contact_type, "cv": body.contact_value}
157
+ )).fetchone()
158
+ if not row:
159
+ raise HTTPException(404, detail={"ok": False, "error": "NOT_FOUND",
160
+ "message": "未找到匹配的账户", "_discovery": DISC})
161
+ agent_id = row.id
162
+ code = gen_verify_code()
163
+ expires = datetime.now(timezone.utc) + timedelta(minutes=10)
164
+ await session.execute(
165
+ text("INSERT INTO verification_codes (contact_type, contact_value, code, purpose, agent_id, expires_at) "
166
+ "VALUES (:ct, :cv, :code, 'recover', :aid, :exp)"),
167
+ {"ct": body.contact_type, "cv": body.contact_value, "code": code, "aid": agent_id, "exp": expires}
168
+ )
169
+ await session.commit()
170
+ return {"ok": True, "data": {"agent_id": agent_id,
171
+ "message": "验证码已发送,请在 10 分钟内完成验证"}, "_discovery": DISC}
172
+
173
+
174
+ @router.post("/recover/verify")
175
+ async def recover_verify(body: RecoverVerifyRequest, session: AsyncSession = Depends(get_session)):
176
+ row = (await session.execute(
177
+ text("SELECT id, expires_at, used FROM verification_codes "
178
+ "WHERE agent_id=:aid AND code=:code AND purpose='recover' ORDER BY created_at DESC LIMIT 1"),
179
+ {"aid": body.agent_id, "code": body.code}
180
+ )).fetchone()
181
+ if not row:
182
+ raise HTTPException(400, detail={"ok": False, "error": "INVALID_CODE",
183
+ "message": "验证码无效", "_discovery": DISC})
184
+ if row.used:
185
+ raise HTTPException(400, detail={"ok": False, "error": "CODE_USED",
186
+ "message": "验证码已使用", "_discovery": DISC})
187
+ if datetime.now(timezone.utc) > row.expires_at.replace(tzinfo=timezone.utc):
188
+ raise HTTPException(400, detail={"ok": False, "error": "CODE_EXPIRED",
189
+ "message": "验证码已过期", "_discovery": DISC})
190
+ await session.execute(
191
+ text("UPDATE verification_codes SET used=1 WHERE id=:id"), {"id": row.id}
192
+ )
193
+ api_key = gen_api_key()
194
+ key_id = gen_key_id()
195
+ await session.execute(
196
+ text("INSERT INTO api_keys (id, agent_id, key_hash, key_prefix, name, permissions, status) "
197
+ "VALUES (:kid, :aid, :hash, :prefix, 'recovered', '[\"read\",\"write\"]', 'active')"),
198
+ {"kid": key_id, "aid": body.agent_id, "hash": hash_api_key(api_key), "prefix": api_key[:10]}
199
+ )
200
+ await session.commit()
201
+ return {"ok": True, "data": {"agent_id": body.agent_id, "api_key": api_key,
202
+ "message": "恢复成功,新 API Key 已生成"}, "_discovery": DISC}
203
+
204
+
205
+ @router.get("/profile")
206
+ async def get_profile(agent_id: str = Depends(require_auth), session: AsyncSession = Depends(get_session)):
207
+ row = (await session.execute(
208
+ text("SELECT id, name, type, description, callback_url, verified, credits, status, created_at "
209
+ "FROM agents WHERE id=:id"),
210
+ {"id": agent_id}
211
+ )).fetchone()
212
+ if not row:
213
+ raise HTTPException(404, detail={"ok": False, "error": "NOT_FOUND", "_discovery": DISC})
214
+ return {"ok": True, "data": {
215
+ "agent_id": row.id, "name": row.name, "type": row.type,
216
+ "description": row.description, "callback_url": row.callback_url,
217
+ "verified": bool(row.verified), "credits": float(row.credits),
218
+ "status": row.status, "created_at": row.created_at.isoformat()
219
+ }, "_discovery": DISC}
220
+
221
+
222
+ @router.patch("/profile")
223
+ async def update_profile(body: UpdateProfileRequest, agent_id: str = Depends(require_auth),
224
+ session: AsyncSession = Depends(get_session)):
225
+ fields, params = [], {"id": agent_id}
226
+ if body.name is not None:
227
+ fields.append("name=:name"); params["name"] = body.name
228
+ if body.description is not None:
229
+ fields.append("description=:description"); params["description"] = body.description
230
+ if body.callback_url is not None:
231
+ fields.append("callback_url=:callback_url"); params["callback_url"] = body.callback_url
232
+ if not fields:
233
+ raise HTTPException(400, detail={"ok": False, "error": "NO_FIELDS",
234
+ "message": "没有提供任何更新字段", "_discovery": DISC})
235
+ await session.execute(text(f"UPDATE agents SET {', '.join(fields)} WHERE id=:id"), params)
236
+ await session.commit()
237
+ return {"ok": True, "data": {"message": "资料已更新"}, "_discovery": DISC}
238
+
239
+
240
+ @router.get("/keys")
241
+ async def list_keys(agent_id: str = Depends(require_auth), session: AsyncSession = Depends(get_session)):
242
+ rows = (await session.execute(
243
+ text("SELECT id, name, key_prefix, permissions, status, created_at, expires_at, last_used_at "
244
+ "FROM api_keys WHERE agent_id=:aid AND status!='deleted' ORDER BY created_at DESC"),
245
+ {"aid": agent_id}
246
+ )).fetchall()
247
+ keys = []
248
+ for r in rows:
249
+ keys.append({
250
+ "key_id": r.id, "name": r.name, "key_prefix": r.key_prefix,
251
+ "permissions": r.permissions, "status": r.status,
252
+ "created_at": r.created_at.isoformat(),
253
+ "expires_at": r.expires_at.isoformat() if r.expires_at else None,
254
+ "last_used_at": r.last_used_at.isoformat() if r.last_used_at else None
255
+ })
256
+ return {"ok": True, "data": {"keys": keys, "total": len(keys)}, "_discovery": DISC}
257
+
258
+
259
+ @router.post("/keys")
260
+ async def create_key(body: CreateKeyRequest, agent_id: str = Depends(require_auth),
261
+ session: AsyncSession = Depends(get_session)):
262
+ api_key = gen_api_key()
263
+ key_id = gen_key_id()
264
+ expires_at = None
265
+ if body.expires_days:
266
+ expires_at = datetime.now(timezone.utc) + timedelta(days=body.expires_days)
267
+ import json
268
+ await session.execute(
269
+ text("INSERT INTO api_keys (id, agent_id, key_hash, key_prefix, name, permissions, status, expires_at) "
270
+ "VALUES (:kid, :aid, :hash, :prefix, :name, :perms, 'active', :exp)"),
271
+ {"kid": key_id, "aid": agent_id, "hash": hash_api_key(api_key),
272
+ "prefix": api_key[:10], "name": body.name,
273
+ "perms": json.dumps(body.permissions), "exp": expires_at}
274
+ )
275
+ await session.commit()
276
+ return {"ok": True, "data": {"key_id": key_id, "api_key": api_key,
277
+ "name": body.name, "permissions": body.permissions,
278
+ "expires_at": expires_at.isoformat() if expires_at else None,
279
+ "message": "API Key 已创建,请妥善保存,此后不再显示完整密钥"}, "_discovery": DISC}
280
+
281
+
282
+ class SendCodeRequest(BaseModel):
283
+ phone: Optional[str] = None
284
+ email: Optional[str] = None
285
+ purpose: str = "login"
286
+
287
+
288
+ class LoginRequest(BaseModel):
289
+ phone: Optional[str] = None
290
+ email: Optional[str] = None
291
+ code: str = ""
292
+ api_key: Optional[str] = None
293
+
294
+
295
+ @router.post("/send-code")
296
+ async def send_code(body: SendCodeRequest, session: AsyncSession = Depends(get_session)):
297
+ """发送验证码(手机或邮箱)"""
298
+ code = gen_verify_code()
299
+ expires = datetime.now(timezone.utc) + timedelta(minutes=10)
300
+
301
+ if body.phone:
302
+ contact_type, contact_value = "phone", body.phone
303
+ elif body.email:
304
+ contact_type, contact_value = "email", body.email
305
+ else:
306
+ raise HTTPException(400, detail={"ok": False, "error": "MISSING_CONTACT",
307
+ "message": "请提供手机号或邮箱", "_discovery": DISC})
308
+
309
+ # 查找关联的 agent_id(如果存在)
310
+ agent_id = None
311
+ row = (await session.execute(
312
+ text("SELECT agent_id FROM agent_contacts WHERE contact_type=:ct AND contact_value=:cv LIMIT 1"),
313
+ {"ct": contact_type, "cv": contact_value}
314
+ )).fetchone()
315
+ if row:
316
+ agent_id = row.agent_id
317
+
318
+ await session.execute(
319
+ text("INSERT INTO verification_codes (contact_type, contact_value, code, purpose, agent_id, expires_at) "
320
+ "VALUES (:ct, :cv, :code, :purpose, :aid, :exp)"),
321
+ {"ct": contact_type, "cv": contact_value, "code": code,
322
+ "purpose": body.purpose, "aid": agent_id, "exp": expires}
323
+ )
324
+ await session.commit()
325
+
326
+ # 尝试实际发送(降级模式下只打印日志)
327
+ if contact_type == "phone":
328
+ try:
329
+ from services.sms_service import send_verification_code
330
+ await send_verification_code(body.phone, code)
331
+ except Exception as e:
332
+ print(f"[dataclaw] send sms failed: {e}")
333
+ elif contact_type == "email":
334
+ try:
335
+ from services.email_service import send_verification_email
336
+ await send_verification_email(body.email, code)
337
+ except Exception as e:
338
+ print(f"[dataclaw] send email failed: {e}")
339
+
340
+ return {"ok": True, "data": {"message": "验证码已发送", "expires_in": 600,
341
+ "contact": contact_value}, "_discovery": DISC}
342
+
343
+
344
+ @router.post("/login")
345
+ async def login(body: LoginRequest, session: AsyncSession = Depends(get_session)):
346
+ """通过手机号/邮箱 + 验证码登录"""
347
+ if body.api_key:
348
+ # API Key 直接登录
349
+ from utils.crypto import hash_api_key
350
+ key_hash = hash_api_key(body.api_key)
351
+ row = (await session.execute(
352
+ text("SELECT agent_id FROM api_keys WHERE key_hash=:h AND status='active' LIMIT 1"),
353
+ {"h": key_hash}
354
+ )).fetchone()
355
+ if not row:
356
+ raise HTTPException(401, detail={"ok": False, "error": "INVALID_KEY",
357
+ "message": "API Key 无效", "_discovery": DISC})
358
+ token = create_access_token(row.agent_id)
359
+ return {"ok": True, "data": {"agent_id": row.agent_id, "token": token,
360
+ "api_key": body.api_key}, "_discovery": DISC}
361
+
362
+ # 验证码登录
363
+ if body.phone:
364
+ contact_type, contact_value = "phone", body.phone
365
+ elif body.email:
366
+ contact_type, contact_value = "email", body.email
367
+ else:
368
+ raise HTTPException(400, detail={"ok": False, "error": "MISSING_CONTACT",
369
+ "message": "请提供手机号或邮箱", "_discovery": DISC})
370
+
371
+ if not body.code:
372
+ raise HTTPException(400, detail={"ok": False, "error": "MISSING_CODE",
373
+ "message": "请输入验证码", "_discovery": DISC})
374
+
375
+ # 验证码校验
376
+ code_row = (await session.execute(
377
+ text("SELECT id, agent_id, expires_at, used FROM verification_codes "
378
+ "WHERE contact_type=:ct AND contact_value=:cv AND code=:code AND used=0 "
379
+ "ORDER BY created_at DESC LIMIT 1"),
380
+ {"ct": contact_type, "cv": contact_value, "code": body.code}
381
+ )).fetchone()
382
+ if not code_row:
383
+ raise HTTPException(400, detail={"ok": False, "error": "INVALID_CODE",
384
+ "message": "验证码无效", "_discovery": DISC})
385
+ if datetime.now(timezone.utc) > code_row.expires_at.replace(tzinfo=timezone.utc):
386
+ raise HTTPException(400, detail={"ok": False, "error": "CODE_EXPIRED",
387
+ "message": "验证码已过期", "_discovery": DISC})
388
+
389
+ # 标记验证码已使用
390
+ await session.execute(
391
+ text("UPDATE verification_codes SET used=1 WHERE id=:id"), {"id": code_row.id}
392
+ )
393
+
394
+ agent_id = code_row.agent_id
395
+ if not agent_id:
396
+ # 查找联系方式关联的 agent
397
+ contact = (await session.execute(
398
+ text("SELECT agent_id FROM agent_contacts WHERE contact_type=:ct AND contact_value=:cv LIMIT 1"),
399
+ {"ct": contact_type, "cv": contact_value}
400
+ )).fetchone()
401
+ if contact:
402
+ agent_id = contact.agent_id
403
+
404
+ if not agent_id:
405
+ # 自动注册新 agent
406
+ agent_id = gen_agent_id()
407
+ initial_credits = 100.0
408
+ name = f"{contact_type}_{contact_value[-4:]}"
409
+ await session.execute(
410
+ text("INSERT INTO agents (id, name, type, description, verified, credits, status) "
411
+ "VALUES (:id, :name, 'custom', :desc, 1, :credits, 'active')"),
412
+ {"id": agent_id, "name": name, "desc": f"通过{contact_type}注册", "credits": initial_credits}
413
+ )
414
+ await session.execute(
415
+ text("INSERT INTO agent_contacts (agent_id, contact_type, contact_value, verified) "
416
+ "VALUES (:aid, :ct, :cv, 1)"),
417
+ {"aid": agent_id, "ct": contact_type, "cv": contact_value}
418
+ )
419
+ # 创建 API Key
420
+ api_key = gen_api_key()
421
+ key_id = gen_key_id()
422
+ await session.execute(
423
+ text("INSERT INTO api_keys (id, agent_id, key_hash, key_prefix, name, permissions, status) "
424
+ "VALUES (:kid, :aid, :hash, :prefix, 'login_default', '[\"read\",\"write\"]', 'active')"),
425
+ {"kid": key_id, "aid": agent_id, "hash": hash_api_key(api_key), "prefix": api_key[:10]}
426
+ )
427
+ tid = gen_uuid()[:32]
428
+ await session.execute(
429
+ text("INSERT INTO credit_transactions (id, agent_id, type, amount, balance_after, description) "
430
+ "VALUES (:tid, :aid, 'bonus', :amt, :bal, '注册奖励')"),
431
+ {"tid": tid, "aid": agent_id, "amt": initial_credits, "bal": initial_credits}
432
+ )
433
+
434
+ await session.commit()
435
+ token = create_access_token(agent_id)
436
+ return {"ok": True, "data": {"agent_id": agent_id, "token": token,
437
+ "message": "登录成功"}, "_discovery": DISC}
438
+
439
+
440
+ @router.get("/oauth/{provider}/authorize")
441
+ async def oauth_authorize(provider: str):
442
+ """OAuth 授权跳转"""
443
+ from fastapi.responses import RedirectResponse
444
+ if provider == "github":
445
+ from services.oauth_service import get_github_authorize_url
446
+ url = get_github_authorize_url(f"/v1/identity/oauth/github/callback")
447
+ if not url:
448
+ raise HTTPException(503, detail={"ok": False, "error": "OAUTH_NOT_CONFIGURED",
449
+ "message": "GitHub OAuth 未配置", "_discovery": DISC})
450
+ return RedirectResponse(url)
451
+ elif provider == "google":
452
+ from services.oauth_service import get_google_authorize_url
453
+ url = get_google_authorize_url(f"/v1/identity/oauth/google/callback")
454
+ if not url:
455
+ raise HTTPException(503, detail={"ok": False, "error": "OAUTH_NOT_CONFIGURED",
456
+ "message": "Google OAuth 未配置", "_discovery": DISC})
457
+ return RedirectResponse(url)
458
+ else:
459
+ raise HTTPException(400, detail={"ok": False, "error": "UNSUPPORTED_PROVIDER",
460
+ "message": f"不支持的 OAuth 提供商: {provider}", "_discovery": DISC})
461
+
462
+
463
+ @router.get("/oauth/{provider}/callback")
464
+ async def oauth_callback(provider: str, code: str = "", session: AsyncSession = Depends(get_session)):
465
+ """OAuth 回调处理"""
466
+ from fastapi.responses import RedirectResponse
467
+ if not code:
468
+ return RedirectResponse("/login?error=oauth_failed")
469
+
470
+ user_info = None
471
+ if provider == "github":
472
+ from services.oauth_service import exchange_github_code
473
+ user_info = await exchange_github_code(code, f"/v1/identity/oauth/github/callback")
474
+ elif provider == "google":
475
+ from services.oauth_service import exchange_google_code
476
+ user_info = await exchange_google_code(code, f"/v1/identity/oauth/google/callback")
477
+
478
+ if not user_info:
479
+ return RedirectResponse("/login?error=oauth_failed")
480
+
481
+ from services.oauth_service import find_or_create_agent_by_oauth
482
+ agent_id, is_new = await find_or_create_agent_by_oauth(provider, user_info, session)
483
+
484
+ # 生成 JWT token
485
+ token = create_access_token(agent_id)
486
+
487
+ # 重定向到前端并携带 token
488
+ return RedirectResponse(f"/dashboard?oauth_token={token}&provider={provider}&new={'1' if is_new else '0'}")
489
+
490
+
491
+ @router.delete("/keys/{key_id}")
492
+ async def delete_key(key_id: str, agent_id: str = Depends(require_auth),
493
+ session: AsyncSession = Depends(get_session)):
494
+ row = (await session.execute(
495
+ text("SELECT id FROM api_keys WHERE id=:kid AND agent_id=:aid AND status!='deleted'"),
496
+ {"kid": key_id, "aid": agent_id}
497
+ )).fetchone()
498
+ if not row:
499
+ raise HTTPException(404, detail={"ok": False, "error": "NOT_FOUND",
500
+ "message": "Key 不存在", "_discovery": DISC})
501
+ await session.execute(
502
+ text("UPDATE api_keys SET status='deleted' WHERE id=:kid"), {"kid": key_id}
503
+ )
504
+ await session.commit()
505
+ return {"ok": True, "data": {"message": "API Key 已吊销"}, "_discovery": DISC}
@@ -0,0 +1,104 @@
1
+ """
2
+ 通知系统 API
3
+ """
4
+ from typing import Optional
5
+
6
+ from fastapi import APIRouter, Depends, Query
7
+ from pydantic import BaseModel
8
+ from sqlalchemy import text
9
+ from sqlalchemy.ext.asyncio import AsyncSession
10
+
11
+ from core.auth import require_auth
12
+ from core.database import get_session
13
+ from utils.id_generator import gen_notif_id
14
+
15
+ router = APIRouter()
16
+ DISC = {"entry": "/.well-known/agent-data-exchange.json"}
17
+
18
+
19
+ @router.get("")
20
+ async def list_notifications(
21
+ unread_only: bool = Query(False),
22
+ limit: int = Query(50, ge=1, le=200),
23
+ offset: int = Query(0, ge=0),
24
+ agent_id: str = Depends(require_auth),
25
+ session: AsyncSession = Depends(get_session)
26
+ ):
27
+ conditions = ["agent_id=:aid"]
28
+ params = {"aid": agent_id, "lim": limit, "off": offset}
29
+ if unread_only:
30
+ conditions.append("read_at IS NULL")
31
+ where = " AND ".join(conditions)
32
+ rows = (await session.execute(
33
+ text(f"SELECT id, type, title, content, link, read_at, created_at "
34
+ f"FROM notifications WHERE {where} ORDER BY created_at DESC LIMIT :lim OFFSET :off"),
35
+ params
36
+ )).fetchall()
37
+ total = (await session.execute(
38
+ text(f"SELECT COUNT(*) FROM notifications WHERE {where}"),
39
+ {k: v for k, v in params.items() if k not in ('lim', 'off')}
40
+ )).scalar()
41
+ notifs = [
42
+ {
43
+ "id": r.id, "type": r.type, "title": r.title,
44
+ "content": r.content, "link": r.link,
45
+ "read": r.read_at is not None,
46
+ "created_at": r.created_at.isoformat() if r.created_at else None
47
+ }
48
+ for r in rows
49
+ ]
50
+ return {"ok": True, "data": {"notifications": notifs, "total": total,
51
+ "limit": limit, "offset": offset}, "_discovery": DISC}
52
+
53
+
54
+ @router.get("/unread")
55
+ async def get_unread_count(
56
+ agent_id: str = Depends(require_auth),
57
+ session: AsyncSession = Depends(get_session)
58
+ ):
59
+ count = (await session.execute(
60
+ text("SELECT COUNT(*) FROM notifications WHERE agent_id=:aid AND read_at IS NULL"),
61
+ {"aid": agent_id}
62
+ )).scalar()
63
+ return {"ok": True, "data": {"unread_count": count}, "_discovery": DISC}
64
+
65
+
66
+ @router.post("/read-all")
67
+ async def mark_all_read(
68
+ agent_id: str = Depends(require_auth),
69
+ session: AsyncSession = Depends(get_session)
70
+ ):
71
+ result = await session.execute(
72
+ text("UPDATE notifications SET read_at=NOW() WHERE agent_id=:aid AND read_at IS NULL"),
73
+ {"aid": agent_id}
74
+ )
75
+ await session.commit()
76
+ return {"ok": True, "data": {"updated": result.rowcount}, "_discovery": DISC}
77
+
78
+
79
+ @router.post("/{notif_id}/read")
80
+ async def mark_read(
81
+ notif_id: str,
82
+ agent_id: str = Depends(require_auth),
83
+ session: AsyncSession = Depends(get_session)
84
+ ):
85
+ await session.execute(
86
+ text("UPDATE notifications SET read_at=NOW() WHERE id=:nid AND agent_id=:aid"),
87
+ {"nid": notif_id, "aid": agent_id}
88
+ )
89
+ await session.commit()
90
+ return {"ok": True, "data": {"id": notif_id}, "_discovery": DISC}
91
+
92
+
93
+ @router.delete("/{notif_id}")
94
+ async def delete_notification(
95
+ notif_id: str,
96
+ agent_id: str = Depends(require_auth),
97
+ session: AsyncSession = Depends(get_session)
98
+ ):
99
+ await session.execute(
100
+ text("DELETE FROM notifications WHERE id=:nid AND agent_id=:aid"),
101
+ {"nid": notif_id, "aid": agent_id}
102
+ )
103
+ await session.commit()
104
+ return {"ok": True, "data": {"deleted": notif_id}, "_discovery": DISC}