@swarmclawai/swarmclaw 0.7.2 → 0.7.3

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 (197) hide show
  1. package/README.md +81 -22
  2. package/package.json +1 -1
  3. package/src/app/api/agents/[id]/route.ts +26 -0
  4. package/src/app/api/agents/[id]/thread/route.ts +36 -7
  5. package/src/app/api/agents/route.ts +12 -1
  6. package/src/app/api/auth/route.ts +76 -7
  7. package/src/app/api/chatrooms/[id]/chat/route.ts +7 -2
  8. package/src/app/api/chats/[id]/browser/route.ts +5 -1
  9. package/src/app/api/chats/[id]/chat/route.ts +7 -3
  10. package/src/app/api/chats/[id]/main-loop/route.ts +7 -88
  11. package/src/app/api/chats/[id]/messages/route.ts +19 -13
  12. package/src/app/api/chats/[id]/route.ts +18 -0
  13. package/src/app/api/chats/[id]/stop/route.ts +6 -1
  14. package/src/app/api/chats/route.ts +16 -0
  15. package/src/app/api/connectors/[id]/doctor/route.ts +26 -0
  16. package/src/app/api/connectors/doctor/route.ts +13 -0
  17. package/src/app/api/files/open/route.ts +16 -14
  18. package/src/app/api/memory/maintenance/route.ts +11 -1
  19. package/src/app/api/openclaw/agent-files/route.ts +27 -4
  20. package/src/app/api/openclaw/skills/route.ts +11 -3
  21. package/src/app/api/plugins/dependencies/route.ts +24 -0
  22. package/src/app/api/plugins/install/route.ts +15 -92
  23. package/src/app/api/plugins/route.ts +3 -26
  24. package/src/app/api/plugins/settings/route.ts +17 -12
  25. package/src/app/api/plugins/ui/route.ts +1 -0
  26. package/src/app/api/settings/route.ts +49 -7
  27. package/src/app/api/tasks/[id]/route.ts +15 -6
  28. package/src/app/api/tasks/bulk/route.ts +2 -2
  29. package/src/app/api/tasks/route.ts +9 -4
  30. package/src/app/api/webhooks/[id]/route.ts +8 -1
  31. package/src/app/page.tsx +9 -2
  32. package/src/cli/index.js +4 -0
  33. package/src/cli/index.ts +3 -10
  34. package/src/components/agents/agent-card.tsx +15 -12
  35. package/src/components/agents/agent-chat-list.tsx +101 -1
  36. package/src/components/agents/agent-list.tsx +46 -9
  37. package/src/components/agents/agent-sheet.tsx +207 -16
  38. package/src/components/agents/inspector-panel.tsx +108 -48
  39. package/src/components/auth/access-key-gate.tsx +36 -97
  40. package/src/components/chat/chat-area.tsx +29 -13
  41. package/src/components/chat/chat-card.tsx +4 -20
  42. package/src/components/chat/chat-header.tsx +255 -353
  43. package/src/components/chat/chat-list.tsx +7 -9
  44. package/src/components/chat/checkpoint-timeline.tsx +1 -1
  45. package/src/components/chat/message-list.tsx +3 -1
  46. package/src/components/chatrooms/chatroom-view.tsx +347 -205
  47. package/src/components/connectors/connector-list.tsx +265 -127
  48. package/src/components/connectors/connector-sheet.tsx +217 -0
  49. package/src/components/home/home-view.tsx +128 -4
  50. package/src/components/layout/app-layout.tsx +383 -194
  51. package/src/components/layout/mobile-header.tsx +26 -8
  52. package/src/components/plugins/plugin-list.tsx +15 -3
  53. package/src/components/plugins/plugin-sheet.tsx +118 -9
  54. package/src/components/projects/project-detail.tsx +183 -0
  55. package/src/components/shared/agent-picker-list.tsx +2 -2
  56. package/src/components/shared/command-palette.tsx +111 -24
  57. package/src/components/shared/settings/plugin-manager.tsx +20 -4
  58. package/src/components/shared/settings/section-capability-policy.tsx +105 -0
  59. package/src/components/shared/settings/section-heartbeat.tsx +77 -0
  60. package/src/components/shared/settings/section-orchestrator.tsx +3 -3
  61. package/src/components/shared/settings/section-runtime-loop.tsx +5 -5
  62. package/src/components/shared/settings/section-secrets.tsx +6 -6
  63. package/src/components/shared/settings/section-user-preferences.tsx +1 -1
  64. package/src/components/shared/settings/section-voice.tsx +5 -1
  65. package/src/components/shared/settings/section-web-search.tsx +10 -2
  66. package/src/components/shared/settings/settings-page.tsx +245 -46
  67. package/src/components/tasks/approvals-panel.tsx +205 -18
  68. package/src/components/tasks/task-board.tsx +242 -46
  69. package/src/components/usage/metrics-dashboard.tsx +74 -1
  70. package/src/components/wallets/wallet-panel.tsx +17 -5
  71. package/src/components/webhooks/webhook-sheet.tsx +7 -7
  72. package/src/lib/auth.ts +17 -0
  73. package/src/lib/chat-streaming-state.test.ts +108 -0
  74. package/src/lib/chat-streaming-state.ts +108 -0
  75. package/src/lib/openclaw-agent-id.test.ts +14 -0
  76. package/src/lib/openclaw-agent-id.ts +31 -0
  77. package/src/lib/server/agent-assignment.test.ts +112 -0
  78. package/src/lib/server/agent-assignment.ts +169 -0
  79. package/src/lib/server/approval-connector-notify.test.ts +253 -0
  80. package/src/lib/server/approvals-auto-approve.test.ts +205 -0
  81. package/src/lib/server/approvals.ts +483 -75
  82. package/src/lib/server/autonomy-runtime.test.ts +341 -0
  83. package/src/lib/server/browser-state.test.ts +118 -0
  84. package/src/lib/server/browser-state.ts +123 -0
  85. package/src/lib/server/build-llm.test.ts +36 -0
  86. package/src/lib/server/build-llm.ts +11 -4
  87. package/src/lib/server/builtin-plugins.ts +34 -0
  88. package/src/lib/server/chat-execution-heartbeat.test.ts +40 -0
  89. package/src/lib/server/chat-execution-tool-events.test.ts +134 -0
  90. package/src/lib/server/chat-execution.ts +250 -61
  91. package/src/lib/server/chatroom-health.test.ts +26 -0
  92. package/src/lib/server/chatroom-health.ts +2 -3
  93. package/src/lib/server/chatroom-helpers.test.ts +67 -2
  94. package/src/lib/server/chatroom-helpers.ts +45 -5
  95. package/src/lib/server/connectors/discord.ts +175 -11
  96. package/src/lib/server/connectors/doctor.test.ts +80 -0
  97. package/src/lib/server/connectors/doctor.ts +116 -0
  98. package/src/lib/server/connectors/manager.ts +946 -110
  99. package/src/lib/server/connectors/policy.test.ts +222 -0
  100. package/src/lib/server/connectors/policy.ts +452 -0
  101. package/src/lib/server/connectors/slack.ts +188 -9
  102. package/src/lib/server/connectors/telegram.ts +65 -15
  103. package/src/lib/server/connectors/thread-context.test.ts +44 -0
  104. package/src/lib/server/connectors/thread-context.ts +72 -0
  105. package/src/lib/server/connectors/types.ts +41 -11
  106. package/src/lib/server/daemon-state.ts +59 -1
  107. package/src/lib/server/data-dir.ts +13 -0
  108. package/src/lib/server/delegation-jobs.test.ts +140 -0
  109. package/src/lib/server/delegation-jobs.ts +248 -0
  110. package/src/lib/server/document-utils.test.ts +47 -0
  111. package/src/lib/server/document-utils.ts +397 -0
  112. package/src/lib/server/heartbeat-service.ts +13 -39
  113. package/src/lib/server/heartbeat-source.test.ts +22 -0
  114. package/src/lib/server/heartbeat-source.ts +7 -0
  115. package/src/lib/server/identity-continuity.test.ts +77 -0
  116. package/src/lib/server/identity-continuity.ts +127 -0
  117. package/src/lib/server/mailbox-utils.ts +347 -0
  118. package/src/lib/server/main-agent-loop.ts +27 -967
  119. package/src/lib/server/memory-db.ts +4 -6
  120. package/src/lib/server/memory-tiers.ts +40 -0
  121. package/src/lib/server/openclaw-agent-resolver.test.ts +70 -0
  122. package/src/lib/server/openclaw-agent-resolver.ts +128 -0
  123. package/src/lib/server/openclaw-exec-config.ts +5 -6
  124. package/src/lib/server/openclaw-skills-normalize.test.ts +56 -0
  125. package/src/lib/server/openclaw-skills-normalize.ts +136 -0
  126. package/src/lib/server/openclaw-sync.ts +3 -2
  127. package/src/lib/server/orchestrator-lg.ts +17 -6
  128. package/src/lib/server/orchestrator.ts +2 -2
  129. package/src/lib/server/playwright-proxy.mjs +27 -3
  130. package/src/lib/server/plugins.test.ts +207 -0
  131. package/src/lib/server/plugins.ts +822 -69
  132. package/src/lib/server/provider-health.ts +33 -3
  133. package/src/lib/server/queue.ts +3 -20
  134. package/src/lib/server/scheduler.ts +2 -0
  135. package/src/lib/server/session-archive-memory.test.ts +85 -0
  136. package/src/lib/server/session-archive-memory.ts +230 -0
  137. package/src/lib/server/session-mailbox.ts +8 -18
  138. package/src/lib/server/session-reset-policy.test.ts +99 -0
  139. package/src/lib/server/session-reset-policy.ts +311 -0
  140. package/src/lib/server/session-run-manager.ts +33 -80
  141. package/src/lib/server/session-tools/autonomy-tools.test.ts +105 -0
  142. package/src/lib/server/session-tools/calendar.ts +2 -12
  143. package/src/lib/server/session-tools/connector.ts +109 -8
  144. package/src/lib/server/session-tools/context.ts +14 -2
  145. package/src/lib/server/session-tools/crawl.ts +447 -0
  146. package/src/lib/server/session-tools/crud.ts +70 -32
  147. package/src/lib/server/session-tools/delegate-fallback.test.ts +219 -0
  148. package/src/lib/server/session-tools/delegate.ts +406 -20
  149. package/src/lib/server/session-tools/discovery.ts +22 -4
  150. package/src/lib/server/session-tools/document.ts +283 -0
  151. package/src/lib/server/session-tools/email.ts +1 -3
  152. package/src/lib/server/session-tools/extract.ts +137 -0
  153. package/src/lib/server/session-tools/file-normalize.test.ts +93 -0
  154. package/src/lib/server/session-tools/file-send.test.ts +84 -1
  155. package/src/lib/server/session-tools/file.ts +237 -24
  156. package/src/lib/server/session-tools/human-loop.ts +227 -0
  157. package/src/lib/server/session-tools/image-gen.ts +1 -3
  158. package/src/lib/server/session-tools/index.ts +56 -1
  159. package/src/lib/server/session-tools/mailbox.ts +276 -0
  160. package/src/lib/server/session-tools/memory.ts +35 -3
  161. package/src/lib/server/session-tools/monitor.ts +150 -7
  162. package/src/lib/server/session-tools/normalize-tool-args.ts +17 -14
  163. package/src/lib/server/session-tools/platform-normalize.test.ts +142 -0
  164. package/src/lib/server/session-tools/platform.ts +142 -4
  165. package/src/lib/server/session-tools/plugin-creator.ts +86 -23
  166. package/src/lib/server/session-tools/primitive-tools.test.ts +257 -0
  167. package/src/lib/server/session-tools/replicate.ts +1 -3
  168. package/src/lib/server/session-tools/schedule.ts +20 -10
  169. package/src/lib/server/session-tools/session-info.ts +36 -3
  170. package/src/lib/server/session-tools/session-tools-wiring.test.ts +31 -17
  171. package/src/lib/server/session-tools/subagent.ts +193 -27
  172. package/src/lib/server/session-tools/table.ts +587 -0
  173. package/src/lib/server/session-tools/wallet.ts +13 -10
  174. package/src/lib/server/session-tools/web-browser-config.test.ts +39 -0
  175. package/src/lib/server/session-tools/web.ts +896 -100
  176. package/src/lib/server/storage.ts +226 -7
  177. package/src/lib/server/stream-agent-chat.ts +46 -21
  178. package/src/lib/server/structured-extract.test.ts +72 -0
  179. package/src/lib/server/structured-extract.ts +373 -0
  180. package/src/lib/server/task-mention.test.ts +16 -2
  181. package/src/lib/server/task-mention.ts +61 -10
  182. package/src/lib/server/tool-aliases.ts +44 -7
  183. package/src/lib/server/tool-capability-policy.ts +6 -0
  184. package/src/lib/server/tool-retry.ts +2 -0
  185. package/src/lib/server/watch-jobs.test.ts +173 -0
  186. package/src/lib/server/watch-jobs.ts +532 -0
  187. package/src/lib/server/ws-hub.ts +5 -3
  188. package/src/lib/validation/schemas.test.ts +26 -0
  189. package/src/lib/validation/schemas.ts +7 -0
  190. package/src/lib/ws-client.ts +14 -12
  191. package/src/proxy.ts +5 -5
  192. package/src/stores/use-app-store.ts +0 -6
  193. package/src/stores/use-chat-store.ts +31 -2
  194. package/src/types/index.ts +287 -44
  195. package/src/components/chat/new-chat-sheet.tsx +0 -253
  196. package/src/lib/server/main-session.ts +0 -17
  197. package/src/lib/server/session-run-manager.test.ts +0 -26
@@ -0,0 +1,207 @@
1
+ import assert from 'node:assert/strict'
2
+ import { describe, it } from 'node:test'
3
+ import fs from 'node:fs'
4
+ import path from 'node:path'
5
+ import { getPluginManager, normalizeMarketplacePluginUrl, sanitizePluginFilename } from './plugins'
6
+ import { canonicalizePluginId, expandPluginIds } from './tool-aliases'
7
+ import { DATA_DIR } from './data-dir'
8
+
9
+ let testPluginSeq = 0
10
+
11
+ function uniquePluginId(prefix: string): string {
12
+ testPluginSeq += 1
13
+ return `${prefix}_${Date.now()}_${testPluginSeq}`
14
+ }
15
+
16
+ describe('plugin id canonicalization', () => {
17
+ it('normalizes built-in aliases to canonical plugin families', () => {
18
+ assert.equal(canonicalizePluginId('session_info'), 'manage_sessions')
19
+ assert.equal(canonicalizePluginId('connectors'), 'manage_connectors')
20
+ assert.equal(canonicalizePluginId('subagent'), 'spawn_subagent')
21
+ assert.equal(canonicalizePluginId('http'), 'http_request')
22
+ assert.equal(canonicalizePluginId('human_loop'), 'ask_human')
23
+ assert.equal(canonicalizePluginId('dataframe'), 'table')
24
+ assert.equal(canonicalizePluginId('extract_structured'), 'extract')
25
+ })
26
+
27
+ it('expands aliases to include the canonical family id', () => {
28
+ const expanded = expandPluginIds(['session_info', 'http', 'human_loop'])
29
+ assert.equal(expanded.includes('manage_sessions'), true)
30
+ assert.equal(expanded.includes('session_info'), true)
31
+ assert.equal(expanded.includes('http_request'), true)
32
+ assert.equal(expanded.includes('http'), true)
33
+ assert.equal(expanded.includes('ask_human'), true)
34
+ assert.equal(expanded.includes('human_loop'), true)
35
+ })
36
+ })
37
+
38
+ describe('plugin install helpers', () => {
39
+ it('rewrites legacy marketplace URLs to the canonical raw source', () => {
40
+ const normalized = normalizeMarketplacePluginUrl('https://github.com/swarmclawai/plugins/blob/master/foo/bar.js')
41
+ assert.equal(normalized, 'https://raw.githubusercontent.com/swarmclawai/swarmforge/main/foo/bar.js')
42
+ })
43
+
44
+ it('allows .js and .mjs plugin filenames and blocks traversal', () => {
45
+ assert.equal(sanitizePluginFilename('plugin.js'), 'plugin.js')
46
+ assert.equal(sanitizePluginFilename('plugin.mjs'), 'plugin.mjs')
47
+ assert.throws(() => sanitizePluginFilename('../plugin.js'), /Invalid filename/)
48
+ assert.throws(() => sanitizePluginFilename('plugin.ts'), /Filename must end/)
49
+ })
50
+ })
51
+
52
+ describe('plugin manager hook execution', () => {
53
+ it('applies beforeToolExec mutations only for explicitly enabled plugins', async () => {
54
+ const pluginId = uniquePluginId('before_tool_exec')
55
+ getPluginManager().registerBuiltin(pluginId, {
56
+ name: 'Before Tool Exec Test',
57
+ hooks: {
58
+ beforeToolExec: ({ input }) => ({ ...(input || {}), patched: true }),
59
+ },
60
+ })
61
+
62
+ const withoutEnable = await getPluginManager().runBeforeToolExec(
63
+ { toolName: 'shell', input: { original: true } },
64
+ {},
65
+ )
66
+ assert.deepEqual(withoutEnable, { original: true })
67
+
68
+ const withEnable = await getPluginManager().runBeforeToolExec(
69
+ { toolName: 'shell', input: { original: true } },
70
+ { enabledIds: [pluginId] },
71
+ )
72
+ assert.deepEqual(withEnable, { original: true, patched: true })
73
+ })
74
+
75
+ it('chains text transforms in plugin order', async () => {
76
+ const pluginA = uniquePluginId('transform_a')
77
+ const pluginB = uniquePluginId('transform_b')
78
+ getPluginManager().registerBuiltin(pluginA, {
79
+ name: 'Transform A',
80
+ hooks: {
81
+ transformOutboundMessage: ({ text }) => `${text} A`,
82
+ },
83
+ })
84
+ getPluginManager().registerBuiltin(pluginB, {
85
+ name: 'Transform B',
86
+ hooks: {
87
+ transformOutboundMessage: ({ text }) => `${text} B`,
88
+ },
89
+ })
90
+
91
+ const transformed = await getPluginManager().transformText(
92
+ 'transformOutboundMessage',
93
+ {
94
+ session: {
95
+ id: 's1',
96
+ name: 'Test Session',
97
+ cwd: process.cwd(),
98
+ user: 'tester',
99
+ provider: 'openai',
100
+ model: 'gpt-test',
101
+ claudeSessionId: null,
102
+ messages: [],
103
+ createdAt: Date.now(),
104
+ lastActiveAt: Date.now(),
105
+ plugins: [pluginA, pluginB],
106
+ },
107
+ text: 'base',
108
+ },
109
+ { enabledIds: [pluginA, pluginB] },
110
+ )
111
+
112
+ assert.equal(transformed, 'base A B')
113
+ })
114
+
115
+ it('does not run generic hooks unless scope is provided explicitly', async () => {
116
+ const pluginId = uniquePluginId('scoped_hook')
117
+ let callCount = 0
118
+ getPluginManager().registerBuiltin(pluginId, {
119
+ name: 'Scoped Hook Test',
120
+ hooks: {
121
+ afterChatTurn: () => {
122
+ callCount += 1
123
+ },
124
+ },
125
+ })
126
+
127
+ await getPluginManager().runHook(
128
+ 'afterChatTurn',
129
+ {
130
+ session: {
131
+ id: 's2',
132
+ name: 'Scoped Hook Session',
133
+ cwd: process.cwd(),
134
+ user: 'tester',
135
+ provider: 'openai',
136
+ model: 'gpt-test',
137
+ claudeSessionId: null,
138
+ messages: [],
139
+ createdAt: Date.now(),
140
+ lastActiveAt: Date.now(),
141
+ },
142
+ message: 'hi',
143
+ response: 'hello',
144
+ source: 'chat',
145
+ internal: false,
146
+ },
147
+ {},
148
+ )
149
+ assert.equal(callCount, 0)
150
+
151
+ await getPluginManager().runHook(
152
+ 'afterChatTurn',
153
+ {
154
+ session: {
155
+ id: 's3',
156
+ name: 'Scoped Hook Session Enabled',
157
+ cwd: process.cwd(),
158
+ user: 'tester',
159
+ provider: 'openai',
160
+ model: 'gpt-test',
161
+ claudeSessionId: null,
162
+ messages: [],
163
+ createdAt: Date.now(),
164
+ lastActiveAt: Date.now(),
165
+ plugins: [pluginId],
166
+ },
167
+ message: 'hi',
168
+ response: 'hello',
169
+ source: 'chat',
170
+ internal: false,
171
+ },
172
+ { enabledIds: [pluginId] },
173
+ )
174
+ assert.equal(callCount, 1)
175
+ })
176
+
177
+ it('stores dependency-aware plugins in managed workspaces', async () => {
178
+ const filename = `${uniquePluginId('workspace_plugin')}.js`
179
+ const manager = getPluginManager()
180
+
181
+ await manager.savePluginSource(
182
+ filename,
183
+ 'module.exports = { name: "Workspace Plugin", tools: [] }',
184
+ {
185
+ packageJson: {
186
+ name: 'workspace-plugin',
187
+ dependencies: {
188
+ lodash: '^4.17.21',
189
+ },
190
+ },
191
+ packageManager: 'npm',
192
+ },
193
+ )
194
+
195
+ const meta = manager.listPlugins().find((plugin) => plugin.filename === filename)
196
+ assert.equal(meta?.isBuiltin, false)
197
+ assert.equal(meta?.hasDependencyManifest, true)
198
+ assert.equal(meta?.dependencyCount, 1)
199
+ assert.equal(meta?.packageManager, 'npm')
200
+ assert.equal(manager.readPluginSource(filename).includes('Workspace Plugin'), true)
201
+
202
+ const shimPath = path.join(DATA_DIR, 'plugins', filename)
203
+ assert.equal(fs.readFileSync(shimPath, 'utf8').includes('Auto-generated plugin workspace shim'), true)
204
+
205
+ assert.equal(manager.deletePlugin(filename), true)
206
+ })
207
+ })