@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.
Files changed (144) hide show
  1. package/README.md +9 -0
  2. package/package.json +2 -2
  3. package/skills/coding-agent/SKILL.md +111 -0
  4. package/skills/github/SKILL.md +140 -0
  5. package/skills/nano-banana-pro/SKILL.md +62 -0
  6. package/skills/nano-banana-pro/scripts/generate_image.py +235 -0
  7. package/skills/nano-pdf/SKILL.md +53 -0
  8. package/skills/openai-image-gen/SKILL.md +78 -0
  9. package/skills/openai-image-gen/scripts/gen.py +328 -0
  10. package/skills/resourceful-problem-solving/SKILL.md +49 -0
  11. package/skills/skill-creator/SKILL.md +147 -0
  12. package/skills/skill-creator/scripts/init_skill.py +378 -0
  13. package/skills/skill-creator/scripts/quick_validate.py +159 -0
  14. package/skills/summarize/SKILL.md +77 -0
  15. package/src/app/api/auth/route.ts +20 -5
  16. package/src/app/api/chats/[id]/devserver/route.ts +13 -19
  17. package/src/app/api/chats/[id]/messages/route.ts +13 -15
  18. package/src/app/api/chats/[id]/route.ts +9 -10
  19. package/src/app/api/chats/[id]/stop/route.ts +5 -7
  20. package/src/app/api/chats/messages-route.test.ts +8 -6
  21. package/src/app/api/chats/route.ts +9 -10
  22. package/src/app/api/ip/route.ts +2 -2
  23. package/src/app/api/preview-server/route.ts +1 -1
  24. package/src/app/api/projects/[id]/route.ts +7 -46
  25. package/src/components/chat/chat-area.tsx +45 -23
  26. package/src/components/chat/message-bubble.test.ts +35 -0
  27. package/src/components/chat/message-bubble.tsx +19 -9
  28. package/src/components/chat/message-list.tsx +37 -3
  29. package/src/components/input/chat-input.tsx +34 -14
  30. package/src/instrumentation.ts +1 -1
  31. package/src/lib/chat/assistant-render-id.ts +3 -0
  32. package/src/lib/chat/chat-streaming-state.test.ts +42 -3
  33. package/src/lib/chat/chat-streaming-state.ts +20 -8
  34. package/src/lib/chat/queued-message-queue.test.ts +23 -1
  35. package/src/lib/chat/queued-message-queue.ts +11 -2
  36. package/src/lib/providers/cli-utils.test.ts +124 -0
  37. package/src/lib/server/activity/activity-log.ts +21 -0
  38. package/src/lib/server/agents/agent-availability.test.ts +10 -5
  39. package/src/lib/server/agents/agent-cascade.ts +79 -59
  40. package/src/lib/server/agents/agent-registry.ts +3 -1
  41. package/src/lib/server/agents/agent-repository.ts +90 -0
  42. package/src/lib/server/agents/delegation-job-repository.ts +53 -0
  43. package/src/lib/server/agents/delegation-jobs.ts +11 -4
  44. package/src/lib/server/agents/guardian-checkpoint-repository.ts +35 -0
  45. package/src/lib/server/agents/guardian.ts +2 -2
  46. package/src/lib/server/agents/main-agent-loop.ts +10 -3
  47. package/src/lib/server/agents/main-loop-state-repository.ts +38 -0
  48. package/src/lib/server/agents/subagent-runtime.ts +9 -6
  49. package/src/lib/server/agents/subagent-swarm.ts +3 -2
  50. package/src/lib/server/agents/task-session.ts +3 -4
  51. package/src/lib/server/approvals/approval-repository.ts +30 -0
  52. package/src/lib/server/autonomy/supervisor-incident-repository.ts +42 -0
  53. package/src/lib/server/chat-execution/chat-execution-types.ts +38 -0
  54. package/src/lib/server/chat-execution/chat-execution-utils.ts +1 -1
  55. package/src/lib/server/chat-execution/chat-execution.ts +84 -1926
  56. package/src/lib/server/chat-execution/chat-turn-finalization.ts +620 -0
  57. package/src/lib/server/chat-execution/chat-turn-partial-persistence.ts +221 -0
  58. package/src/lib/server/chat-execution/chat-turn-preflight.ts +133 -0
  59. package/src/lib/server/chat-execution/chat-turn-preparation.ts +817 -0
  60. package/src/lib/server/chat-execution/chat-turn-stream-execution.ts +296 -0
  61. package/src/lib/server/chat-execution/chat-turn-tool-routing.ts +5 -5
  62. package/src/lib/server/chat-execution/message-classifier.test.ts +329 -0
  63. package/src/lib/server/chat-execution/post-stream-finalization.ts +1 -1
  64. package/src/lib/server/chat-execution/prompt-builder.ts +11 -0
  65. package/src/lib/server/chat-execution/prompt-sections.ts +5 -6
  66. package/src/lib/server/chat-execution/situational-awareness.ts +12 -7
  67. package/src/lib/server/chat-execution/stream-agent-chat.ts +16 -13
  68. package/src/lib/server/chatrooms/chatroom-repository.ts +32 -0
  69. package/src/lib/server/connectors/connector-repository.ts +58 -0
  70. package/src/lib/server/connectors/runtime-state.test.ts +117 -0
  71. package/src/lib/server/credentials/credential-repository.ts +7 -0
  72. package/src/lib/server/gateways/gateway-profile-repository.ts +4 -0
  73. package/src/lib/server/memory/memory-abstract.test.ts +59 -0
  74. package/src/lib/server/missions/mission-repository.ts +74 -0
  75. package/src/lib/server/missions/mission-service/actions.ts +6 -0
  76. package/src/lib/server/missions/mission-service/bindings.ts +9 -0
  77. package/src/lib/server/missions/mission-service/context.ts +4 -0
  78. package/src/lib/server/missions/mission-service/core.ts +2269 -0
  79. package/src/lib/server/missions/mission-service/queries.ts +12 -0
  80. package/src/lib/server/missions/mission-service/recovery.ts +5 -0
  81. package/src/lib/server/missions/mission-service/ticks.ts +9 -0
  82. package/src/lib/server/missions/mission-service.test.ts +9 -2
  83. package/src/lib/server/missions/mission-service.ts +6 -2266
  84. package/src/lib/server/persistence/repository-utils.ts +154 -0
  85. package/src/lib/server/persistence/storage-context.ts +51 -0
  86. package/src/lib/server/persistence/transaction.ts +1 -0
  87. package/src/lib/server/projects/project-repository.ts +36 -0
  88. package/src/lib/server/projects/project-service.ts +79 -0
  89. package/src/lib/server/protocols/protocol-normalization.test.ts +6 -4
  90. package/src/lib/server/runtime/alert-dispatch.ts +1 -1
  91. package/src/lib/server/runtime/daemon-policy.ts +1 -1
  92. package/src/lib/server/runtime/daemon-state/core.ts +1570 -0
  93. package/src/lib/server/runtime/daemon-state/health.ts +6 -0
  94. package/src/lib/server/runtime/daemon-state/policy.ts +7 -0
  95. package/src/lib/server/runtime/daemon-state/supervisor.ts +6 -0
  96. package/src/lib/server/runtime/daemon-state.test.ts +48 -0
  97. package/src/lib/server/runtime/daemon-state.ts +3 -1470
  98. package/src/lib/server/runtime/estop-repository.ts +4 -0
  99. package/src/lib/server/runtime/estop.ts +3 -1
  100. package/src/lib/server/runtime/heartbeat-service.test.ts +2 -2
  101. package/src/lib/server/runtime/heartbeat-service.ts +55 -34
  102. package/src/lib/server/runtime/heartbeat-wake.ts +6 -4
  103. package/src/lib/server/runtime/idle-window.ts +2 -2
  104. package/src/lib/server/runtime/network.ts +11 -0
  105. package/src/lib/server/runtime/orchestrator-events.ts +2 -2
  106. package/src/lib/server/runtime/queue/claims.ts +4 -0
  107. package/src/lib/server/runtime/queue/core.ts +2079 -0
  108. package/src/lib/server/runtime/queue/execution.ts +7 -0
  109. package/src/lib/server/runtime/queue/followups.ts +4 -0
  110. package/src/lib/server/runtime/queue/queries.ts +12 -0
  111. package/src/lib/server/runtime/queue/recovery.ts +7 -0
  112. package/src/lib/server/runtime/queue-recovery.test.ts +48 -13
  113. package/src/lib/server/runtime/queue-repository.ts +17 -0
  114. package/src/lib/server/runtime/queue.ts +5 -2061
  115. package/src/lib/server/runtime/run-ledger.ts +6 -5
  116. package/src/lib/server/runtime/run-repository.ts +73 -0
  117. package/src/lib/server/runtime/runtime-lock-repository.ts +8 -0
  118. package/src/lib/server/runtime/runtime-settings.ts +1 -1
  119. package/src/lib/server/runtime/runtime-state.ts +99 -0
  120. package/src/lib/server/runtime/scheduler.ts +4 -2
  121. package/src/lib/server/runtime/session-run-manager/cancellation.ts +157 -0
  122. package/src/lib/server/runtime/session-run-manager/drain.ts +246 -0
  123. package/src/lib/server/runtime/session-run-manager/enqueue.ts +287 -0
  124. package/src/lib/server/runtime/session-run-manager/queries.ts +117 -0
  125. package/src/lib/server/runtime/session-run-manager/recovery.ts +238 -0
  126. package/src/lib/server/runtime/session-run-manager/state.ts +441 -0
  127. package/src/lib/server/runtime/session-run-manager/types.ts +74 -0
  128. package/src/lib/server/runtime/session-run-manager.ts +72 -1377
  129. package/src/lib/server/runtime/watch-job-repository.ts +35 -0
  130. package/src/lib/server/runtime/watch-jobs.ts +3 -1
  131. package/src/lib/server/schedules/schedule-repository.ts +42 -0
  132. package/src/lib/server/sessions/session-repository.ts +85 -0
  133. package/src/lib/server/settings/settings-repository.ts +25 -0
  134. package/src/lib/server/skills/skill-discovery.test.ts +2 -2
  135. package/src/lib/server/skills/skill-discovery.ts +2 -2
  136. package/src/lib/server/skills/skill-repository.ts +14 -0
  137. package/src/lib/server/storage.ts +13 -24
  138. package/src/lib/server/tasks/task-repository.ts +54 -0
  139. package/src/lib/server/usage/usage-repository.ts +30 -0
  140. package/src/lib/server/webhooks/webhook-repository.ts +10 -0
  141. package/src/lib/strip-internal-metadata.test.ts +42 -41
  142. package/src/stores/use-chat-store.test.ts +54 -0
  143. package/src/stores/use-chat-store.ts +21 -5
  144. /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. On permanent delete the
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
- loadTasks,
12
- upsertStoredItems,
13
- loadSchedules,
14
- loadWatchJobs,
15
- loadConnectors,
11
+ deleteDelegationJob,
16
12
  loadDelegationJobs,
17
- loadWebhooks,
18
- loadChatrooms,
19
- deleteStoredItem,
20
- } from '@/lib/server/storage'
21
- import type { StorageCollection } from '@/lib/server/storage'
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
- upsertStoredItems('tasks', taskUpdates)
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
- upsertStoredItems('schedules', schedUpdates)
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
- upsertStoredItems('watch_jobs', wjUpdates)
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
- const connectors = loadConnectors()
88
- const connUpdates: Array<[string, unknown]> = []
89
- for (const c of Object.values(connectors) as unknown as Array<Record<string, unknown>>) {
90
- if (!c || c.agentId !== agentId) continue
91
- c.agentId = null
92
- connUpdates.push([c.id as string, c])
93
- }
94
- if (connUpdates.length) {
95
- upsertStoredItems('connectors', connUpdates)
96
- counts.connectors = connUpdates.length
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
- upsertStoredItems('delegation_jobs', djUpdates)
112
+ saveDelegationJobRecords(djUpdates)
112
113
  counts.delegationJobs = djUpdates.length
113
114
  }
114
115
 
115
116
  // 6. Webhooks — disable
116
117
  const webhooks = loadWebhooks()
117
- const whUpdates: Array<[string, unknown]> = []
118
- for (const w of Object.values(webhooks) as unknown as Array<Record<string, unknown>>) {
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
- whUpdates.push([w.id as string, w])
123
+ webhookUpdates += 1
123
124
  }
124
- if (whUpdates.length) {
125
- upsertStoredItems('webhooks', whUpdates)
126
- counts.webhooks = whUpdates.length
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('tasks', loadTasks(), agentId)
142
- counts.schedules = deleteMatching('schedules', loadSchedules(), agentId)
143
- counts.watchJobs = deleteMatching('watch_jobs', loadWatchJobs(), agentId)
144
- counts.delegationJobs = deleteMatching('delegation_jobs', loadDelegationJobs(), agentId)
145
- counts.webhooks = deleteMatching('webhooks', loadWebhooks(), agentId)
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
- const connectors = loadConnectors()
149
- const connUpdates: Array<[string, unknown]> = []
150
- for (const c of Object.values(connectors) as unknown as Array<Record<string, unknown>>) {
151
- if (!c || c.agentId !== agentId) continue
152
- c.agentId = null
153
- connUpdates.push([c.id as string, c])
154
- }
155
- if (connUpdates.length) {
156
- upsertStoredItems('connectors', connUpdates)
157
- counts.connectors = connUpdates.length
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
- upsertStoredItems('schedules', updates)
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
- deleteStoredItem(table, item.id)
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
- const updates: Array<[string, unknown]> = []
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) updates.push([room.id as string, room])
226
+ if (changed) changedCount += 1
224
227
  }
225
- if (updates.length) {
226
- upsertStoredItems('chatrooms', updates)
228
+ if (changedCount > 0) {
229
+ saveChatrooms(chatrooms)
227
230
  }
228
- return updates.length
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 { loadDelegationJobs, loadDelegationJobItem, loadSession, logActivity, patchDelegationJob, upsertDelegationJob } from '@/lib/server/storage'
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 = loadDelegationJobItem(id)
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
- const synced = syncDelegationMissionFromJob(id) || result
154
+ syncDelegationMissionFromJob(id)
148
155
  notifyDelegationJobsChanged()
149
- return getDelegationJob(id) || synced
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/storage'
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 = loadSessions()[String(candidate.id)] as Session | undefined
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, loadSessions, saveSessions } from '@/lib/server/storage'
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
- saveSessions(sessions)
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: loadSessions()[sid] || null,
406
- messageCount: Array.isArray(loadSessions()[sid]?.messages) ? loadSessions()[sid].messages.length : 0,
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: loadSessions()[sid] || null,
468
- messageCount: Array.isArray(loadSessions()[sid]?.messages) ? loadSessions()[sid].messages.length : 0,
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
  },