@swarmclawai/swarmclaw 0.7.1 → 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.
- package/README.md +155 -150
- package/package.json +1 -1
- package/src/app/api/agents/[id]/route.ts +26 -0
- package/src/app/api/agents/[id]/thread/route.ts +37 -9
- package/src/app/api/agents/route.ts +13 -2
- package/src/app/api/auth/route.ts +76 -7
- package/src/app/api/chatrooms/[id]/chat/route.ts +7 -2
- package/src/app/api/{sessions → chats}/[id]/browser/route.ts +5 -1
- package/src/app/api/{sessions → chats}/[id]/chat/route.ts +7 -3
- package/src/app/api/{sessions → chats}/[id]/checkpoints/route.ts +1 -1
- package/src/app/api/chats/[id]/main-loop/route.ts +13 -0
- package/src/app/api/{sessions → chats}/[id]/messages/route.ts +19 -13
- package/src/app/api/{sessions → chats}/[id]/restore/route.ts +1 -1
- package/src/app/api/{sessions → chats}/[id]/route.ts +22 -52
- package/src/app/api/{sessions → chats}/[id]/stop/route.ts +6 -1
- package/src/app/api/{sessions → chats}/route.ts +21 -7
- package/src/app/api/connectors/[id]/doctor/route.ts +26 -0
- package/src/app/api/connectors/doctor/route.ts +13 -0
- package/src/app/api/files/open/route.ts +16 -14
- package/src/app/api/memory/maintenance/route.ts +11 -1
- package/src/app/api/openclaw/agent-files/route.ts +27 -4
- package/src/app/api/openclaw/skills/route.ts +11 -3
- package/src/app/api/plugins/dependencies/route.ts +24 -0
- package/src/app/api/plugins/install/route.ts +15 -92
- package/src/app/api/plugins/route.ts +6 -26
- package/src/app/api/plugins/settings/route.ts +40 -0
- package/src/app/api/plugins/ui/route.ts +1 -0
- package/src/app/api/settings/route.ts +49 -7
- package/src/app/api/tasks/[id]/route.ts +15 -6
- package/src/app/api/tasks/bulk/route.ts +2 -2
- package/src/app/api/tasks/route.ts +9 -4
- package/src/app/api/usage/route.ts +30 -0
- package/src/app/api/webhooks/[id]/route.ts +8 -1
- package/src/app/page.tsx +9 -2
- package/src/cli/index.js +39 -33
- package/src/cli/index.ts +43 -49
- package/src/cli/spec.js +29 -27
- package/src/components/agents/agent-card.tsx +16 -13
- package/src/components/agents/agent-chat-list.tsx +104 -4
- package/src/components/agents/agent-list.tsx +54 -22
- package/src/components/agents/agent-sheet.tsx +209 -18
- package/src/components/agents/cron-job-form.tsx +3 -3
- package/src/components/agents/inspector-panel.tsx +110 -50
- package/src/components/auth/access-key-gate.tsx +36 -97
- package/src/components/auth/setup-wizard.tsx +5 -38
- package/src/components/chat/chat-area.tsx +39 -27
- package/src/components/{sessions/session-card.tsx → chat/chat-card.tsx} +7 -23
- package/src/components/chat/chat-header.tsx +299 -314
- package/src/components/{sessions/session-list.tsx → chat/chat-list.tsx} +11 -14
- package/src/components/chat/chat-tool-toggles.tsx +26 -17
- package/src/components/chat/checkpoint-timeline.tsx +4 -4
- package/src/components/chat/message-bubble.tsx +4 -1
- package/src/components/chat/message-list.tsx +5 -3
- package/src/components/chat/session-debug-panel.tsx +1 -1
- package/src/components/chat/tool-request-banner.tsx +3 -3
- package/src/components/chatrooms/agent-hover-card.tsx +3 -3
- package/src/components/chatrooms/chatroom-tool-request-banner.tsx +2 -2
- package/src/components/chatrooms/chatroom-view.tsx +347 -205
- package/src/components/connectors/connector-list.tsx +265 -127
- package/src/components/connectors/connector-sheet.tsx +218 -1
- package/src/components/home/home-view.tsx +129 -5
- package/src/components/layout/app-layout.tsx +392 -182
- package/src/components/layout/mobile-header.tsx +26 -8
- package/src/components/plugins/plugin-list.tsx +487 -254
- package/src/components/plugins/plugin-sheet.tsx +236 -13
- package/src/components/projects/project-detail.tsx +183 -0
- package/src/components/settings/gateway-connection-panel.tsx +1 -1
- package/src/components/shared/agent-picker-list.tsx +2 -2
- package/src/components/shared/command-palette.tsx +111 -25
- package/src/components/shared/settings/plugin-manager.tsx +20 -4
- package/src/components/shared/settings/section-capability-policy.tsx +105 -0
- package/src/components/shared/settings/section-heartbeat.tsx +78 -1
- package/src/components/shared/settings/section-orchestrator.tsx +3 -3
- package/src/components/shared/settings/section-providers.tsx +1 -1
- package/src/components/shared/settings/section-runtime-loop.tsx +5 -5
- package/src/components/shared/settings/section-secrets.tsx +6 -6
- package/src/components/shared/settings/section-user-preferences.tsx +1 -1
- package/src/components/shared/settings/section-voice.tsx +5 -1
- package/src/components/shared/settings/section-web-search.tsx +10 -2
- package/src/components/shared/settings/settings-page.tsx +244 -56
- package/src/components/tasks/approvals-panel.tsx +205 -18
- package/src/components/tasks/task-board.tsx +242 -46
- package/src/components/usage/metrics-dashboard.tsx +147 -1
- package/src/components/wallets/wallet-panel.tsx +17 -5
- package/src/components/webhooks/webhook-sheet.tsx +8 -8
- package/src/lib/auth.ts +17 -0
- package/src/lib/chat-streaming-state.test.ts +108 -0
- package/src/lib/chat-streaming-state.ts +108 -0
- package/src/lib/chat.ts +1 -1
- package/src/lib/{sessions.ts → chats.ts} +28 -18
- package/src/lib/openclaw-agent-id.test.ts +14 -0
- package/src/lib/openclaw-agent-id.ts +31 -0
- package/src/lib/providers/claude-cli.ts +1 -1
- package/src/lib/server/agent-assignment.test.ts +112 -0
- package/src/lib/server/agent-assignment.ts +169 -0
- package/src/lib/server/approval-connector-notify.test.ts +253 -0
- package/src/lib/server/approvals-auto-approve.test.ts +205 -0
- package/src/lib/server/approvals.ts +483 -75
- package/src/lib/server/autonomy-runtime.test.ts +341 -0
- package/src/lib/server/browser-state.test.ts +118 -0
- package/src/lib/server/browser-state.ts +123 -0
- package/src/lib/server/build-llm.test.ts +36 -0
- package/src/lib/server/build-llm.ts +11 -4
- package/src/lib/server/builtin-plugins.ts +34 -0
- package/src/lib/server/capability-router.ts +10 -8
- package/src/lib/server/chat-execution-heartbeat.test.ts +40 -0
- package/src/lib/server/chat-execution-tool-events.test.ts +134 -0
- package/src/lib/server/chat-execution.ts +285 -165
- package/src/lib/server/chatroom-health.test.ts +26 -0
- package/src/lib/server/chatroom-health.ts +2 -3
- package/src/lib/server/chatroom-helpers.test.ts +67 -2
- package/src/lib/server/chatroom-helpers.ts +48 -8
- package/src/lib/server/connectors/discord.ts +175 -11
- package/src/lib/server/connectors/doctor.test.ts +80 -0
- package/src/lib/server/connectors/doctor.ts +116 -0
- package/src/lib/server/connectors/manager.ts +948 -112
- package/src/lib/server/connectors/policy.test.ts +222 -0
- package/src/lib/server/connectors/policy.ts +452 -0
- package/src/lib/server/connectors/slack.ts +188 -9
- package/src/lib/server/connectors/telegram.ts +65 -15
- package/src/lib/server/connectors/thread-context.test.ts +44 -0
- package/src/lib/server/connectors/thread-context.ts +72 -0
- package/src/lib/server/connectors/types.ts +41 -11
- package/src/lib/server/cost.ts +34 -1
- package/src/lib/server/daemon-state.ts +61 -3
- package/src/lib/server/data-dir.ts +13 -0
- package/src/lib/server/delegation-jobs.test.ts +140 -0
- package/src/lib/server/delegation-jobs.ts +248 -0
- package/src/lib/server/document-utils.test.ts +47 -0
- package/src/lib/server/document-utils.ts +397 -0
- package/src/lib/server/heartbeat-service.ts +14 -40
- package/src/lib/server/heartbeat-source.test.ts +22 -0
- package/src/lib/server/heartbeat-source.ts +7 -0
- package/src/lib/server/identity-continuity.test.ts +77 -0
- package/src/lib/server/identity-continuity.ts +127 -0
- package/src/lib/server/mailbox-utils.ts +347 -0
- package/src/lib/server/main-agent-loop.ts +28 -1103
- package/src/lib/server/memory-db.ts +4 -6
- package/src/lib/server/memory-tiers.ts +40 -0
- package/src/lib/server/openclaw-agent-resolver.test.ts +70 -0
- package/src/lib/server/openclaw-agent-resolver.ts +128 -0
- package/src/lib/server/openclaw-exec-config.ts +5 -6
- package/src/lib/server/openclaw-skills-normalize.test.ts +56 -0
- package/src/lib/server/openclaw-skills-normalize.ts +136 -0
- package/src/lib/server/openclaw-sync.ts +3 -2
- package/src/lib/server/orchestrator-lg.ts +20 -9
- package/src/lib/server/orchestrator.ts +7 -7
- package/src/lib/server/playwright-proxy.mjs +27 -3
- package/src/lib/server/plugins.test.ts +207 -0
- package/src/lib/server/plugins.ts +927 -66
- package/src/lib/server/provider-health.ts +38 -6
- package/src/lib/server/queue.ts +13 -28
- package/src/lib/server/scheduler.ts +2 -0
- package/src/lib/server/session-archive-memory.test.ts +85 -0
- package/src/lib/server/session-archive-memory.ts +230 -0
- package/src/lib/server/session-mailbox.ts +8 -18
- package/src/lib/server/session-reset-policy.test.ts +99 -0
- package/src/lib/server/session-reset-policy.ts +311 -0
- package/src/lib/server/session-run-manager.ts +33 -82
- package/src/lib/server/session-tools/autonomy-tools.test.ts +105 -0
- package/src/lib/server/session-tools/calendar.ts +366 -0
- package/src/lib/server/session-tools/canvas.ts +1 -1
- package/src/lib/server/session-tools/chatroom.ts +4 -2
- package/src/lib/server/session-tools/connector.ts +114 -10
- package/src/lib/server/session-tools/context.ts +21 -5
- package/src/lib/server/session-tools/crawl.ts +447 -0
- package/src/lib/server/session-tools/crud.ts +74 -28
- package/src/lib/server/session-tools/delegate-fallback.test.ts +219 -0
- package/src/lib/server/session-tools/delegate.ts +497 -24
- package/src/lib/server/session-tools/discovery.ts +24 -6
- package/src/lib/server/session-tools/document.ts +283 -0
- package/src/lib/server/session-tools/edit_file.ts +4 -2
- package/src/lib/server/session-tools/email.ts +320 -0
- package/src/lib/server/session-tools/extract.ts +137 -0
- package/src/lib/server/session-tools/file-normalize.test.ts +93 -0
- package/src/lib/server/session-tools/file-send.test.ts +84 -1
- package/src/lib/server/session-tools/file.ts +241 -25
- package/src/lib/server/session-tools/git.ts +1 -1
- package/src/lib/server/session-tools/http.ts +1 -1
- package/src/lib/server/session-tools/human-loop.ts +227 -0
- package/src/lib/server/session-tools/image-gen.ts +380 -0
- package/src/lib/server/session-tools/index.ts +130 -50
- package/src/lib/server/session-tools/mailbox.ts +276 -0
- package/src/lib/server/session-tools/memory.ts +172 -3
- package/src/lib/server/session-tools/monitor.ts +151 -8
- package/src/lib/server/session-tools/normalize-tool-args.ts +17 -14
- package/src/lib/server/session-tools/openclaw-nodes.ts +1 -1
- package/src/lib/server/session-tools/openclaw-workspace.ts +1 -1
- package/src/lib/server/session-tools/platform-normalize.test.ts +142 -0
- package/src/lib/server/session-tools/platform.ts +148 -7
- package/src/lib/server/session-tools/plugin-creator.ts +89 -26
- package/src/lib/server/session-tools/primitive-tools.test.ts +257 -0
- package/src/lib/server/session-tools/replicate.ts +301 -0
- package/src/lib/server/session-tools/sample-ui.ts +1 -1
- package/src/lib/server/session-tools/sandbox.ts +4 -2
- package/src/lib/server/session-tools/schedule.ts +24 -12
- package/src/lib/server/session-tools/session-info.ts +43 -7
- package/src/lib/server/session-tools/session-tools-wiring.test.ts +31 -17
- package/src/lib/server/session-tools/shell.ts +5 -2
- package/src/lib/server/session-tools/subagent.ts +194 -28
- package/src/lib/server/session-tools/table.ts +587 -0
- package/src/lib/server/session-tools/wallet.ts +42 -12
- package/src/lib/server/session-tools/web-browser-config.test.ts +39 -0
- package/src/lib/server/session-tools/web.ts +926 -91
- package/src/lib/server/storage.ts +255 -16
- package/src/lib/server/stream-agent-chat.ts +116 -268
- package/src/lib/server/structured-extract.test.ts +72 -0
- package/src/lib/server/structured-extract.ts +373 -0
- package/src/lib/server/task-mention.test.ts +16 -2
- package/src/lib/server/task-mention.ts +61 -10
- package/src/lib/server/tool-aliases.ts +66 -18
- package/src/lib/server/tool-capability-policy.test.ts +9 -9
- package/src/lib/server/tool-capability-policy.ts +38 -27
- package/src/lib/server/tool-retry.ts +2 -0
- package/src/lib/server/watch-jobs.test.ts +173 -0
- package/src/lib/server/watch-jobs.ts +532 -0
- package/src/lib/server/ws-hub.ts +5 -3
- package/src/lib/tool-definitions.ts +4 -0
- package/src/lib/validation/schemas.test.ts +26 -0
- package/src/lib/validation/schemas.ts +10 -1
- package/src/lib/ws-client.ts +14 -12
- package/src/proxy.ts +5 -5
- package/src/stores/use-app-store.ts +5 -11
- package/src/stores/use-chat-store.ts +38 -9
- package/src/types/index.ts +352 -47
- package/src/app/api/sessions/[id]/main-loop/route.ts +0 -94
- package/src/components/sessions/new-session-sheet.tsx +0 -253
- package/src/lib/server/main-session.ts +0 -24
- package/src/lib/server/session-run-manager.test.ts +0 -23
- /package/src/app/api/{sessions → chats}/[id]/clear/route.ts +0 -0
- /package/src/app/api/{sessions → chats}/[id]/deploy/route.ts +0 -0
- /package/src/app/api/{sessions → chats}/[id]/devserver/route.ts +0 -0
- /package/src/app/api/{sessions → chats}/[id]/edit-resend/route.ts +0 -0
- /package/src/app/api/{sessions → chats}/[id]/fork/route.ts +0 -0
- /package/src/app/api/{sessions → chats}/[id]/mailbox/route.ts +0 -0
- /package/src/app/api/{sessions → chats}/[id]/retry/route.ts +0 -0
- /package/src/app/api/{sessions → chats}/heartbeat/route.ts +0 -0
|
@@ -42,10 +42,10 @@ export function createOrchestratorSession(
|
|
|
42
42
|
messages: [] as any[],
|
|
43
43
|
createdAt: Date.now(),
|
|
44
44
|
lastActiveAt: Date.now(),
|
|
45
|
-
sessionType: '
|
|
45
|
+
sessionType: 'human' as const,
|
|
46
46
|
agentId: orchestrator.id,
|
|
47
47
|
parentSessionId: parentSessionId || null,
|
|
48
|
-
|
|
48
|
+
plugins: Array.isArray(orchestrator.plugins) ? [...orchestrator.plugins] : (Array.isArray(orchestrator.tools) ? [...orchestrator.tools] : []),
|
|
49
49
|
heartbeatEnabled: false,
|
|
50
50
|
}
|
|
51
51
|
saveSessions(sessions)
|
|
@@ -95,9 +95,9 @@ async function executeOrchestratorLegacy(
|
|
|
95
95
|
const agentIds = orchestrator.subAgentIds || []
|
|
96
96
|
const agents = agentIds.map((id) => allAgents[id]).filter(Boolean)
|
|
97
97
|
const agentList = agents.map((a) => {
|
|
98
|
-
const
|
|
98
|
+
const plugins = (a.plugins || a.tools)?.length ? ` [plugins: ${(a.plugins || a.tools)!.join(', ')}]` : ''
|
|
99
99
|
const skills = a.skills?.length ? ` [skills: ${a.skills.join(', ')}]` : ''
|
|
100
|
-
return `- ${a.name}: ${a.description}${
|
|
100
|
+
return `- ${a.name}: ${a.description}${plugins}${skills}`
|
|
101
101
|
}).join('\n')
|
|
102
102
|
|
|
103
103
|
// Load relevant memories
|
|
@@ -288,10 +288,10 @@ async function executeSubTask(
|
|
|
288
288
|
messages: [] as any[],
|
|
289
289
|
createdAt: Date.now(),
|
|
290
290
|
lastActiveAt: Date.now(),
|
|
291
|
-
sessionType: '
|
|
291
|
+
sessionType: 'human' as const,
|
|
292
292
|
agentId: agent.id,
|
|
293
293
|
parentSessionId,
|
|
294
|
-
|
|
294
|
+
plugins: agent.plugins || agent.tools || [],
|
|
295
295
|
}
|
|
296
296
|
sessions[childId] = childSession
|
|
297
297
|
saveSessions(sessions)
|
|
@@ -339,7 +339,7 @@ export async function callProvider(
|
|
|
339
339
|
credentialId: agent.credentialId,
|
|
340
340
|
apiEndpoint: agent.apiEndpoint,
|
|
341
341
|
cwd: WORKSPACE_DIR,
|
|
342
|
-
|
|
342
|
+
plugins: agent.plugins || agent.tools || [],
|
|
343
343
|
messages: history.map((h) => ({
|
|
344
344
|
role: h.role as 'user' | 'assistant',
|
|
345
345
|
text: h.text,
|
|
@@ -11,9 +11,33 @@ import path from 'path'
|
|
|
11
11
|
const UPLOAD_DIR = process.env.SWARMCLAW_UPLOAD_DIR || path.join(process.env.DATA_DIR || path.join(process.cwd(), 'data'), 'uploads')
|
|
12
12
|
if (!fs.existsSync(UPLOAD_DIR)) fs.mkdirSync(UPLOAD_DIR, { recursive: true })
|
|
13
13
|
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
14
|
+
function resolvePlaywrightCli() {
|
|
15
|
+
const candidates = [
|
|
16
|
+
path.join(process.cwd(), 'node_modules', '@playwright', 'mcp', 'cli.js'),
|
|
17
|
+
path.join(process.cwd(), '[project]', 'node_modules', '@playwright', 'mcp', 'cli.js'),
|
|
18
|
+
]
|
|
19
|
+
return candidates.find((candidate) => fs.existsSync(candidate)) || null
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
function sanitizePlaywrightEnv(baseEnv) {
|
|
23
|
+
const env = { ...baseEnv }
|
|
24
|
+
for (const key of Object.keys(env)) {
|
|
25
|
+
if (!key.toUpperCase().startsWith('PLAYWRIGHT_MCP_')) continue
|
|
26
|
+
delete env[key]
|
|
27
|
+
}
|
|
28
|
+
return env
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
const cliPath = resolvePlaywrightCli()
|
|
32
|
+
const child = cliPath
|
|
33
|
+
? spawn(process.execPath, [cliPath], {
|
|
34
|
+
stdio: ['pipe', 'pipe', 'pipe'],
|
|
35
|
+
env: sanitizePlaywrightEnv(process.env),
|
|
36
|
+
})
|
|
37
|
+
: spawn('npx', ['@playwright/mcp@latest'], {
|
|
38
|
+
stdio: ['pipe', 'pipe', 'pipe'],
|
|
39
|
+
env: sanitizePlaywrightEnv(process.env),
|
|
40
|
+
})
|
|
17
41
|
|
|
18
42
|
// Forward stdin → child
|
|
19
43
|
process.stdin.on('data', (chunk) => child.stdin.write(chunk))
|
|
@@ -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
|
+
})
|