@swarmclawai/swarmclaw 1.2.1 → 1.2.2
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 +9 -0
- package/package.json +2 -2
- package/skills/coding-agent/SKILL.md +111 -0
- package/skills/github/SKILL.md +140 -0
- package/skills/nano-banana-pro/SKILL.md +62 -0
- package/skills/nano-banana-pro/scripts/generate_image.py +235 -0
- package/skills/nano-pdf/SKILL.md +53 -0
- package/skills/openai-image-gen/SKILL.md +78 -0
- package/skills/openai-image-gen/scripts/gen.py +328 -0
- package/skills/resourceful-problem-solving/SKILL.md +49 -0
- package/skills/skill-creator/SKILL.md +147 -0
- package/skills/skill-creator/scripts/init_skill.py +378 -0
- package/skills/skill-creator/scripts/quick_validate.py +159 -0
- package/skills/summarize/SKILL.md +77 -0
- package/src/app/api/auth/route.ts +20 -5
- package/src/app/api/chats/[id]/devserver/route.ts +13 -19
- package/src/app/api/chats/[id]/messages/route.ts +13 -15
- package/src/app/api/chats/[id]/route.ts +9 -10
- package/src/app/api/chats/[id]/stop/route.ts +5 -7
- package/src/app/api/chats/messages-route.test.ts +8 -6
- package/src/app/api/chats/route.ts +9 -10
- package/src/app/api/ip/route.ts +2 -2
- package/src/app/api/preview-server/route.ts +1 -1
- package/src/app/api/projects/[id]/route.ts +7 -46
- package/src/components/chat/chat-area.tsx +45 -23
- package/src/components/chat/message-bubble.test.ts +35 -0
- package/src/components/chat/message-bubble.tsx +19 -9
- package/src/components/chat/message-list.tsx +37 -3
- package/src/components/input/chat-input.tsx +34 -14
- package/src/instrumentation.ts +1 -1
- package/src/lib/chat/assistant-render-id.ts +3 -0
- package/src/lib/chat/chat-streaming-state.test.ts +42 -3
- package/src/lib/chat/chat-streaming-state.ts +20 -8
- package/src/lib/chat/queued-message-queue.test.ts +23 -1
- package/src/lib/chat/queued-message-queue.ts +11 -2
- package/src/lib/providers/cli-utils.test.ts +124 -0
- package/src/lib/server/activity/activity-log.ts +21 -0
- package/src/lib/server/agents/agent-availability.test.ts +10 -5
- package/src/lib/server/agents/agent-cascade.ts +79 -59
- package/src/lib/server/agents/agent-registry.ts +3 -1
- package/src/lib/server/agents/agent-repository.ts +90 -0
- package/src/lib/server/agents/delegation-job-repository.ts +53 -0
- package/src/lib/server/agents/delegation-jobs.ts +11 -4
- package/src/lib/server/agents/guardian-checkpoint-repository.ts +35 -0
- package/src/lib/server/agents/guardian.ts +2 -2
- package/src/lib/server/agents/main-agent-loop.ts +10 -3
- package/src/lib/server/agents/main-loop-state-repository.ts +38 -0
- package/src/lib/server/agents/subagent-runtime.ts +9 -6
- package/src/lib/server/agents/subagent-swarm.ts +3 -2
- package/src/lib/server/agents/task-session.ts +3 -4
- package/src/lib/server/approvals/approval-repository.ts +30 -0
- package/src/lib/server/autonomy/supervisor-incident-repository.ts +42 -0
- package/src/lib/server/chat-execution/chat-execution-types.ts +38 -0
- package/src/lib/server/chat-execution/chat-execution-utils.ts +1 -1
- package/src/lib/server/chat-execution/chat-execution.ts +84 -1926
- package/src/lib/server/chat-execution/chat-turn-finalization.ts +620 -0
- package/src/lib/server/chat-execution/chat-turn-partial-persistence.ts +221 -0
- package/src/lib/server/chat-execution/chat-turn-preflight.ts +133 -0
- package/src/lib/server/chat-execution/chat-turn-preparation.ts +817 -0
- package/src/lib/server/chat-execution/chat-turn-stream-execution.ts +296 -0
- package/src/lib/server/chat-execution/chat-turn-tool-routing.ts +5 -5
- package/src/lib/server/chat-execution/message-classifier.test.ts +329 -0
- package/src/lib/server/chat-execution/post-stream-finalization.ts +1 -1
- package/src/lib/server/chat-execution/prompt-builder.ts +11 -0
- package/src/lib/server/chat-execution/prompt-sections.ts +5 -6
- package/src/lib/server/chat-execution/situational-awareness.ts +12 -7
- package/src/lib/server/chat-execution/stream-agent-chat.ts +16 -13
- package/src/lib/server/chatrooms/chatroom-repository.ts +32 -0
- package/src/lib/server/connectors/connector-repository.ts +58 -0
- package/src/lib/server/connectors/runtime-state.test.ts +117 -0
- package/src/lib/server/credentials/credential-repository.ts +7 -0
- package/src/lib/server/gateways/gateway-profile-repository.ts +4 -0
- package/src/lib/server/memory/memory-abstract.test.ts +59 -0
- package/src/lib/server/missions/mission-repository.ts +74 -0
- package/src/lib/server/missions/mission-service/actions.ts +6 -0
- package/src/lib/server/missions/mission-service/bindings.ts +9 -0
- package/src/lib/server/missions/mission-service/context.ts +4 -0
- package/src/lib/server/missions/mission-service/core.ts +2269 -0
- package/src/lib/server/missions/mission-service/queries.ts +12 -0
- package/src/lib/server/missions/mission-service/recovery.ts +5 -0
- package/src/lib/server/missions/mission-service/ticks.ts +9 -0
- package/src/lib/server/missions/mission-service.test.ts +9 -2
- package/src/lib/server/missions/mission-service.ts +6 -2266
- package/src/lib/server/persistence/repository-utils.ts +154 -0
- package/src/lib/server/persistence/storage-context.ts +51 -0
- package/src/lib/server/persistence/transaction.ts +1 -0
- package/src/lib/server/projects/project-repository.ts +36 -0
- package/src/lib/server/projects/project-service.ts +79 -0
- package/src/lib/server/protocols/protocol-normalization.test.ts +6 -4
- package/src/lib/server/runtime/alert-dispatch.ts +1 -1
- package/src/lib/server/runtime/daemon-policy.ts +1 -1
- package/src/lib/server/runtime/daemon-state/core.ts +1570 -0
- package/src/lib/server/runtime/daemon-state/health.ts +6 -0
- package/src/lib/server/runtime/daemon-state/policy.ts +7 -0
- package/src/lib/server/runtime/daemon-state/supervisor.ts +6 -0
- package/src/lib/server/runtime/daemon-state.test.ts +48 -0
- package/src/lib/server/runtime/daemon-state.ts +3 -1470
- package/src/lib/server/runtime/estop-repository.ts +4 -0
- package/src/lib/server/runtime/estop.ts +3 -1
- package/src/lib/server/runtime/heartbeat-service.test.ts +2 -2
- package/src/lib/server/runtime/heartbeat-service.ts +55 -34
- package/src/lib/server/runtime/heartbeat-wake.ts +6 -4
- package/src/lib/server/runtime/idle-window.ts +2 -2
- package/src/lib/server/runtime/network.ts +11 -0
- package/src/lib/server/runtime/orchestrator-events.ts +2 -2
- package/src/lib/server/runtime/queue/claims.ts +4 -0
- package/src/lib/server/runtime/queue/core.ts +2079 -0
- package/src/lib/server/runtime/queue/execution.ts +7 -0
- package/src/lib/server/runtime/queue/followups.ts +4 -0
- package/src/lib/server/runtime/queue/queries.ts +12 -0
- package/src/lib/server/runtime/queue/recovery.ts +7 -0
- package/src/lib/server/runtime/queue-recovery.test.ts +48 -13
- package/src/lib/server/runtime/queue-repository.ts +17 -0
- package/src/lib/server/runtime/queue.ts +5 -2061
- package/src/lib/server/runtime/run-ledger.ts +6 -5
- package/src/lib/server/runtime/run-repository.ts +73 -0
- package/src/lib/server/runtime/runtime-lock-repository.ts +8 -0
- package/src/lib/server/runtime/runtime-settings.ts +1 -1
- package/src/lib/server/runtime/runtime-state.ts +99 -0
- package/src/lib/server/runtime/scheduler.ts +4 -2
- package/src/lib/server/runtime/session-run-manager/cancellation.ts +157 -0
- package/src/lib/server/runtime/session-run-manager/drain.ts +246 -0
- package/src/lib/server/runtime/session-run-manager/enqueue.ts +287 -0
- package/src/lib/server/runtime/session-run-manager/queries.ts +117 -0
- package/src/lib/server/runtime/session-run-manager/recovery.ts +238 -0
- package/src/lib/server/runtime/session-run-manager/state.ts +441 -0
- package/src/lib/server/runtime/session-run-manager/types.ts +74 -0
- package/src/lib/server/runtime/session-run-manager.ts +72 -1377
- package/src/lib/server/runtime/watch-job-repository.ts +35 -0
- package/src/lib/server/runtime/watch-jobs.ts +3 -1
- package/src/lib/server/schedules/schedule-repository.ts +42 -0
- package/src/lib/server/sessions/session-repository.ts +85 -0
- package/src/lib/server/settings/settings-repository.ts +25 -0
- package/src/lib/server/skills/skill-discovery.test.ts +2 -2
- package/src/lib/server/skills/skill-discovery.ts +2 -2
- package/src/lib/server/skills/skill-repository.ts +14 -0
- package/src/lib/server/storage.ts +13 -24
- package/src/lib/server/tasks/task-repository.ts +54 -0
- package/src/lib/server/usage/usage-repository.ts +30 -0
- package/src/lib/server/webhooks/webhook-repository.ts +10 -0
- package/src/lib/strip-internal-metadata.test.ts +42 -41
- package/src/stores/use-chat-store.test.ts +54 -0
- package/src/stores/use-chat-store.ts +21 -5
- /package/{bundled-skills → skills}/google-workspace/SKILL.md +0 -0
|
@@ -3,22 +3,21 @@
|
|
|
3
3
|
*
|
|
4
4
|
* When an agent is trashed, related entities (tasks, schedules, watch jobs,
|
|
5
5
|
* connectors, webhooks, delegation jobs, chatroom memberships) must be
|
|
6
|
-
* suspended to prevent phantom daemon activity.
|
|
6
|
+
* suspended to prevent phantom daemon activity. On permanent delete the
|
|
7
7
|
* referencing rows are hard-removed.
|
|
8
8
|
*/
|
|
9
9
|
|
|
10
10
|
import {
|
|
11
|
-
|
|
12
|
-
upsertStoredItems,
|
|
13
|
-
loadSchedules,
|
|
14
|
-
loadWatchJobs,
|
|
15
|
-
loadConnectors,
|
|
11
|
+
deleteDelegationJob,
|
|
16
12
|
loadDelegationJobs,
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
} from '@/lib/server/
|
|
21
|
-
import
|
|
13
|
+
saveDelegationJobRecords,
|
|
14
|
+
} from '@/lib/server/agents/delegation-job-repository'
|
|
15
|
+
import { loadChatrooms, saveChatrooms } from '@/lib/server/chatrooms/chatroom-repository'
|
|
16
|
+
import { loadConnectors, saveConnectors } from '@/lib/server/connectors/connector-repository'
|
|
17
|
+
import { deleteWatchJob, loadWatchJobs, upsertWatchJobs } from '@/lib/server/runtime/watch-job-repository'
|
|
18
|
+
import { deleteSchedule, loadSchedules, upsertSchedules } from '@/lib/server/schedules/schedule-repository'
|
|
19
|
+
import { deleteTask, loadTasks, saveTaskMany } from '@/lib/server/tasks/task-repository'
|
|
20
|
+
import { loadWebhooks, saveWebhooks } from '@/lib/server/webhooks/webhook-repository'
|
|
22
21
|
|
|
23
22
|
interface CascadeCounts {
|
|
24
23
|
tasks: number
|
|
@@ -39,7 +38,7 @@ export function suspendAgentReferences(agentId: string): CascadeCounts {
|
|
|
39
38
|
|
|
40
39
|
// 1. Tasks — cancel active ones
|
|
41
40
|
const tasks = loadTasks()
|
|
42
|
-
const taskUpdates: Array<[string, unknown]> = []
|
|
41
|
+
const taskUpdates: Array<[string, Record<string, unknown>]> = []
|
|
43
42
|
for (const t of Object.values(tasks) as unknown as Array<Record<string, unknown>>) {
|
|
44
43
|
if (!t || t.agentId !== agentId) continue
|
|
45
44
|
const status = t.status as string | undefined
|
|
@@ -50,13 +49,13 @@ export function suspendAgentReferences(agentId: string): CascadeCounts {
|
|
|
50
49
|
}
|
|
51
50
|
}
|
|
52
51
|
if (taskUpdates.length) {
|
|
53
|
-
|
|
52
|
+
saveTaskMany(taskUpdates)
|
|
54
53
|
counts.tasks = taskUpdates.length
|
|
55
54
|
}
|
|
56
55
|
|
|
57
56
|
// 2. Schedules — pause (with marker for restore)
|
|
58
57
|
const schedules = loadSchedules()
|
|
59
|
-
const schedUpdates: Array<[string, unknown]> = []
|
|
58
|
+
const schedUpdates: Array<[string, Record<string, unknown>]> = []
|
|
60
59
|
for (const s of Object.values(schedules) as unknown as Array<Record<string, unknown>>) {
|
|
61
60
|
if (!s || s.agentId !== agentId) continue
|
|
62
61
|
if (s.enabled === false) continue
|
|
@@ -65,13 +64,13 @@ export function suspendAgentReferences(agentId: string): CascadeCounts {
|
|
|
65
64
|
schedUpdates.push([s.id as string, s])
|
|
66
65
|
}
|
|
67
66
|
if (schedUpdates.length) {
|
|
68
|
-
|
|
67
|
+
upsertSchedules(schedUpdates)
|
|
69
68
|
counts.schedules = schedUpdates.length
|
|
70
69
|
}
|
|
71
70
|
|
|
72
71
|
// 3. Watch jobs — cancel active
|
|
73
72
|
const watchJobs = loadWatchJobs()
|
|
74
|
-
const wjUpdates: Array<[string, unknown]> = []
|
|
73
|
+
const wjUpdates: Array<[string, Record<string, unknown>]> = []
|
|
75
74
|
for (const w of Object.values(watchJobs) as unknown as Array<Record<string, unknown>>) {
|
|
76
75
|
if (!w || w.agentId !== agentId) continue
|
|
77
76
|
if (w.status === 'cancelled') continue
|
|
@@ -79,26 +78,28 @@ export function suspendAgentReferences(agentId: string): CascadeCounts {
|
|
|
79
78
|
wjUpdates.push([w.id as string, w])
|
|
80
79
|
}
|
|
81
80
|
if (wjUpdates.length) {
|
|
82
|
-
|
|
81
|
+
upsertWatchJobs(wjUpdates)
|
|
83
82
|
counts.watchJobs = wjUpdates.length
|
|
84
83
|
}
|
|
85
84
|
|
|
86
85
|
// 4. Connectors — detach agent (keep connector alive but unrouted)
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
86
|
+
{
|
|
87
|
+
const connectors = loadConnectors()
|
|
88
|
+
let connectorUpdates = 0
|
|
89
|
+
for (const c of Object.values(connectors)) {
|
|
90
|
+
if (!c || c.agentId !== agentId) continue
|
|
91
|
+
c.agentId = null
|
|
92
|
+
connectorUpdates += 1
|
|
93
|
+
}
|
|
94
|
+
if (connectorUpdates > 0) {
|
|
95
|
+
saveConnectors(connectors)
|
|
96
|
+
counts.connectors = connectorUpdates
|
|
97
|
+
}
|
|
97
98
|
}
|
|
98
99
|
|
|
99
100
|
// 5. Delegation jobs — cancel queued/running
|
|
100
101
|
const delegationJobs = loadDelegationJobs()
|
|
101
|
-
const djUpdates: Array<[string, unknown]> = []
|
|
102
|
+
const djUpdates: Array<[string, Record<string, unknown>]> = []
|
|
102
103
|
for (const d of Object.values(delegationJobs) as unknown as Array<Record<string, unknown>>) {
|
|
103
104
|
if (!d || d.agentId !== agentId) continue
|
|
104
105
|
const status = d.status as string | undefined
|
|
@@ -108,22 +109,22 @@ export function suspendAgentReferences(agentId: string): CascadeCounts {
|
|
|
108
109
|
}
|
|
109
110
|
}
|
|
110
111
|
if (djUpdates.length) {
|
|
111
|
-
|
|
112
|
+
saveDelegationJobRecords(djUpdates)
|
|
112
113
|
counts.delegationJobs = djUpdates.length
|
|
113
114
|
}
|
|
114
115
|
|
|
115
116
|
// 6. Webhooks — disable
|
|
116
117
|
const webhooks = loadWebhooks()
|
|
117
|
-
|
|
118
|
-
for (const w of Object.values(webhooks)
|
|
118
|
+
let webhookUpdates = 0
|
|
119
|
+
for (const w of Object.values(webhooks)) {
|
|
119
120
|
if (!w || w.agentId !== agentId) continue
|
|
120
121
|
if (w.enabled === false) continue
|
|
121
122
|
w.enabled = false
|
|
122
|
-
|
|
123
|
+
webhookUpdates += 1
|
|
123
124
|
}
|
|
124
|
-
if (
|
|
125
|
-
|
|
126
|
-
counts.webhooks =
|
|
125
|
+
if (webhookUpdates > 0) {
|
|
126
|
+
saveWebhooks(webhooks)
|
|
127
|
+
counts.webhooks = webhookUpdates
|
|
127
128
|
}
|
|
128
129
|
|
|
129
130
|
// 7. Chatrooms — remove agent from member arrays
|
|
@@ -138,23 +139,25 @@ export function suspendAgentReferences(agentId: string): CascadeCounts {
|
|
|
138
139
|
export function purgeAgentReferences(agentId: string): CascadeCounts {
|
|
139
140
|
const counts: CascadeCounts = { tasks: 0, schedules: 0, watchJobs: 0, connectors: 0, delegationJobs: 0, webhooks: 0, chatrooms: 0 }
|
|
140
141
|
|
|
141
|
-
counts.tasks = deleteMatching(
|
|
142
|
-
counts.schedules = deleteMatching(
|
|
143
|
-
counts.watchJobs = deleteMatching(
|
|
144
|
-
counts.delegationJobs = deleteMatching(
|
|
145
|
-
counts.webhooks =
|
|
142
|
+
counts.tasks = deleteMatching(loadTasks(), agentId, deleteTask)
|
|
143
|
+
counts.schedules = deleteMatching(loadSchedules(), agentId, deleteSchedule)
|
|
144
|
+
counts.watchJobs = deleteMatching(loadWatchJobs(), agentId, deleteWatchJob)
|
|
145
|
+
counts.delegationJobs = deleteMatching(loadDelegationJobs(), agentId, deleteDelegationJob)
|
|
146
|
+
counts.webhooks = purgeWebhooks(agentId)
|
|
146
147
|
|
|
147
148
|
// Connectors: detach agent but keep the connector record
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
|
|
157
|
-
|
|
149
|
+
{
|
|
150
|
+
const connectors = loadConnectors()
|
|
151
|
+
let connectorUpdates = 0
|
|
152
|
+
for (const c of Object.values(connectors)) {
|
|
153
|
+
if (!c || c.agentId !== agentId) continue
|
|
154
|
+
c.agentId = null
|
|
155
|
+
connectorUpdates += 1
|
|
156
|
+
}
|
|
157
|
+
if (connectorUpdates > 0) {
|
|
158
|
+
saveConnectors(connectors)
|
|
159
|
+
counts.connectors = connectorUpdates
|
|
160
|
+
}
|
|
158
161
|
}
|
|
159
162
|
|
|
160
163
|
counts.chatrooms = removeAgentFromChatrooms(agentId)
|
|
@@ -167,7 +170,7 @@ export function purgeAgentReferences(agentId: string): CascadeCounts {
|
|
|
167
170
|
/** Re-enable schedules that were paused by trash. */
|
|
168
171
|
export function restoreAgentSchedules(agentId: string): number {
|
|
169
172
|
const schedules = loadSchedules()
|
|
170
|
-
const updates: Array<[string, unknown]> = []
|
|
173
|
+
const updates: Array<[string, Record<string, unknown>]> = []
|
|
171
174
|
for (const s of Object.values(schedules) as unknown as Array<Record<string, unknown>>) {
|
|
172
175
|
if (!s || s.agentId !== agentId) continue
|
|
173
176
|
if (!s.suspendedByTrash) continue
|
|
@@ -176,7 +179,7 @@ export function restoreAgentSchedules(agentId: string): number {
|
|
|
176
179
|
updates.push([s.id as string, s])
|
|
177
180
|
}
|
|
178
181
|
if (updates.length) {
|
|
179
|
-
|
|
182
|
+
upsertSchedules(updates)
|
|
180
183
|
}
|
|
181
184
|
return updates.length
|
|
182
185
|
}
|
|
@@ -184,15 +187,15 @@ export function restoreAgentSchedules(agentId: string): number {
|
|
|
184
187
|
// ── Internals ───────────────────────────────────────────────────────────
|
|
185
188
|
|
|
186
189
|
function deleteMatching<T extends { agentId?: string | null; id?: string | null }>(
|
|
187
|
-
table: StorageCollection,
|
|
188
190
|
collection: Record<string, T>,
|
|
189
191
|
agentId: string,
|
|
192
|
+
deleteItem: (id: string) => void,
|
|
190
193
|
): number {
|
|
191
194
|
let count = 0
|
|
192
195
|
for (const item of Object.values(collection)) {
|
|
193
196
|
if (!item || item.agentId !== agentId) continue
|
|
194
197
|
if (!item.id) continue
|
|
195
|
-
|
|
198
|
+
deleteItem(item.id)
|
|
196
199
|
count++
|
|
197
200
|
}
|
|
198
201
|
return count
|
|
@@ -200,7 +203,7 @@ function deleteMatching<T extends { agentId?: string | null; id?: string | null
|
|
|
200
203
|
|
|
201
204
|
function removeAgentFromChatrooms(agentId: string): number {
|
|
202
205
|
const chatrooms = loadChatrooms()
|
|
203
|
-
|
|
206
|
+
let changedCount = 0
|
|
204
207
|
for (const room of Object.values(chatrooms) as unknown as Array<Record<string, unknown>>) {
|
|
205
208
|
if (!room) continue
|
|
206
209
|
let changed = false
|
|
@@ -220,10 +223,27 @@ function removeAgentFromChatrooms(agentId: string): number {
|
|
|
220
223
|
room.agentIds = agentIds.filter((id) => id !== agentId)
|
|
221
224
|
if ((room.agentIds as string[]).length !== before) changed = true
|
|
222
225
|
}
|
|
223
|
-
if (changed)
|
|
226
|
+
if (changed) changedCount += 1
|
|
224
227
|
}
|
|
225
|
-
if (
|
|
226
|
-
|
|
228
|
+
if (changedCount > 0) {
|
|
229
|
+
saveChatrooms(chatrooms)
|
|
227
230
|
}
|
|
228
|
-
return
|
|
231
|
+
return changedCount
|
|
232
|
+
}
|
|
233
|
+
|
|
234
|
+
function purgeWebhooks(agentId: string): number {
|
|
235
|
+
const webhooks = loadWebhooks() as Record<string, Record<string, unknown>>
|
|
236
|
+
const remaining: Record<string, Record<string, unknown>> = {}
|
|
237
|
+
let count = 0
|
|
238
|
+
for (const [id, webhook] of Object.entries(webhooks)) {
|
|
239
|
+
if (webhook && webhook.agentId === agentId) {
|
|
240
|
+
count += 1
|
|
241
|
+
continue
|
|
242
|
+
}
|
|
243
|
+
remaining[id] = webhook
|
|
244
|
+
}
|
|
245
|
+
if (count > 0) {
|
|
246
|
+
saveWebhooks(remaining)
|
|
247
|
+
}
|
|
248
|
+
return count
|
|
229
249
|
}
|
|
@@ -1,5 +1,7 @@
|
|
|
1
|
-
import { loadAgents, loadTasks, loadSessions } from '@/lib/server/storage'
|
|
2
1
|
import type { Agent, BoardTask } from '@/types'
|
|
2
|
+
import { loadAgents } from '@/lib/server/agents/agent-repository'
|
|
3
|
+
import { loadSessions } from '@/lib/server/sessions/session-repository'
|
|
4
|
+
import { loadTasks } from '@/lib/server/tasks/task-repository'
|
|
3
5
|
|
|
4
6
|
export interface AgentDirectoryEntry {
|
|
5
7
|
id: string
|
|
@@ -0,0 +1,90 @@
|
|
|
1
|
+
import type { Agent } from '@/types'
|
|
2
|
+
|
|
3
|
+
import {
|
|
4
|
+
deleteAgent as deleteStoredAgent,
|
|
5
|
+
loadAgent as loadStoredAgent,
|
|
6
|
+
loadAgents as loadStoredAgents,
|
|
7
|
+
loadTrashedAgents as loadStoredTrashedAgents,
|
|
8
|
+
patchAgent as patchStoredAgent,
|
|
9
|
+
saveAgents as saveStoredAgents,
|
|
10
|
+
upsertAgent as upsertStoredAgent,
|
|
11
|
+
} from '@/lib/server/storage'
|
|
12
|
+
import { createRecordRepository } from '@/lib/server/persistence/repository-utils'
|
|
13
|
+
|
|
14
|
+
type AgentRepositoryOptions = { includeTrashed?: boolean }
|
|
15
|
+
|
|
16
|
+
export const agentRepository = createRecordRepository<Agent, AgentRepositoryOptions>(
|
|
17
|
+
'agents',
|
|
18
|
+
{
|
|
19
|
+
get(id, options) {
|
|
20
|
+
return loadStoredAgent(id, options) as Agent | null
|
|
21
|
+
},
|
|
22
|
+
list(options) {
|
|
23
|
+
return loadStoredAgents(options) as Record<string, Agent>
|
|
24
|
+
},
|
|
25
|
+
upsert(id, value) {
|
|
26
|
+
upsertStoredAgent(id, value)
|
|
27
|
+
},
|
|
28
|
+
replace(data) {
|
|
29
|
+
saveStoredAgents(data)
|
|
30
|
+
},
|
|
31
|
+
patch(id, updater) {
|
|
32
|
+
return patchStoredAgent(id, updater as (current: Agent | null) => Agent | null) as Agent | null
|
|
33
|
+
},
|
|
34
|
+
delete(id) {
|
|
35
|
+
deleteStoredAgent(id)
|
|
36
|
+
},
|
|
37
|
+
},
|
|
38
|
+
)
|
|
39
|
+
|
|
40
|
+
export function getAgent(id: string, options?: AgentRepositoryOptions): Agent | null {
|
|
41
|
+
return agentRepository.get(id, options)
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
export function getAgents(ids: string[], options?: AgentRepositoryOptions): Record<string, Agent> {
|
|
45
|
+
return agentRepository.getMany(ids, options)
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
export function listAgents(options?: AgentRepositoryOptions): Record<string, Agent> {
|
|
49
|
+
return agentRepository.list(options)
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
export function saveAgent(id: string, agent: Agent | Record<string, unknown>): void {
|
|
53
|
+
agentRepository.upsert(id, agent as Agent)
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
export function saveAgentMany(entries: Array<[string, Agent | Record<string, unknown>]>): void {
|
|
57
|
+
agentRepository.upsertMany(entries as Array<[string, Agent]>)
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
export function replaceAgents(agents: Record<string, Agent | Record<string, unknown>>): void {
|
|
61
|
+
agentRepository.replace(agents as Record<string, Agent>)
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
export function patchAgent(id: string, updater: (current: Agent | null) => Agent | null): Agent | null {
|
|
65
|
+
return agentRepository.patch(id, updater)
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
export function deleteAgent(id: string): void {
|
|
69
|
+
agentRepository.delete(id)
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
export function loadAgents(options?: AgentRepositoryOptions): Record<string, Agent> {
|
|
73
|
+
return listAgents(options)
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
export function loadAgent(id: string, options?: AgentRepositoryOptions): Agent | null {
|
|
77
|
+
return getAgent(id, options)
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
export function saveAgents(agents: Record<string, Agent | Record<string, unknown>>): void {
|
|
81
|
+
replaceAgents(agents)
|
|
82
|
+
}
|
|
83
|
+
|
|
84
|
+
export function upsertAgent(id: string, agent: Agent | Record<string, unknown>): void {
|
|
85
|
+
saveAgent(id, agent)
|
|
86
|
+
}
|
|
87
|
+
|
|
88
|
+
export function loadTrashedAgents(): Record<string, Agent> {
|
|
89
|
+
return loadStoredTrashedAgents() as Record<string, Agent>
|
|
90
|
+
}
|
|
@@ -0,0 +1,53 @@
|
|
|
1
|
+
import type { DelegationJobRecord } from '@/types'
|
|
2
|
+
|
|
3
|
+
import {
|
|
4
|
+
deleteDelegationJob as deleteStoredDelegationJob,
|
|
5
|
+
loadDelegationJobItem as loadStoredDelegationJob,
|
|
6
|
+
loadDelegationJobs as loadStoredDelegationJobs,
|
|
7
|
+
patchDelegationJob as patchStoredDelegationJob,
|
|
8
|
+
upsertDelegationJob as upsertStoredDelegationJob,
|
|
9
|
+
} from '@/lib/server/storage'
|
|
10
|
+
import { createRecordRepository } from '@/lib/server/persistence/repository-utils'
|
|
11
|
+
|
|
12
|
+
export const delegationJobRepository = createRecordRepository<DelegationJobRecord>(
|
|
13
|
+
'delegation-jobs',
|
|
14
|
+
{
|
|
15
|
+
get(id) {
|
|
16
|
+
return loadStoredDelegationJob(id) as DelegationJobRecord | null
|
|
17
|
+
},
|
|
18
|
+
list() {
|
|
19
|
+
return loadStoredDelegationJobs() as Record<string, DelegationJobRecord>
|
|
20
|
+
},
|
|
21
|
+
upsert(id, value) {
|
|
22
|
+
upsertStoredDelegationJob(id, value as DelegationJobRecord)
|
|
23
|
+
},
|
|
24
|
+
patch(id, updater) {
|
|
25
|
+
return patchStoredDelegationJob(
|
|
26
|
+
id,
|
|
27
|
+
updater as (current: DelegationJobRecord | null) => DelegationJobRecord | null,
|
|
28
|
+
) as DelegationJobRecord | null
|
|
29
|
+
},
|
|
30
|
+
delete(id) {
|
|
31
|
+
deleteStoredDelegationJob(id)
|
|
32
|
+
},
|
|
33
|
+
},
|
|
34
|
+
)
|
|
35
|
+
|
|
36
|
+
export const getDelegationJobRecord = (id: string) => delegationJobRepository.get(id)
|
|
37
|
+
export const getDelegationJobRecords = (ids: string[]) => delegationJobRepository.getMany(ids)
|
|
38
|
+
export const listDelegationJobRecords = () => delegationJobRepository.list()
|
|
39
|
+
export const saveDelegationJobRecord = (id: string, value: DelegationJobRecord | Record<string, unknown>) =>
|
|
40
|
+
delegationJobRepository.upsert(id, value as DelegationJobRecord)
|
|
41
|
+
export const saveDelegationJobRecords = (entries: Array<[string, DelegationJobRecord | Record<string, unknown>]>) =>
|
|
42
|
+
delegationJobRepository.upsertMany(entries as Array<[string, DelegationJobRecord]>)
|
|
43
|
+
export const patchDelegationJobRecord = (
|
|
44
|
+
id: string,
|
|
45
|
+
updater: (current: DelegationJobRecord | null) => DelegationJobRecord | null,
|
|
46
|
+
) => delegationJobRepository.patch(id, updater)
|
|
47
|
+
export const deleteDelegationJobRecord = (id: string) => delegationJobRepository.delete(id)
|
|
48
|
+
|
|
49
|
+
export const loadDelegationJobs = listDelegationJobRecords
|
|
50
|
+
export const loadDelegationJob = getDelegationJobRecord
|
|
51
|
+
export const upsertDelegationJob = saveDelegationJobRecord
|
|
52
|
+
export const patchDelegationJob = patchDelegationJobRecord
|
|
53
|
+
export const deleteDelegationJob = deleteDelegationJobRecord
|
|
@@ -1,13 +1,20 @@
|
|
|
1
1
|
import { genId } from '@/lib/id'
|
|
2
2
|
import { hmrSingleton } from '@/lib/shared-utils'
|
|
3
3
|
import type { DelegationJobArtifact, DelegationJobCheckpoint, DelegationJobRecord, DelegationJobStatus } from '@/types'
|
|
4
|
-
import {
|
|
4
|
+
import {
|
|
5
|
+
loadDelegationJob,
|
|
6
|
+
loadDelegationJobs,
|
|
7
|
+
patchDelegationJob,
|
|
8
|
+
upsertDelegationJob,
|
|
9
|
+
} from '@/lib/server/agents/delegation-job-repository'
|
|
10
|
+
import { logActivity } from '@/lib/server/activity/activity-log'
|
|
5
11
|
import { logExecution } from '@/lib/server/execution-log'
|
|
6
12
|
import { log } from '@/lib/server/logger'
|
|
7
13
|
import { debug } from '@/lib/server/debug'
|
|
8
14
|
import { createNotification } from '@/lib/server/create-notification'
|
|
9
15
|
import { enqueueSystemEvent } from '@/lib/server/runtime/system-events'
|
|
10
16
|
import { ensureDelegationMission, syncDelegationMissionFromJob } from '@/lib/server/missions/mission-service'
|
|
17
|
+
import { loadSession } from '@/lib/server/sessions/session-repository'
|
|
11
18
|
import { notify } from '@/lib/server/ws-hub'
|
|
12
19
|
|
|
13
20
|
interface DelegationRuntimeHandle {
|
|
@@ -115,7 +122,7 @@ export function createDelegationJob(input: CreateDelegationJobInput): Delegation
|
|
|
115
122
|
}
|
|
116
123
|
|
|
117
124
|
export function getDelegationJob(id: string): DelegationJobRecord | null {
|
|
118
|
-
const current =
|
|
125
|
+
const current = loadDelegationJob(id)
|
|
119
126
|
if (!current || typeof current !== 'object') return null
|
|
120
127
|
return current as DelegationJobRecord
|
|
121
128
|
}
|
|
@@ -144,9 +151,9 @@ export function updateDelegationJob(
|
|
|
144
151
|
}
|
|
145
152
|
})
|
|
146
153
|
if (!result) return null
|
|
147
|
-
|
|
154
|
+
syncDelegationMissionFromJob(id)
|
|
148
155
|
notifyDelegationJobsChanged()
|
|
149
|
-
return getDelegationJob(id) ||
|
|
156
|
+
return getDelegationJob(id) || result
|
|
150
157
|
}
|
|
151
158
|
|
|
152
159
|
export function appendDelegationCheckpoint(
|
|
@@ -0,0 +1,35 @@
|
|
|
1
|
+
import type { GuardianCheckpoint } from '@/types'
|
|
2
|
+
|
|
3
|
+
import {
|
|
4
|
+
loadGuardianCheckpoints as loadStoredGuardianCheckpoints,
|
|
5
|
+
patchGuardianCheckpoint as patchStoredGuardianCheckpoint,
|
|
6
|
+
upsertGuardianCheckpoint as upsertStoredGuardianCheckpoint,
|
|
7
|
+
} from '@/lib/server/storage'
|
|
8
|
+
import { createRecordRepository } from '@/lib/server/persistence/repository-utils'
|
|
9
|
+
|
|
10
|
+
export const guardianCheckpointRepository = createRecordRepository<GuardianCheckpoint>(
|
|
11
|
+
'guardian-checkpoints',
|
|
12
|
+
{
|
|
13
|
+
get(id) {
|
|
14
|
+
return loadStoredGuardianCheckpoints()[id] || null
|
|
15
|
+
},
|
|
16
|
+
list() {
|
|
17
|
+
return loadStoredGuardianCheckpoints()
|
|
18
|
+
},
|
|
19
|
+
upsert(id, value) {
|
|
20
|
+
upsertStoredGuardianCheckpoint(id, value as GuardianCheckpoint)
|
|
21
|
+
},
|
|
22
|
+
patch(id, updater) {
|
|
23
|
+
return patchStoredGuardianCheckpoint(id, updater)
|
|
24
|
+
},
|
|
25
|
+
},
|
|
26
|
+
)
|
|
27
|
+
|
|
28
|
+
export const loadGuardianCheckpoints = () => guardianCheckpointRepository.list()
|
|
29
|
+
export const loadGuardianCheckpoint = (id: string) => guardianCheckpointRepository.get(id)
|
|
30
|
+
export const upsertGuardianCheckpoint = (id: string, value: GuardianCheckpoint | Record<string, unknown>) =>
|
|
31
|
+
guardianCheckpointRepository.upsert(id, value as GuardianCheckpoint)
|
|
32
|
+
export const patchGuardianCheckpoint = (
|
|
33
|
+
id: string,
|
|
34
|
+
updater: (current: GuardianCheckpoint | null) => GuardianCheckpoint | null,
|
|
35
|
+
) => guardianCheckpointRepository.patch(id, updater)
|
|
@@ -1,12 +1,12 @@
|
|
|
1
1
|
import { execSync } from 'node:child_process'
|
|
2
2
|
import path from 'node:path'
|
|
3
3
|
import type { ApprovalRequest, GuardianCheckpoint } from '@/types'
|
|
4
|
+
import { loadApprovals } from '@/lib/server/approvals/approval-repository'
|
|
4
5
|
import {
|
|
5
|
-
loadApprovals,
|
|
6
6
|
loadGuardianCheckpoints,
|
|
7
7
|
patchGuardianCheckpoint,
|
|
8
8
|
upsertGuardianCheckpoint,
|
|
9
|
-
} from '@/lib/server/
|
|
9
|
+
} from '@/lib/server/agents/guardian-checkpoint-repository'
|
|
10
10
|
import { requestApproval } from '@/lib/server/approvals'
|
|
11
11
|
import { errorMessage } from '@/lib/shared-utils'
|
|
12
12
|
import { genId } from '@/lib/id'
|
|
@@ -1,10 +1,17 @@
|
|
|
1
1
|
import { hmrSingleton } from '@/lib/shared-utils'
|
|
2
2
|
import type { GoalContract, Message, MessageToolEvent, Session } from '@/types'
|
|
3
3
|
import { mergeGoalContracts, parseGoalContractFromText, parseMainLoopPlan, parseMainLoopReview } from '@/lib/server/agents/autonomy-contract'
|
|
4
|
+
import {
|
|
5
|
+
deletePersistedMainLoopState,
|
|
6
|
+
loadPersistedMainLoopState,
|
|
7
|
+
upsertPersistedMainLoopState,
|
|
8
|
+
} from '@/lib/server/agents/main-loop-state-repository'
|
|
9
|
+
import { loadAgents } from '@/lib/server/agents/agent-repository'
|
|
4
10
|
import { assessAutonomyRun } from '@/lib/server/autonomy/supervisor-reflection'
|
|
5
11
|
import { enqueueSystemEvent } from '@/lib/server/runtime/system-events'
|
|
6
|
-
import { loadAgents, loadSessions, loadSettings, loadPersistedMainLoopState, upsertPersistedMainLoopState, deletePersistedMainLoopState } from '@/lib/server/storage'
|
|
7
12
|
import { buildMissionHeartbeatPrompt as buildMissionHeartbeatPromptFromMission, getMissionForSession } from '@/lib/server/missions/mission-service'
|
|
13
|
+
import { loadSettings } from '@/lib/server/settings/settings-repository'
|
|
14
|
+
import { getSession, loadSessions } from '@/lib/server/sessions/session-repository'
|
|
8
15
|
|
|
9
16
|
const LEGACY_META_LINE_RE = /\[(?:MAIN_LOOP_META|MAIN_LOOP_PLAN|MAIN_LOOP_REVIEW|AGENT_HEARTBEAT_META)\]\s*(\{[^\n]*\})?/i
|
|
10
17
|
const HEARTBEAT_META_RE = /\[AGENT_HEARTBEAT_META\]\s*(\{[^\n]*\})/i
|
|
@@ -437,7 +444,7 @@ function hydrateStateFromSession(sessionId: string): MainLoopState | null {
|
|
|
437
444
|
}
|
|
438
445
|
|
|
439
446
|
function persistState(sessionId: string, state: MainLoopState): void {
|
|
440
|
-
upsertPersistedMainLoopState(sessionId, state as unknown)
|
|
447
|
+
upsertPersistedMainLoopState(sessionId, state as unknown as Record<string, unknown>)
|
|
441
448
|
}
|
|
442
449
|
|
|
443
450
|
function getOrCreateState(sessionId: string): MainLoopState | null {
|
|
@@ -738,7 +745,7 @@ export function isMainSession(session: unknown): boolean {
|
|
|
738
745
|
export function buildMainLoopHeartbeatPrompt(session: unknown, fallbackPrompt: string): string {
|
|
739
746
|
const candidate = asSession(session)
|
|
740
747
|
if (!candidate?.id) return fallbackPrompt
|
|
741
|
-
const persistedSession =
|
|
748
|
+
const persistedSession = getSession(String(candidate.id)) as Session | undefined
|
|
742
749
|
const missionPrompt = buildMissionHeartbeatPromptFromMission(
|
|
743
750
|
persistedSession || candidate as Session,
|
|
744
751
|
fallbackPrompt,
|
|
@@ -0,0 +1,38 @@
|
|
|
1
|
+
import { perf } from '@/lib/server/runtime/perf'
|
|
2
|
+
import {
|
|
3
|
+
deletePersistedMainLoopState as deleteStoredMainLoopState,
|
|
4
|
+
loadPersistedMainLoopState as loadStoredMainLoopState,
|
|
5
|
+
upsertPersistedMainLoopState as upsertStoredMainLoopState,
|
|
6
|
+
} from '@/lib/server/storage'
|
|
7
|
+
|
|
8
|
+
export type PersistedMainLoopState = Record<string, unknown>
|
|
9
|
+
|
|
10
|
+
export function loadPersistedMainLoopState(sessionId: string): PersistedMainLoopState | null {
|
|
11
|
+
return perf.measureSync(
|
|
12
|
+
'repository',
|
|
13
|
+
'main-loop-state.get',
|
|
14
|
+
() => loadStoredMainLoopState(sessionId) as PersistedMainLoopState | null,
|
|
15
|
+
{ sessionId },
|
|
16
|
+
)
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
export function upsertPersistedMainLoopState(
|
|
20
|
+
sessionId: string,
|
|
21
|
+
value: PersistedMainLoopState,
|
|
22
|
+
): void {
|
|
23
|
+
perf.measureSync(
|
|
24
|
+
'repository',
|
|
25
|
+
'main-loop-state.upsert',
|
|
26
|
+
() => upsertStoredMainLoopState(sessionId, value),
|
|
27
|
+
{ sessionId },
|
|
28
|
+
)
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
export function deletePersistedMainLoopState(sessionId: string): void {
|
|
32
|
+
perf.measureSync(
|
|
33
|
+
'repository',
|
|
34
|
+
'main-loop-state.delete',
|
|
35
|
+
() => deleteStoredMainLoopState(sessionId),
|
|
36
|
+
{ sessionId },
|
|
37
|
+
)
|
|
38
|
+
}
|
|
@@ -8,7 +8,7 @@
|
|
|
8
8
|
|
|
9
9
|
import { genId } from '@/lib/id'
|
|
10
10
|
import { DEFAULT_DELEGATION_MAX_DEPTH } from '@/lib/runtime/runtime-loop'
|
|
11
|
-
import { loadAgents
|
|
11
|
+
import { loadAgents } from '@/lib/server/agents/agent-repository'
|
|
12
12
|
import { enqueueSessionRun, type EnqueueSessionRunResult } from '@/lib/server/runtime/session-run-manager'
|
|
13
13
|
import { loadRuntimeSettings } from '@/lib/server/runtime/runtime-settings'
|
|
14
14
|
import { applyResolvedRoute, resolvePrimaryAgentRoute } from '@/lib/server/agents/agent-runtime-config'
|
|
@@ -48,6 +48,7 @@ import { debug } from '@/lib/server/debug'
|
|
|
48
48
|
import { logExecution } from '@/lib/server/execution-log'
|
|
49
49
|
import { enqueueSystemEvent } from '@/lib/server/runtime/system-events'
|
|
50
50
|
import { getEnabledCapabilityIds, splitCapabilityIds } from '@/lib/capability-selection'
|
|
51
|
+
import { getSession, loadSessions, saveSession } from '@/lib/server/sessions/session-repository'
|
|
51
52
|
|
|
52
53
|
// ---------------------------------------------------------------------------
|
|
53
54
|
// Types
|
|
@@ -276,7 +277,7 @@ async function spawnSubagentImpl(
|
|
|
276
277
|
browserProfileId,
|
|
277
278
|
}
|
|
278
279
|
sessions[sid] = applyResolvedRoute(nextSession, resolvePrimaryAgentRoute(agent))
|
|
279
|
-
|
|
280
|
+
saveSession(sid, sessions[sid])
|
|
280
281
|
|
|
281
282
|
log.info('subagent', 'Spawning', { agentId: agent.id, agentName: agent.name, depth: depth + 1, jobId: job.id, sessionId: sid })
|
|
282
283
|
logExecution(sid, 'delegation_start', `Subagent spawning: ${agent.name}`, {
|
|
@@ -398,12 +399,13 @@ async function spawnSubagentImpl(
|
|
|
398
399
|
`subagent:${job.id}`,
|
|
399
400
|
)
|
|
400
401
|
}
|
|
402
|
+
const completedSession = getSession(sid)
|
|
401
403
|
await runCapabilityHook(
|
|
402
404
|
'sessionEnd',
|
|
403
405
|
{
|
|
404
406
|
sessionId: sid,
|
|
405
|
-
session:
|
|
406
|
-
messageCount: Array.isArray(
|
|
407
|
+
session: completedSession,
|
|
408
|
+
messageCount: Array.isArray(completedSession?.messages) ? completedSession.messages.length : 0,
|
|
407
409
|
durationMs: subagentResult.durationMs,
|
|
408
410
|
reason: subagentResult.status,
|
|
409
411
|
},
|
|
@@ -460,12 +462,13 @@ async function spawnSubagentImpl(
|
|
|
460
462
|
`subagent:${job.id}`,
|
|
461
463
|
)
|
|
462
464
|
}
|
|
465
|
+
const failedSession = getSession(sid)
|
|
463
466
|
await runCapabilityHook(
|
|
464
467
|
'sessionEnd',
|
|
465
468
|
{
|
|
466
469
|
sessionId: sid,
|
|
467
|
-
session:
|
|
468
|
-
messageCount: Array.isArray(
|
|
470
|
+
session: failedSession,
|
|
471
|
+
messageCount: Array.isArray(failedSession?.messages) ? failedSession.messages.length : 0,
|
|
469
472
|
durationMs: subagentResult.durationMs,
|
|
470
473
|
reason: subagentResult.status,
|
|
471
474
|
},
|