@ouro.bot/cli 0.1.0-alpha.42 → 0.1.0-alpha.420

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 (334) hide show
  1. package/README.md +118 -15
  2. package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/agent.json +3 -2
  3. package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/SOUL.md +2 -2
  4. package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-serpent.md +1 -1
  5. package/changelog.json +2627 -9
  6. package/dist/arc/attention-types.js +8 -0
  7. package/dist/arc/cares.js +140 -0
  8. package/dist/arc/episodes.js +117 -0
  9. package/dist/arc/intentions.js +133 -0
  10. package/dist/arc/json-store.js +117 -0
  11. package/dist/arc/obligations.js +237 -0
  12. package/dist/arc/packets.js +193 -0
  13. package/dist/arc/presence.js +185 -0
  14. package/dist/arc/task-lifecycle.js +65 -0
  15. package/dist/heart/active-work.js +832 -0
  16. package/dist/heart/agent-entry.js +58 -3
  17. package/dist/heart/attachments/image-normalize.js +194 -0
  18. package/dist/heart/attachments/materialize.js +97 -0
  19. package/dist/heart/attachments/originals.js +88 -0
  20. package/dist/heart/attachments/render.js +29 -0
  21. package/dist/heart/attachments/sources/adapter.js +2 -0
  22. package/dist/heart/attachments/sources/bluebubbles.js +156 -0
  23. package/dist/heart/attachments/sources/cli-local-file.js +78 -0
  24. package/dist/heart/attachments/sources/index.js +16 -0
  25. package/dist/heart/attachments/store.js +103 -0
  26. package/dist/heart/attachments/types.js +93 -0
  27. package/dist/heart/auth/auth-flow.js +424 -0
  28. package/dist/heart/bridges/manager.js +358 -0
  29. package/dist/heart/bridges/state-machine.js +135 -0
  30. package/dist/heart/bridges/store.js +123 -0
  31. package/dist/heart/bundle-state.js +168 -0
  32. package/dist/heart/commitments.js +111 -0
  33. package/dist/heart/config-registry.js +304 -0
  34. package/dist/heart/config.js +110 -128
  35. package/dist/heart/core.js +801 -217
  36. package/dist/heart/cross-chat-delivery.js +131 -0
  37. package/dist/heart/daemon/agent-config-check.js +419 -0
  38. package/dist/heart/daemon/agent-discovery.js +79 -3
  39. package/dist/heart/daemon/agent-service.js +360 -0
  40. package/dist/heart/daemon/agentic-repair.js +214 -0
  41. package/dist/heart/daemon/bluebubbles-health-diagnostics.js +122 -0
  42. package/dist/heart/daemon/cadence.js +70 -0
  43. package/dist/heart/daemon/cli-defaults.js +605 -0
  44. package/dist/heart/daemon/cli-exec.js +4140 -0
  45. package/dist/heart/daemon/cli-help.js +413 -0
  46. package/dist/heart/daemon/cli-parse.js +1151 -0
  47. package/dist/heart/daemon/cli-render-doctor.js +57 -0
  48. package/dist/heart/daemon/cli-render.js +561 -0
  49. package/dist/heart/daemon/cli-types.js +8 -0
  50. package/dist/heart/daemon/daemon-cli.js +28 -1582
  51. package/dist/heart/daemon/daemon-entry.js +356 -3
  52. package/dist/heart/daemon/daemon-health.js +141 -0
  53. package/dist/heart/daemon/daemon-runtime-sync.js +171 -12
  54. package/dist/heart/daemon/daemon-tombstone.js +236 -0
  55. package/dist/heart/daemon/daemon.js +684 -58
  56. package/dist/heart/daemon/doctor-types.js +8 -0
  57. package/dist/heart/daemon/doctor.js +427 -0
  58. package/dist/heart/daemon/health-monitor.js +79 -1
  59. package/dist/heart/daemon/hooks/agent-config-v2.js +33 -0
  60. package/dist/heart/daemon/hooks/bundle-meta.js +115 -1
  61. package/dist/heart/daemon/http-health-probe.js +80 -0
  62. package/dist/heart/daemon/inner-status.js +89 -0
  63. package/dist/heart/daemon/interactive-repair.js +307 -0
  64. package/dist/heart/daemon/launchd.js +46 -9
  65. package/dist/heart/daemon/log-tailer.js +82 -12
  66. package/dist/heart/daemon/logs-prune.js +105 -0
  67. package/dist/heart/daemon/message-router.js +2 -2
  68. package/dist/heart/daemon/os-cron-deps.js +134 -0
  69. package/dist/heart/daemon/ouro-bot-entry.js +4 -2
  70. package/dist/heart/daemon/ouro-entry.js +3 -1
  71. package/dist/heart/daemon/process-manager.js +214 -0
  72. package/dist/heart/daemon/provider-discovery.js +137 -0
  73. package/dist/heart/daemon/pulse.js +475 -0
  74. package/dist/heart/daemon/readiness-repair.js +250 -0
  75. package/dist/heart/daemon/run-hooks.js +2 -0
  76. package/dist/heart/daemon/runtime-logging.js +67 -16
  77. package/dist/heart/daemon/runtime-metadata.js +73 -0
  78. package/dist/heart/daemon/runtime-mode.js +67 -0
  79. package/dist/heart/daemon/safe-mode.js +161 -0
  80. package/dist/heart/daemon/sense-manager.js +145 -32
  81. package/dist/heart/daemon/session-id-resolver.js +131 -0
  82. package/dist/heart/daemon/skill-management-installer.js +94 -0
  83. package/dist/heart/daemon/socket-client.js +307 -0
  84. package/dist/heart/daemon/stale-bundle-prune.js +96 -0
  85. package/dist/heart/daemon/startup-tui.js +259 -0
  86. package/dist/heart/daemon/task-scheduler.js +3 -25
  87. package/dist/heart/daemon/thoughts.js +510 -0
  88. package/dist/heart/daemon/up-progress.js +218 -0
  89. package/dist/heart/delegation.js +62 -0
  90. package/dist/heart/habits/habit-migration.js +181 -0
  91. package/dist/heart/habits/habit-parser.js +140 -0
  92. package/dist/heart/habits/habit-scheduler.js +371 -0
  93. package/dist/heart/{daemon → hatch}/hatch-flow.js +53 -117
  94. package/dist/heart/{daemon → hatch}/hatch-specialist.js +3 -3
  95. package/dist/heart/{daemon → hatch}/specialist-prompt.js +12 -9
  96. package/dist/heart/{daemon → hatch}/specialist-tools.js +35 -12
  97. package/dist/heart/identity.js +161 -65
  98. package/dist/heart/kept-notes.js +357 -0
  99. package/dist/heart/kicks.js +1 -1
  100. package/dist/heart/machine-identity.js +161 -0
  101. package/dist/heart/mcp/mcp-server.js +653 -0
  102. package/dist/heart/migrate-config.js +100 -0
  103. package/dist/heart/model-capabilities.js +59 -0
  104. package/dist/heart/outlook/outlook-http-hooks.js +64 -0
  105. package/dist/heart/outlook/outlook-http-response.js +7 -0
  106. package/dist/heart/outlook/outlook-http-routes.js +232 -0
  107. package/dist/heart/outlook/outlook-http-static.js +99 -0
  108. package/dist/heart/outlook/outlook-http-transport.js +116 -0
  109. package/dist/heart/outlook/outlook-http.js +99 -0
  110. package/dist/heart/outlook/outlook-read.js +28 -0
  111. package/dist/heart/outlook/outlook-types.js +27 -0
  112. package/dist/heart/outlook/outlook-view.js +195 -0
  113. package/dist/heart/outlook/readers/agent-machine.js +359 -0
  114. package/dist/heart/outlook/readers/continuity-readers.js +332 -0
  115. package/dist/heart/outlook/readers/runtime-readers.js +660 -0
  116. package/dist/heart/outlook/readers/sessions.js +232 -0
  117. package/dist/heart/outlook/readers/shared.js +111 -0
  118. package/dist/heart/platform.js +81 -0
  119. package/dist/heart/progress-story.js +42 -0
  120. package/dist/heart/provider-attempt.js +133 -0
  121. package/dist/heart/provider-binding-resolver.js +239 -0
  122. package/dist/heart/provider-credentials.js +389 -0
  123. package/dist/heart/provider-failover.js +266 -0
  124. package/dist/heart/provider-models.js +81 -0
  125. package/dist/heart/provider-ping.js +237 -0
  126. package/dist/heart/provider-state.js +216 -0
  127. package/dist/heart/provider-visibility.js +186 -0
  128. package/dist/heart/providers/anthropic-token.js +131 -0
  129. package/dist/heart/providers/anthropic.js +193 -55
  130. package/dist/heart/providers/azure.js +103 -12
  131. package/dist/heart/providers/error-classification.js +63 -0
  132. package/dist/heart/providers/github-copilot.js +145 -0
  133. package/dist/heart/providers/minimax-vlm.js +189 -0
  134. package/dist/heart/providers/minimax.js +29 -7
  135. package/dist/heart/providers/openai-codex.js +62 -38
  136. package/dist/heart/runtime-credentials.js +260 -0
  137. package/dist/heart/sense-truth.js +3 -0
  138. package/dist/heart/session-activity.js +190 -0
  139. package/dist/heart/session-events.js +855 -0
  140. package/dist/heart/session-transcript.js +167 -0
  141. package/dist/heart/start-of-turn-packet.js +345 -0
  142. package/dist/heart/streaming.js +36 -27
  143. package/dist/heart/sync.js +332 -0
  144. package/dist/heart/target-resolution.js +127 -0
  145. package/dist/heart/tempo.js +93 -0
  146. package/dist/heart/temporal-view.js +41 -0
  147. package/dist/heart/tool-activity-callbacks.js +36 -0
  148. package/dist/heart/tool-description.js +135 -0
  149. package/dist/heart/tool-friction.js +55 -0
  150. package/dist/heart/tool-loop.js +200 -0
  151. package/dist/heart/turn-context.js +351 -0
  152. package/dist/heart/turn-coordinator.js +28 -0
  153. package/dist/heart/{daemon → versioning}/ouro-bot-global-installer.js +1 -1
  154. package/dist/heart/{daemon → versioning}/ouro-bot-wrapper.js +1 -1
  155. package/dist/heart/versioning/ouro-path-installer.js +301 -0
  156. package/dist/heart/versioning/ouro-version-manager.js +295 -0
  157. package/dist/heart/{daemon → versioning}/staged-restart.js +40 -8
  158. package/dist/heart/{daemon → versioning}/update-checker.js +3 -1
  159. package/dist/heart/{daemon → versioning}/update-hooks.js +63 -59
  160. package/dist/mind/bundle-manifest.js +7 -1
  161. package/dist/mind/context.js +134 -87
  162. package/dist/mind/diary-integrity.js +60 -0
  163. package/dist/mind/{memory.js → diary.js} +74 -93
  164. package/dist/mind/embedding-provider.js +60 -0
  165. package/dist/mind/file-state.js +179 -0
  166. package/dist/mind/first-impressions.js +14 -1
  167. package/dist/mind/friends/channel.js +21 -0
  168. package/dist/mind/friends/group-context.js +144 -0
  169. package/dist/mind/friends/resolver.js +38 -1
  170. package/dist/mind/friends/store-file.js +39 -3
  171. package/dist/mind/friends/trust-explanation.js +74 -0
  172. package/dist/mind/friends/types.js +1 -1
  173. package/dist/mind/journal-index.js +161 -0
  174. package/dist/mind/note-search.js +268 -0
  175. package/dist/mind/obligation-steering.js +221 -0
  176. package/dist/mind/pending.js +66 -7
  177. package/dist/mind/prompt-refresh.js +3 -2
  178. package/dist/mind/prompt.js +948 -168
  179. package/dist/mind/provenance-trust.js +26 -0
  180. package/dist/mind/scrutiny.js +173 -0
  181. package/dist/nerves/cli-logging.js +7 -1
  182. package/dist/nerves/coverage/audit-rules.js +15 -6
  183. package/dist/nerves/coverage/audit.js +28 -2
  184. package/dist/nerves/coverage/cli.js +1 -1
  185. package/dist/nerves/coverage/contract.js +5 -5
  186. package/dist/nerves/coverage/file-completeness.js +83 -5
  187. package/dist/nerves/coverage/run-artifacts.js +1 -1
  188. package/dist/nerves/event-buffer.js +111 -0
  189. package/dist/nerves/index.js +224 -4
  190. package/dist/nerves/observation.js +20 -0
  191. package/dist/nerves/redact.js +79 -0
  192. package/dist/nerves/runtime.js +5 -1
  193. package/dist/outlook-ui/assets/index-BAcU08c-.css +1 -0
  194. package/dist/outlook-ui/assets/index-D7l3l4vY.js +61 -0
  195. package/dist/outlook-ui/index.html +15 -0
  196. package/dist/repertoire/ado-client.js +15 -56
  197. package/dist/repertoire/ado-semantic.js +11 -10
  198. package/dist/repertoire/api-client.js +97 -0
  199. package/dist/repertoire/bitwarden-store.js +702 -0
  200. package/dist/repertoire/bundle-templates.js +72 -0
  201. package/dist/repertoire/bw-installer.js +79 -0
  202. package/dist/repertoire/coding/codex-jsonl.js +64 -0
  203. package/dist/repertoire/coding/context-pack.js +330 -0
  204. package/dist/repertoire/coding/feedback.js +197 -30
  205. package/dist/repertoire/coding/manager.js +158 -9
  206. package/dist/repertoire/coding/spawner.js +55 -9
  207. package/dist/repertoire/coding/tools.js +170 -7
  208. package/dist/repertoire/commerce-errors.js +109 -0
  209. package/dist/repertoire/commerce-self-test.js +156 -0
  210. package/dist/repertoire/credential-access.js +111 -0
  211. package/dist/repertoire/duffel-client.js +185 -0
  212. package/dist/repertoire/github-client.js +14 -55
  213. package/dist/repertoire/graph-client.js +11 -52
  214. package/dist/repertoire/guardrails.js +371 -0
  215. package/dist/repertoire/mcp-client.js +255 -0
  216. package/dist/repertoire/mcp-manager.js +305 -0
  217. package/dist/repertoire/mcp-tools.js +63 -0
  218. package/dist/repertoire/shell-sessions.js +133 -0
  219. package/dist/repertoire/skills.js +15 -24
  220. package/dist/repertoire/stripe-client.js +131 -0
  221. package/dist/repertoire/tasks/board.js +43 -5
  222. package/dist/repertoire/tasks/fix.js +182 -0
  223. package/dist/repertoire/tasks/index.js +26 -1
  224. package/dist/repertoire/tasks/lifecycle.js +2 -2
  225. package/dist/repertoire/tasks/parser.js +3 -2
  226. package/dist/repertoire/tasks/scanner.js +194 -37
  227. package/dist/repertoire/tasks/transitions.js +16 -78
  228. package/dist/repertoire/tool-results.js +29 -0
  229. package/dist/repertoire/tools-attachments.js +317 -0
  230. package/dist/repertoire/tools-base.js +42 -687
  231. package/dist/repertoire/tools-bluebubbles.js +1 -0
  232. package/dist/repertoire/tools-bridge.js +141 -0
  233. package/dist/repertoire/tools-bundle.js +984 -0
  234. package/dist/repertoire/tools-config.js +185 -0
  235. package/dist/repertoire/tools-continuity.js +248 -0
  236. package/dist/repertoire/tools-credential.js +361 -0
  237. package/dist/repertoire/tools-files.js +342 -0
  238. package/dist/repertoire/tools-flight.js +224 -0
  239. package/dist/repertoire/tools-flow.js +105 -0
  240. package/dist/repertoire/tools-github.js +1 -7
  241. package/dist/repertoire/tools-notes.js +376 -0
  242. package/dist/repertoire/tools-session.js +739 -0
  243. package/dist/repertoire/tools-shell.js +120 -0
  244. package/dist/repertoire/tools-stripe.js +180 -0
  245. package/dist/repertoire/tools-surface.js +243 -0
  246. package/dist/repertoire/tools-teams.js +9 -39
  247. package/dist/repertoire/tools-travel.js +125 -0
  248. package/dist/repertoire/tools-user-profile.js +144 -0
  249. package/dist/repertoire/tools-vault.js +40 -0
  250. package/dist/repertoire/tools.js +144 -113
  251. package/dist/repertoire/travel-api-client.js +360 -0
  252. package/dist/repertoire/user-profile.js +131 -0
  253. package/dist/repertoire/vault-setup.js +246 -0
  254. package/dist/repertoire/vault-unlock.js +421 -0
  255. package/dist/scripts/claude-code-hook.js +41 -0
  256. package/dist/scripts/claude-code-stop-hook.js +47 -0
  257. package/dist/senses/attention-queue.js +116 -0
  258. package/dist/senses/bluebubbles/attachment-cache.js +53 -0
  259. package/dist/senses/bluebubbles/attachment-download.js +137 -0
  260. package/dist/senses/{bluebubbles-client.js → bluebubbles/client.js} +260 -9
  261. package/dist/senses/bluebubbles/entry.js +73 -0
  262. package/dist/senses/bluebubbles/inbound-log.js +113 -0
  263. package/dist/senses/bluebubbles/index.js +1620 -0
  264. package/dist/senses/{bluebubbles-media.js → bluebubbles/media.js} +121 -70
  265. package/dist/senses/{bluebubbles-model.js → bluebubbles/model.js} +33 -12
  266. package/dist/senses/{bluebubbles-mutation-log.js → bluebubbles/mutation-log.js} +45 -3
  267. package/dist/senses/bluebubbles/replay.js +129 -0
  268. package/dist/senses/bluebubbles/runtime-state.js +109 -0
  269. package/dist/senses/{bluebubbles-session-cleanup.js → bluebubbles/session-cleanup.js} +1 -1
  270. package/dist/senses/cli/bracketed-paste.js +82 -0
  271. package/dist/senses/cli/image-paste.js +287 -0
  272. package/dist/senses/cli/image-ref-navigation.js +75 -0
  273. package/dist/senses/cli/ink-app.js +156 -0
  274. package/dist/senses/cli/inline-diff.js +64 -0
  275. package/dist/senses/cli/input-keys.js +174 -0
  276. package/dist/senses/cli/kill-ring.js +86 -0
  277. package/dist/senses/cli/message-list.js +51 -0
  278. package/dist/senses/cli/ouro-tui.js +605 -0
  279. package/dist/senses/cli/spinner-imperative.js +135 -0
  280. package/dist/senses/cli/spinner.js +101 -0
  281. package/dist/senses/cli/status-line.js +60 -0
  282. package/dist/senses/cli/streaming-markdown.js +526 -0
  283. package/dist/senses/cli/tool-display.js +83 -0
  284. package/dist/senses/cli/tool-render.js +85 -0
  285. package/dist/senses/cli/tui-store.js +240 -0
  286. package/dist/senses/cli/virtual-list.js +35 -0
  287. package/dist/senses/cli-entry.js +60 -8
  288. package/dist/senses/cli-layout.js +187 -0
  289. package/dist/senses/cli.js +526 -211
  290. package/dist/senses/commands.js +66 -3
  291. package/dist/senses/continuity.js +94 -0
  292. package/dist/senses/habit-turn-message.js +108 -0
  293. package/dist/senses/inner-dialog-worker.js +112 -19
  294. package/dist/senses/inner-dialog.js +596 -94
  295. package/dist/senses/pipeline.js +539 -61
  296. package/dist/senses/proactive-content-guard.js +51 -0
  297. package/dist/senses/shared-turn.js +205 -0
  298. package/dist/senses/surface-tool.js +68 -0
  299. package/dist/senses/teams-entry.js +60 -8
  300. package/dist/senses/teams.js +569 -237
  301. package/dist/senses/trust-gate.js +5 -5
  302. package/package.json +29 -7
  303. package/skills/agent-commerce.md +106 -0
  304. package/skills/browser-navigation.md +117 -0
  305. package/skills/commerce-setup-guide.md +116 -0
  306. package/skills/commerce-setup.md +84 -0
  307. package/skills/configure-dev-tools.md +101 -0
  308. package/skills/travel-planning.md +138 -0
  309. package/dist/heart/daemon/ouro-path-installer.js +0 -178
  310. package/dist/heart/daemon/subagent-installer.js +0 -134
  311. package/dist/mind/associative-recall.js +0 -209
  312. package/dist/senses/bluebubbles-entry.js +0 -11
  313. package/dist/senses/bluebubbles.js +0 -832
  314. package/dist/senses/debug-activity.js +0 -127
  315. package/subagents/README.md +0 -60
  316. package/subagents/work-doer.md +0 -235
  317. package/subagents/work-merger.md +0 -618
  318. package/subagents/work-planner.md +0 -382
  319. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/basilisk.md +0 -0
  320. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jafar.md +0 -0
  321. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jormungandr.md +0 -0
  322. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/kaa.md +0 -0
  323. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/medusa.md +0 -0
  324. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/monty.md +0 -0
  325. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/nagini.md +0 -0
  326. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/ouroboros.md +0 -0
  327. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/python.md +0 -0
  328. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/quetzalcoatl.md +0 -0
  329. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/sir-hiss.md +0 -0
  330. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-snake.md +0 -0
  331. /package/dist/heart/{daemon → hatch}/hatch-animation.js +0 -0
  332. /package/dist/heart/{daemon → hatch}/specialist-orchestrator.js +0 -0
  333. /package/dist/heart/{daemon → versioning}/ouro-uti.js +0 -0
  334. /package/dist/heart/{daemon → versioning}/wrapper-publish-guard.js +0 -0
@@ -0,0 +1,131 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.deliverCrossChatMessage = deliverCrossChatMessage;
4
+ const types_1 = require("../mind/friends/types");
5
+ const runtime_1 = require("../nerves/runtime");
6
+ function buildPendingEnvelope(request, agentName, now) {
7
+ return {
8
+ from: agentName,
9
+ friendId: request.friendId,
10
+ channel: request.channel,
11
+ key: request.key,
12
+ content: request.content,
13
+ timestamp: now,
14
+ };
15
+ }
16
+ function queueForLater(request, deps, detail) {
17
+ deps.queuePending(buildPendingEnvelope(request, deps.agentName, (deps.now ?? Date.now)()));
18
+ return {
19
+ status: "queued_for_later",
20
+ detail,
21
+ };
22
+ }
23
+ function isExplicitlyAuthorized(request) {
24
+ return request.intent === "explicit_cross_chat"
25
+ && Boolean(request.authorizingSession)
26
+ && (0, types_1.isTrustedLevel)(request.authorizingSession?.trustLevel);
27
+ }
28
+ async function deliverCrossChatMessage(request, deps) {
29
+ (0, runtime_1.emitNervesEvent)({
30
+ component: "engine",
31
+ event: "engine.cross_chat_delivery_start",
32
+ message: "resolving cross-chat delivery",
33
+ meta: {
34
+ friendId: request.friendId,
35
+ channel: request.channel,
36
+ key: request.key,
37
+ intent: request.intent,
38
+ authorizingTrustLevel: request.authorizingSession?.trustLevel ?? null,
39
+ },
40
+ });
41
+ if (!isExplicitlyAuthorized(request) && request.intent !== "generic_outreach") {
42
+ const result = {
43
+ status: "blocked",
44
+ detail: "cross-chat delivery requires a trusted asking session or generic outreach intent",
45
+ };
46
+ (0, runtime_1.emitNervesEvent)({
47
+ level: "warn",
48
+ component: "engine",
49
+ event: "engine.cross_chat_delivery_end",
50
+ message: "blocked cross-chat delivery",
51
+ meta: {
52
+ friendId: request.friendId,
53
+ channel: request.channel,
54
+ key: request.key,
55
+ status: result.status,
56
+ },
57
+ });
58
+ return result;
59
+ }
60
+ const deliverer = deps.deliverers?.[request.channel];
61
+ if (!deliverer) {
62
+ const result = queueForLater(request, deps, "live delivery unavailable right now; queued for the next active turn");
63
+ (0, runtime_1.emitNervesEvent)({
64
+ component: "engine",
65
+ event: "engine.cross_chat_delivery_end",
66
+ message: "queued explicit cross-chat delivery because no live deliverer was available",
67
+ meta: {
68
+ friendId: request.friendId,
69
+ channel: request.channel,
70
+ key: request.key,
71
+ status: result.status,
72
+ },
73
+ });
74
+ return result;
75
+ }
76
+ try {
77
+ const direct = await deliverer(request);
78
+ if (direct.status === "delivered_now" || direct.status === "blocked" || direct.status === "failed") {
79
+ const result = {
80
+ status: direct.status,
81
+ detail: direct.detail,
82
+ };
83
+ (0, runtime_1.emitNervesEvent)({
84
+ level: result.status === "failed" ? "error" : result.status === "blocked" ? "warn" : "info",
85
+ component: "engine",
86
+ event: "engine.cross_chat_delivery_end",
87
+ message: "completed direct cross-chat delivery resolution",
88
+ meta: {
89
+ friendId: request.friendId,
90
+ channel: request.channel,
91
+ key: request.key,
92
+ status: result.status,
93
+ },
94
+ });
95
+ return result;
96
+ }
97
+ const result = queueForLater(request, deps, direct.detail.trim() || "live delivery unavailable right now; queued for the next active turn");
98
+ (0, runtime_1.emitNervesEvent)({
99
+ component: "engine",
100
+ event: "engine.cross_chat_delivery_end",
101
+ message: "queued explicit cross-chat delivery after adapter reported unavailability",
102
+ meta: {
103
+ friendId: request.friendId,
104
+ channel: request.channel,
105
+ key: request.key,
106
+ status: result.status,
107
+ },
108
+ });
109
+ return result;
110
+ }
111
+ catch (error) {
112
+ const result = {
113
+ status: "failed",
114
+ detail: error instanceof Error ? error.message : String(error),
115
+ };
116
+ (0, runtime_1.emitNervesEvent)({
117
+ level: "error",
118
+ component: "engine",
119
+ event: "engine.cross_chat_delivery_end",
120
+ message: "cross-chat delivery threw unexpectedly",
121
+ meta: {
122
+ friendId: request.friendId,
123
+ channel: request.channel,
124
+ key: request.key,
125
+ status: result.status,
126
+ reason: result.detail,
127
+ },
128
+ });
129
+ return result;
130
+ }
131
+ }
@@ -0,0 +1,419 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
15
+ }) : function(o, v) {
16
+ o["default"] = v;
17
+ });
18
+ var __importStar = (this && this.__importStar) || (function () {
19
+ var ownKeys = function(o) {
20
+ ownKeys = Object.getOwnPropertyNames || function (o) {
21
+ var ar = [];
22
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
23
+ return ar;
24
+ };
25
+ return ownKeys(o);
26
+ };
27
+ return function (mod) {
28
+ if (mod && mod.__esModule) return mod;
29
+ var result = {};
30
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
31
+ __setModuleDefault(result, mod);
32
+ return result;
33
+ };
34
+ })();
35
+ Object.defineProperty(exports, "__esModule", { value: true });
36
+ exports.vaultUnlockOrRecoverFix = vaultUnlockOrRecoverFix;
37
+ exports.checkAgentConfig = checkAgentConfig;
38
+ exports.checkAgentConfigWithProviderHealth = checkAgentConfigWithProviderHealth;
39
+ const fs = __importStar(require("fs"));
40
+ const path = __importStar(require("path"));
41
+ const identity_1 = require("../identity");
42
+ const runtime_1 = require("../../nerves/runtime");
43
+ const provider_models_1 = require("../provider-models");
44
+ const machine_identity_1 = require("../machine-identity");
45
+ const provider_state_1 = require("../provider-state");
46
+ const provider_credentials_1 = require("../provider-credentials");
47
+ const vault_unlock_1 = require("../../repertoire/vault-unlock");
48
+ const readiness_repair_1 = require("./readiness-repair");
49
+ function isAgentProvider(value) {
50
+ return Object.prototype.hasOwnProperty.call(identity_1.PROVIDER_CREDENTIALS, value);
51
+ }
52
+ function agentRootFor(agentName, bundlesRoot) {
53
+ return path.join(bundlesRoot, `${agentName}.ouro`);
54
+ }
55
+ function configPathFor(agentName, bundlesRoot) {
56
+ return path.join(agentRootFor(agentName, bundlesRoot), "agent.json");
57
+ }
58
+ function resolveFacingProvider(parsed, facing, agentName, agentJsonPath) {
59
+ const raw = parsed[facing];
60
+ if (!raw || typeof raw !== "object" || Array.isArray(raw)) {
61
+ return {
62
+ ok: false,
63
+ result: {
64
+ ok: false,
65
+ error: `agent.json for '${agentName}' is missing ${facing}.provider`,
66
+ fix: `Add ${facing}: { provider, model } to ${agentJsonPath}. Valid providers: ${Object.keys(identity_1.PROVIDER_CREDENTIALS).join(", ")}`,
67
+ },
68
+ };
69
+ }
70
+ const provider = raw.provider;
71
+ if (typeof provider !== "string" || provider.length === 0) {
72
+ return {
73
+ ok: false,
74
+ result: {
75
+ ok: false,
76
+ error: `agent.json for '${agentName}' is missing ${facing}.provider`,
77
+ fix: `Set ${facing}.provider in ${agentJsonPath}. Valid providers: ${Object.keys(identity_1.PROVIDER_CREDENTIALS).join(", ")}`,
78
+ },
79
+ };
80
+ }
81
+ if (!isAgentProvider(provider)) {
82
+ return {
83
+ ok: false,
84
+ result: {
85
+ ok: false,
86
+ error: `Unknown provider '${provider}' in ${facing}.provider for '${agentName}'`,
87
+ fix: `Set ${facing}.provider to one of: ${Object.keys(identity_1.PROVIDER_CREDENTIALS).join(", ")}`,
88
+ },
89
+ };
90
+ }
91
+ return { ok: true, selected: { facing, provider } };
92
+ }
93
+ function readAgentConfigForProviderState(agentName, bundlesRoot) {
94
+ const agentJsonPath = configPathFor(agentName, bundlesRoot);
95
+ let raw;
96
+ try {
97
+ raw = fs.readFileSync(agentJsonPath, "utf-8");
98
+ }
99
+ catch {
100
+ return {
101
+ ok: false,
102
+ result: {
103
+ ok: false,
104
+ error: `agent.json not found at ${agentJsonPath}`,
105
+ fix: `Run 'ouro hatch ${agentName}' to create the agent bundle, or verify that ${bundlesRoot}/${agentName}.ouro/ exists.`,
106
+ },
107
+ };
108
+ }
109
+ let parsed;
110
+ try {
111
+ parsed = JSON.parse(raw);
112
+ }
113
+ catch {
114
+ return {
115
+ ok: false,
116
+ result: {
117
+ ok: false,
118
+ error: `agent.json at ${agentJsonPath} contains invalid JSON`,
119
+ fix: `Open ${agentJsonPath} and fix the JSON syntax.`,
120
+ },
121
+ };
122
+ }
123
+ if (parsed.enabled === false) {
124
+ return { ok: true, disabled: true, agentJsonPath, parsed };
125
+ }
126
+ return { ok: true, disabled: false, agentJsonPath, parsed };
127
+ }
128
+ function readFacingForBootstrap(parsed, facing, agentName, agentJsonPath) {
129
+ const providerResult = resolveFacingProvider(parsed, facing, agentName, agentJsonPath);
130
+ if (!providerResult.ok) {
131
+ return {
132
+ ok: false,
133
+ result: {
134
+ ok: false,
135
+ error: providerResult.result.error,
136
+ fix: `Run 'ouro use --agent ${agentName} --lane ${facing === "humanFacing" ? "outward" : "inner"} --provider <provider> --model <model>' to configure this machine's provider binding.`,
137
+ },
138
+ };
139
+ }
140
+ const raw = parsed[facing];
141
+ const model = typeof raw.model === "string" && raw.model.trim().length > 0
142
+ ? raw.model.trim()
143
+ : (0, provider_models_1.getDefaultModelForProvider)(providerResult.selected.provider);
144
+ return { ok: true, provider: providerResult.selected.provider, model };
145
+ }
146
+ function bootstrapMissingProviderState(input) {
147
+ const outward = readFacingForBootstrap(input.parsed, "humanFacing", input.agentName, input.agentJsonPath);
148
+ if (!outward.ok)
149
+ return { ok: false, error: outward.result.error, fix: outward.result.fix };
150
+ const inner = readFacingForBootstrap(input.parsed, "agentFacing", input.agentName, input.agentJsonPath);
151
+ if (!inner.ok)
152
+ return { ok: false, error: inner.result.error, fix: inner.result.fix };
153
+ const now = new Date();
154
+ const homeDir = (0, provider_credentials_1.providerCredentialMachineHomeDir)(input.homeDir);
155
+ const machine = (0, machine_identity_1.loadOrCreateMachineIdentity)({ homeDir, now: () => now });
156
+ const state = (0, provider_state_1.bootstrapProviderStateFromAgentConfig)({
157
+ machineId: machine.machineId,
158
+ now,
159
+ agentConfig: {
160
+ humanFacing: { provider: outward.provider, model: outward.model },
161
+ agentFacing: { provider: inner.provider, model: inner.model },
162
+ },
163
+ });
164
+ const agentRoot = agentRootFor(input.agentName, input.bundlesRoot);
165
+ (0, provider_state_1.writeProviderState)(agentRoot, state);
166
+ (0, runtime_1.emitNervesEvent)({
167
+ component: "daemon",
168
+ event: "daemon.provider_state_bootstrapped",
169
+ message: "bootstrapped local provider state from agent config",
170
+ meta: { agent: input.agentName, agentRoot },
171
+ });
172
+ return { ok: true, agentRoot, state };
173
+ }
174
+ function readOrBootstrapProviderStateForCheck(agentName, bundlesRoot, deps = {}) {
175
+ const configResult = readAgentConfigForProviderState(agentName, bundlesRoot);
176
+ if (!configResult.ok)
177
+ return { ok: false, result: configResult.result };
178
+ if (configResult.disabled)
179
+ return { ok: true, disabled: true };
180
+ const agentRoot = agentRootFor(agentName, bundlesRoot);
181
+ const stateResult = (0, provider_state_1.readProviderState)(agentRoot);
182
+ if (stateResult.ok) {
183
+ return { ok: true, disabled: false, agentRoot, state: stateResult.state };
184
+ }
185
+ if (stateResult.reason === "invalid") {
186
+ return {
187
+ ok: false,
188
+ result: {
189
+ ok: false,
190
+ error: `provider state for ${agentName} is invalid at ${stateResult.statePath}: ${stateResult.error}`,
191
+ fix: `Run 'ouro use --agent ${agentName} --lane outward --provider <provider> --model <model> --force' to rewrite this machine's provider binding.`,
192
+ },
193
+ };
194
+ }
195
+ const bootstrap = bootstrapMissingProviderState({
196
+ agentName,
197
+ bundlesRoot,
198
+ parsed: configResult.parsed,
199
+ agentJsonPath: configResult.agentJsonPath,
200
+ homeDir: deps.homeDir,
201
+ });
202
+ if (!bootstrap.ok)
203
+ return { ok: false, result: bootstrap };
204
+ return { ok: true, disabled: false, agentRoot: bootstrap.agentRoot, state: bootstrap.state };
205
+ }
206
+ function providerCredentialConfig(record) {
207
+ return {
208
+ ...record.credentials,
209
+ ...record.config,
210
+ };
211
+ }
212
+ function pingAttemptCount(result) {
213
+ if (Array.isArray(result.attempts))
214
+ return result.attempts.length;
215
+ return undefined;
216
+ }
217
+ function writeLaneReadiness(input) {
218
+ const binding = input.state.lanes[input.lane];
219
+ const checkedAt = new Date().toISOString();
220
+ input.state.updatedAt = checkedAt;
221
+ input.state.readiness[input.lane] = {
222
+ status: input.status,
223
+ provider: binding.provider,
224
+ model: binding.model,
225
+ checkedAt,
226
+ credentialRevision: input.credentialRevision,
227
+ ...(input.error ? { error: input.error } : {}),
228
+ ...(input.attempts !== undefined ? { attempts: input.attempts } : {}),
229
+ };
230
+ (0, provider_state_1.writeProviderState)(input.agentRoot, input.state);
231
+ }
232
+ function missingCredentialResult(agentName, lane, provider, model, credentialPath) {
233
+ return {
234
+ ok: false,
235
+ error: `${lane} provider ${provider} model ${model} has no credentials in ${agentName}'s vault at ${credentialPath}`,
236
+ fix: `Run 'ouro auth --agent ${agentName} --provider ${provider}' to authenticate.`,
237
+ issue: (0, readiness_repair_1.providerCredentialMissingIssue)({
238
+ agentName,
239
+ lane,
240
+ provider,
241
+ model,
242
+ credentialPath,
243
+ }),
244
+ };
245
+ }
246
+ function isTransientVaultError(error) {
247
+ const normalized = error.toLowerCase();
248
+ return (normalized.includes("timed out") ||
249
+ normalized.includes("econnrefused") ||
250
+ normalized.includes("socket hang up") ||
251
+ normalized.includes("etimedout"));
252
+ }
253
+ function invalidPoolResult(agentName, lane, provider, model, pool) {
254
+ if (pool.reason === "unavailable" && isTransientVaultError(pool.error)) {
255
+ return {
256
+ ok: false,
257
+ error: `${lane} provider ${provider} model ${model} cannot read provider credentials from ${agentName}'s vault: ${pool.error}`,
258
+ fix: `Vault read timed out -- this usually resolves on retry. Run 'ouro up' again.`,
259
+ };
260
+ }
261
+ if (pool.reason === "unavailable" && isVaultLockedError(pool.error)) {
262
+ return {
263
+ ok: false,
264
+ error: `${lane} provider ${provider} model ${model} cannot read provider credentials because ${agentName}'s credential vault is locked on this machine.`,
265
+ fix: vaultUnlockOrRecoverFix(agentName),
266
+ issue: (0, readiness_repair_1.vaultLockedIssue)(agentName),
267
+ };
268
+ }
269
+ if (pool.reason === "unavailable" && (0, vault_unlock_1.isCredentialVaultNotConfiguredError)(pool.error)) {
270
+ return {
271
+ ok: false,
272
+ error: `${lane} provider ${provider} model ${model} cannot read provider credentials because ${agentName}'s credential vault is not configured in agent.json.`,
273
+ fix: (0, vault_unlock_1.vaultCreateRecoverFix)(agentName, `Then run 'ouro auth --agent ${agentName} --provider ${provider}' and rerun 'ouro up'.`),
274
+ issue: (0, readiness_repair_1.vaultUnconfiguredIssue)(agentName),
275
+ };
276
+ }
277
+ if (pool.reason === "invalid") {
278
+ return {
279
+ ok: false,
280
+ error: `${lane} provider ${provider} model ${model} cannot read provider credentials from ${agentName}'s vault at ${pool.poolPath}: ${pool.error}`,
281
+ fix: `Run 'ouro auth --agent ${agentName} --provider ${provider}' to rewrite this provider credential, then run 'ouro up' again.`,
282
+ };
283
+ }
284
+ return {
285
+ ok: false,
286
+ error: `${lane} provider ${provider} model ${model} cannot read provider credentials from ${agentName}'s vault at ${pool.poolPath}: ${pool.error}`,
287
+ fix: vaultUnlockOrRecoverFix(agentName, `Then run 'ouro up' again. If the credential is missing or stale after unlock or recovery, run 'ouro auth --agent ${agentName} --provider ${provider}'.`),
288
+ };
289
+ }
290
+ function isVaultLockedError(error) {
291
+ const normalized = error.toLowerCase();
292
+ return /(?:ouro )?credential vault is locked|vault(?: is)? locked/.test(normalized);
293
+ }
294
+ function vaultUnlockOrRecoverFix(agentName, nextStep = "Then run 'ouro up' again.") {
295
+ return (0, vault_unlock_1.vaultUnlockReplaceRecoverFix)(agentName, nextStep);
296
+ }
297
+ function failedPingResult(agentName, lane, provider, model, result) {
298
+ return {
299
+ ok: false,
300
+ error: `${lane} provider ${provider} model ${model} failed live check: ${result.message}`,
301
+ fix: `Run 'ouro auth --agent ${agentName} --provider ${provider}' to refresh credentials.`,
302
+ issue: (0, readiness_repair_1.providerLiveCheckFailedIssue)({
303
+ agentName,
304
+ lane,
305
+ provider,
306
+ model,
307
+ message: result.message,
308
+ }),
309
+ };
310
+ }
311
+ function credentialRecordForLane(pool, provider) {
312
+ return pool.providers[provider];
313
+ }
314
+ /**
315
+ * Structural validation only. Live provider credential validation belongs to
316
+ * checkAgentConfigWithProviderHealth(), which reads the agent vault and pings.
317
+ */
318
+ function checkAgentConfig(agentName, bundlesRoot) {
319
+ const configResult = readAgentConfigForProviderState(agentName, bundlesRoot);
320
+ if (!configResult.ok)
321
+ return configResult.result;
322
+ if (configResult.disabled)
323
+ return { ok: true };
324
+ const outward = readFacingForBootstrap(configResult.parsed, "humanFacing", agentName, configResult.agentJsonPath);
325
+ if (!outward.ok)
326
+ return outward.result;
327
+ const inner = readFacingForBootstrap(configResult.parsed, "agentFacing", agentName, configResult.agentJsonPath);
328
+ if (!inner.ok)
329
+ return inner.result;
330
+ (0, runtime_1.emitNervesEvent)({
331
+ component: "daemon",
332
+ event: "daemon.agent_config_valid",
333
+ message: "agent config validation passed",
334
+ meta: {
335
+ agent: agentName,
336
+ providers: [...new Set([outward.provider, inner.provider])],
337
+ liveProviderCheck: false,
338
+ },
339
+ });
340
+ return { ok: true };
341
+ }
342
+ async function checkAgentConfigWithProviderHealth(agentName, bundlesRoot, deps = {}) {
343
+ const stateResult = readOrBootstrapProviderStateForCheck(agentName, bundlesRoot, deps);
344
+ if (!stateResult.ok)
345
+ return stateResult.result;
346
+ if (stateResult.disabled)
347
+ return { ok: true };
348
+ const ping = deps.pingProvider ?? (await Promise.resolve().then(() => __importStar(require("../provider-ping")))).pingProvider;
349
+ const poolResult = await (0, provider_credentials_1.refreshProviderCredentialPool)(agentName, deps.onProgress ? { onProgress: deps.onProgress } : undefined);
350
+ const pingGroups = new Map();
351
+ const lanes = ["outward", "inner"];
352
+ for (const lane of lanes) {
353
+ const binding = stateResult.state.lanes[lane];
354
+ if (!poolResult.ok) {
355
+ if (poolResult.reason === "missing") {
356
+ return missingCredentialResult(agentName, lane, binding.provider, binding.model, poolResult.poolPath);
357
+ }
358
+ return invalidPoolResult(agentName, lane, binding.provider, binding.model, {
359
+ ...poolResult,
360
+ reason: poolResult.reason,
361
+ });
362
+ }
363
+ const record = credentialRecordForLane(poolResult.pool, binding.provider);
364
+ if (!record) {
365
+ return missingCredentialResult(agentName, lane, binding.provider, binding.model, poolResult.poolPath);
366
+ }
367
+ const key = `${binding.provider}\0${binding.model}\0${record.revision}`;
368
+ const group = pingGroups.get(key);
369
+ if (group) {
370
+ group.lanes.push(lane);
371
+ }
372
+ else {
373
+ pingGroups.set(key, {
374
+ provider: binding.provider,
375
+ model: binding.model,
376
+ record,
377
+ lanes: [lane],
378
+ });
379
+ }
380
+ }
381
+ for (const group of pingGroups.values()) {
382
+ const result = await ping(group.provider, providerCredentialConfig(group.record), { model: group.model });
383
+ if (!result.ok) {
384
+ for (const lane of group.lanes) {
385
+ writeLaneReadiness({
386
+ agentRoot: stateResult.agentRoot,
387
+ state: stateResult.state,
388
+ lane,
389
+ status: "failed",
390
+ credentialRevision: group.record.revision,
391
+ error: result.message,
392
+ attempts: pingAttemptCount(result),
393
+ });
394
+ }
395
+ return failedPingResult(agentName, group.lanes[0], group.provider, group.model, result);
396
+ }
397
+ for (const lane of group.lanes) {
398
+ writeLaneReadiness({
399
+ agentRoot: stateResult.agentRoot,
400
+ state: stateResult.state,
401
+ lane,
402
+ status: "ready",
403
+ credentialRevision: group.record.revision,
404
+ attempts: pingAttemptCount(result),
405
+ });
406
+ }
407
+ }
408
+ (0, runtime_1.emitNervesEvent)({
409
+ component: "daemon",
410
+ event: "daemon.agent_config_valid",
411
+ message: "agent config validation passed",
412
+ meta: {
413
+ agent: agentName,
414
+ providers: [...new Set([...pingGroups.values()].map((group) => group.provider))],
415
+ liveProviderCheck: true,
416
+ },
417
+ });
418
+ return { ok: true };
419
+ }
@@ -33,12 +33,25 @@ var __importStar = (this && this.__importStar) || (function () {
33
33
  };
34
34
  })();
35
35
  Object.defineProperty(exports, "__esModule", { value: true });
36
+ exports.listAllBundleAgents = listAllBundleAgents;
36
37
  exports.listEnabledBundleAgents = listEnabledBundleAgents;
38
+ exports.listBundleSyncRows = listBundleSyncRows;
37
39
  const fs = __importStar(require("fs"));
38
40
  const path = __importStar(require("path"));
41
+ const child_process_1 = require("child_process");
39
42
  const identity_1 = require("../identity");
40
43
  const runtime_1 = require("../../nerves/runtime");
41
- function listEnabledBundleAgents(options = {}) {
44
+ /**
45
+ * Walk the bundles root and return one row per `<name>.ouro` directory whose
46
+ * `agent.json` is readable and parseable. Includes both enabled and disabled
47
+ * agents — the caller decides what to do with the `enabled` flag.
48
+ *
49
+ * Bundles whose `agent.json` is missing, malformed, or unreadable are skipped
50
+ * silently (they aren't real agents from the harness's perspective).
51
+ *
52
+ * Sorted alphabetically by name for stable display.
53
+ */
54
+ function listAllBundleAgents(options = {}) {
42
55
  const bundlesRoot = options.bundlesRoot ?? (0, identity_1.getAgentBundlesRoot)();
43
56
  const readdirSync = options.readdirSync ?? fs.readdirSync;
44
57
  const readFileSync = options.readFileSync ?? fs.readFileSync;
@@ -73,9 +86,72 @@ function listEnabledBundleAgents(options = {}) {
73
86
  catch {
74
87
  continue;
75
88
  }
89
+ discovered.push({ name: agentName, enabled });
90
+ }
91
+ return discovered.sort((left, right) => left.name.localeCompare(right.name));
92
+ }
93
+ function listEnabledBundleAgents(options = {}) {
94
+ return listAllBundleAgents(options).filter((row) => row.enabled).map((row) => row.name);
95
+ }
96
+ /**
97
+ * Read the per-agent sync block from each enabled bundle's agent.json.
98
+ * Used by the daemon (and stopped-state status renderer) to build per-agent
99
+ * sync rows without depending on argv-derived global identity. Bundles that
100
+ * cannot be read are skipped silently.
101
+ *
102
+ * For rows with sync enabled, also checks whether the bundle is a git repo
103
+ * (via .git directory presence) and resolves the remote URL via
104
+ * `git remote get-url <remote>`. On any error the URL is left undefined and
105
+ * the status renderer falls back to "local only" or "not a git repo".
106
+ */
107
+ function listBundleSyncRows(options = {}) {
108
+ const bundlesRoot = options.bundlesRoot ?? (0, identity_1.getAgentBundlesRoot)();
109
+ const readFileSync = options.readFileSync ?? fs.readFileSync;
110
+ const execFileSync = options.execFileSync ?? child_process_1.execFileSync;
111
+ const existsSync = options.existsSync ?? fs.existsSync;
112
+ const agents = listEnabledBundleAgents(options);
113
+ const rows = [];
114
+ for (const agent of agents) {
115
+ const bundleRoot = path.join(bundlesRoot, `${agent}.ouro`);
116
+ const configPath = path.join(bundleRoot, "agent.json");
117
+ let enabled = false;
118
+ let remote = "origin";
119
+ try {
120
+ const raw = readFileSync(configPath, "utf-8");
121
+ const parsed = JSON.parse(raw);
122
+ if (parsed.sync && typeof parsed.sync === "object") {
123
+ if (typeof parsed.sync.enabled === "boolean")
124
+ enabled = parsed.sync.enabled;
125
+ if (typeof parsed.sync.remote === "string" && parsed.sync.remote.length > 0) {
126
+ remote = parsed.sync.remote;
127
+ }
128
+ }
129
+ }
130
+ catch {
131
+ // Best-effort: bundle without readable config still gets a row with defaults
132
+ }
133
+ const row = { agent, enabled, remote };
76
134
  if (enabled) {
77
- discovered.push(agentName);
135
+ // Only meaningful when sync is enabled — we don't care about disabled bundles'
136
+ // git state. Check .git presence before attempting any git invocation.
137
+ const gitInitialized = existsSync(path.join(bundleRoot, ".git"));
138
+ row.gitInitialized = gitInitialized;
139
+ if (gitInitialized) {
140
+ try {
141
+ const out = execFileSync("git", ["remote", "get-url", remote], {
142
+ cwd: bundleRoot,
143
+ stdio: "pipe",
144
+ timeout: 5000,
145
+ }).toString().trim();
146
+ if (out.length > 0)
147
+ row.remoteUrl = out;
148
+ }
149
+ catch {
150
+ // No remote configured or git missing — leave remoteUrl undefined.
151
+ }
152
+ }
78
153
  }
154
+ rows.push(row);
79
155
  }
80
- return discovered.sort((left, right) => left.localeCompare(right));
156
+ return rows;
81
157
  }