@ouro.bot/cli 0.1.0-alpha.43 → 0.1.0-alpha.431

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 (338) hide show
  1. package/README.md +123 -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 +2715 -0
  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 +426 -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 +745 -227
  36. package/dist/heart/cross-chat-delivery.js +131 -0
  37. package/dist/heart/daemon/agent-config-check.js +424 -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 +604 -0
  44. package/dist/heart/daemon/cli-exec.js +5234 -0
  45. package/dist/heart/daemon/cli-help.js +428 -0
  46. package/dist/heart/daemon/cli-parse.js +1156 -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/connect-bay.js +442 -0
  51. package/dist/heart/daemon/daemon-cli.js +28 -1582
  52. package/dist/heart/daemon/daemon-entry.js +356 -3
  53. package/dist/heart/daemon/daemon-health.js +141 -0
  54. package/dist/heart/daemon/daemon-runtime-sync.js +175 -12
  55. package/dist/heart/daemon/daemon-tombstone.js +236 -0
  56. package/dist/heart/daemon/daemon.js +684 -58
  57. package/dist/heart/daemon/doctor-types.js +8 -0
  58. package/dist/heart/daemon/doctor.js +427 -0
  59. package/dist/heart/daemon/health-monitor.js +79 -1
  60. package/dist/heart/daemon/hooks/agent-config-v2.js +33 -0
  61. package/dist/heart/daemon/hooks/bundle-meta.js +115 -1
  62. package/dist/heart/daemon/http-health-probe.js +80 -0
  63. package/dist/heart/daemon/human-command-screens.js +140 -0
  64. package/dist/heart/daemon/human-readiness.js +114 -0
  65. package/dist/heart/daemon/inner-status.js +89 -0
  66. package/dist/heart/daemon/interactive-repair.js +307 -0
  67. package/dist/heart/daemon/launchd.js +46 -9
  68. package/dist/heart/daemon/log-tailer.js +82 -12
  69. package/dist/heart/daemon/logs-prune.js +105 -0
  70. package/dist/heart/daemon/message-router.js +2 -2
  71. package/dist/heart/daemon/os-cron-deps.js +134 -0
  72. package/dist/heart/daemon/ouro-bot-entry.js +4 -2
  73. package/dist/heart/daemon/ouro-entry.js +3 -1
  74. package/dist/heart/daemon/process-manager.js +214 -0
  75. package/dist/heart/daemon/provider-discovery.js +137 -0
  76. package/dist/heart/daemon/pulse.js +475 -0
  77. package/dist/heart/daemon/readiness-repair.js +275 -0
  78. package/dist/heart/daemon/run-hooks.js +2 -0
  79. package/dist/heart/daemon/runtime-logging.js +67 -16
  80. package/dist/heart/daemon/runtime-metadata.js +73 -0
  81. package/dist/heart/daemon/runtime-mode.js +67 -0
  82. package/dist/heart/daemon/safe-mode.js +161 -0
  83. package/dist/heart/daemon/sense-manager.js +145 -32
  84. package/dist/heart/daemon/session-id-resolver.js +131 -0
  85. package/dist/heart/daemon/skill-management-installer.js +94 -0
  86. package/dist/heart/daemon/socket-client.js +307 -0
  87. package/dist/heart/daemon/stale-bundle-prune.js +96 -0
  88. package/dist/heart/daemon/startup-tui.js +272 -0
  89. package/dist/heart/daemon/task-scheduler.js +3 -25
  90. package/dist/heart/daemon/terminal-ui.js +169 -0
  91. package/dist/heart/daemon/thoughts.js +510 -0
  92. package/dist/heart/daemon/up-progress.js +271 -0
  93. package/dist/heart/delegation.js +62 -0
  94. package/dist/heart/habits/habit-migration.js +189 -0
  95. package/dist/heart/habits/habit-parser.js +140 -0
  96. package/dist/heart/habits/habit-runtime-state.js +100 -0
  97. package/dist/heart/habits/habit-scheduler.js +372 -0
  98. package/dist/heart/{daemon → hatch}/hatch-flow.js +52 -117
  99. package/dist/heart/{daemon → hatch}/hatch-specialist.js +3 -3
  100. package/dist/heart/{daemon → hatch}/specialist-prompt.js +12 -9
  101. package/dist/heart/{daemon → hatch}/specialist-tools.js +35 -12
  102. package/dist/heart/identity.js +161 -65
  103. package/dist/heart/kept-notes.js +357 -0
  104. package/dist/heart/kicks.js +1 -1
  105. package/dist/heart/machine-identity.js +161 -0
  106. package/dist/heart/mcp/mcp-server.js +653 -0
  107. package/dist/heart/migrate-config.js +100 -0
  108. package/dist/heart/model-capabilities.js +59 -0
  109. package/dist/heart/outlook/outlook-http-hooks.js +64 -0
  110. package/dist/heart/outlook/outlook-http-response.js +7 -0
  111. package/dist/heart/outlook/outlook-http-routes.js +232 -0
  112. package/dist/heart/outlook/outlook-http-static.js +99 -0
  113. package/dist/heart/outlook/outlook-http-transport.js +116 -0
  114. package/dist/heart/outlook/outlook-http.js +99 -0
  115. package/dist/heart/outlook/outlook-read.js +28 -0
  116. package/dist/heart/outlook/outlook-types.js +27 -0
  117. package/dist/heart/outlook/outlook-view.js +195 -0
  118. package/dist/heart/outlook/readers/agent-machine.js +359 -0
  119. package/dist/heart/outlook/readers/continuity-readers.js +332 -0
  120. package/dist/heart/outlook/readers/runtime-readers.js +644 -0
  121. package/dist/heart/outlook/readers/sessions.js +232 -0
  122. package/dist/heart/outlook/readers/shared.js +111 -0
  123. package/dist/heart/platform.js +81 -0
  124. package/dist/heart/progress-story.js +42 -0
  125. package/dist/heart/provider-attempt.js +133 -0
  126. package/dist/heart/provider-binding-resolver.js +239 -0
  127. package/dist/heart/provider-credentials.js +391 -0
  128. package/dist/heart/provider-failover.js +266 -0
  129. package/dist/heart/provider-models.js +81 -0
  130. package/dist/heart/provider-ping.js +237 -0
  131. package/dist/heart/provider-state.js +216 -0
  132. package/dist/heart/provider-visibility.js +186 -0
  133. package/dist/heart/providers/anthropic-token.js +131 -0
  134. package/dist/heart/providers/anthropic.js +193 -55
  135. package/dist/heart/providers/azure.js +103 -12
  136. package/dist/heart/providers/error-classification.js +63 -0
  137. package/dist/heart/providers/github-copilot.js +145 -0
  138. package/dist/heart/providers/minimax-vlm.js +189 -0
  139. package/dist/heart/providers/minimax.js +29 -7
  140. package/dist/heart/providers/openai-codex.js +62 -38
  141. package/dist/heart/runtime-capability-check.js +170 -0
  142. package/dist/heart/runtime-credentials.js +260 -0
  143. package/dist/heart/sense-truth.js +3 -0
  144. package/dist/heart/session-activity.js +190 -0
  145. package/dist/heart/session-events.js +855 -0
  146. package/dist/heart/session-transcript.js +167 -0
  147. package/dist/heart/start-of-turn-packet.js +345 -0
  148. package/dist/heart/streaming.js +36 -27
  149. package/dist/heart/sync.js +332 -0
  150. package/dist/heart/target-resolution.js +127 -0
  151. package/dist/heart/tempo.js +93 -0
  152. package/dist/heart/temporal-view.js +41 -0
  153. package/dist/heart/tool-activity-callbacks.js +36 -0
  154. package/dist/heart/tool-description.js +135 -0
  155. package/dist/heart/tool-friction.js +55 -0
  156. package/dist/heart/tool-loop.js +200 -0
  157. package/dist/heart/turn-context.js +351 -0
  158. package/dist/heart/turn-coordinator.js +28 -0
  159. package/dist/heart/{daemon → versioning}/ouro-bot-global-installer.js +1 -1
  160. package/dist/heart/{daemon → versioning}/ouro-bot-wrapper.js +1 -1
  161. package/dist/heart/versioning/ouro-path-installer.js +301 -0
  162. package/dist/heart/versioning/ouro-version-manager.js +295 -0
  163. package/dist/heart/{daemon → versioning}/staged-restart.js +40 -8
  164. package/dist/heart/{daemon → versioning}/update-checker.js +3 -1
  165. package/dist/heart/{daemon → versioning}/update-hooks.js +63 -59
  166. package/dist/mind/bundle-manifest.js +7 -1
  167. package/dist/mind/context.js +132 -93
  168. package/dist/mind/diary-integrity.js +60 -0
  169. package/dist/mind/{memory.js → diary.js} +74 -93
  170. package/dist/mind/embedding-provider.js +60 -0
  171. package/dist/mind/file-state.js +179 -0
  172. package/dist/mind/friends/channel.js +21 -0
  173. package/dist/mind/friends/group-context.js +144 -0
  174. package/dist/mind/friends/resolver.js +38 -1
  175. package/dist/mind/friends/store-file.js +39 -3
  176. package/dist/mind/friends/trust-explanation.js +74 -0
  177. package/dist/mind/friends/types.js +1 -1
  178. package/dist/mind/journal-index.js +161 -0
  179. package/dist/mind/note-search.js +268 -0
  180. package/dist/mind/obligation-steering.js +221 -0
  181. package/dist/mind/pending.js +66 -7
  182. package/dist/mind/prompt-refresh.js +3 -2
  183. package/dist/mind/prompt.js +947 -165
  184. package/dist/mind/provenance-trust.js +26 -0
  185. package/dist/mind/scrutiny.js +173 -0
  186. package/dist/nerves/cli-logging.js +7 -1
  187. package/dist/nerves/coverage/audit-rules.js +15 -6
  188. package/dist/nerves/coverage/audit.js +28 -2
  189. package/dist/nerves/coverage/cli.js +1 -1
  190. package/dist/nerves/coverage/contract.js +5 -5
  191. package/dist/nerves/coverage/file-completeness.js +83 -5
  192. package/dist/nerves/coverage/run-artifacts.js +1 -1
  193. package/dist/nerves/event-buffer.js +111 -0
  194. package/dist/nerves/index.js +224 -4
  195. package/dist/nerves/observation.js +20 -0
  196. package/dist/nerves/redact.js +79 -0
  197. package/dist/nerves/runtime.js +5 -1
  198. package/dist/outlook-ui/assets/index-BAcU08c-.css +1 -0
  199. package/dist/outlook-ui/assets/index-D7l3l4vY.js +61 -0
  200. package/dist/outlook-ui/index.html +15 -0
  201. package/dist/repertoire/ado-client.js +15 -56
  202. package/dist/repertoire/ado-semantic.js +11 -10
  203. package/dist/repertoire/api-client.js +97 -0
  204. package/dist/repertoire/bitwarden-store.js +702 -0
  205. package/dist/repertoire/bundle-templates.js +72 -0
  206. package/dist/repertoire/bw-installer.js +79 -0
  207. package/dist/repertoire/coding/codex-jsonl.js +64 -0
  208. package/dist/repertoire/coding/context-pack.js +330 -0
  209. package/dist/repertoire/coding/feedback.js +197 -30
  210. package/dist/repertoire/coding/manager.js +158 -9
  211. package/dist/repertoire/coding/spawner.js +55 -9
  212. package/dist/repertoire/coding/tools.js +170 -7
  213. package/dist/repertoire/commerce-errors.js +109 -0
  214. package/dist/repertoire/commerce-self-test.js +156 -0
  215. package/dist/repertoire/credential-access.js +111 -0
  216. package/dist/repertoire/duffel-client.js +185 -0
  217. package/dist/repertoire/github-client.js +14 -55
  218. package/dist/repertoire/graph-client.js +11 -52
  219. package/dist/repertoire/guardrails.js +371 -0
  220. package/dist/repertoire/mcp-client.js +255 -0
  221. package/dist/repertoire/mcp-manager.js +305 -0
  222. package/dist/repertoire/mcp-tools.js +63 -0
  223. package/dist/repertoire/shell-sessions.js +133 -0
  224. package/dist/repertoire/skills.js +15 -24
  225. package/dist/repertoire/stripe-client.js +131 -0
  226. package/dist/repertoire/tasks/board.js +43 -5
  227. package/dist/repertoire/tasks/fix.js +182 -0
  228. package/dist/repertoire/tasks/index.js +37 -4
  229. package/dist/repertoire/tasks/lifecycle.js +2 -2
  230. package/dist/repertoire/tasks/parser.js +3 -2
  231. package/dist/repertoire/tasks/scanner.js +194 -37
  232. package/dist/repertoire/tasks/transitions.js +16 -78
  233. package/dist/repertoire/tool-results.js +29 -0
  234. package/dist/repertoire/tools-attachments.js +317 -0
  235. package/dist/repertoire/tools-base.js +42 -690
  236. package/dist/repertoire/tools-bluebubbles.js +1 -0
  237. package/dist/repertoire/tools-bridge.js +141 -0
  238. package/dist/repertoire/tools-bundle.js +984 -0
  239. package/dist/repertoire/tools-config.js +185 -0
  240. package/dist/repertoire/tools-continuity.js +248 -0
  241. package/dist/repertoire/tools-credential.js +361 -0
  242. package/dist/repertoire/tools-files.js +342 -0
  243. package/dist/repertoire/tools-flight.js +224 -0
  244. package/dist/repertoire/tools-flow.js +105 -0
  245. package/dist/repertoire/tools-github.js +1 -7
  246. package/dist/repertoire/tools-notes.js +376 -0
  247. package/dist/repertoire/tools-session.js +739 -0
  248. package/dist/repertoire/tools-shell.js +120 -0
  249. package/dist/repertoire/tools-stripe.js +180 -0
  250. package/dist/repertoire/tools-surface.js +243 -0
  251. package/dist/repertoire/tools-teams.js +9 -39
  252. package/dist/repertoire/tools-travel.js +125 -0
  253. package/dist/repertoire/tools-user-profile.js +144 -0
  254. package/dist/repertoire/tools-vault.js +40 -0
  255. package/dist/repertoire/tools.js +144 -113
  256. package/dist/repertoire/travel-api-client.js +360 -0
  257. package/dist/repertoire/user-profile.js +131 -0
  258. package/dist/repertoire/vault-setup.js +246 -0
  259. package/dist/repertoire/vault-unlock.js +421 -0
  260. package/dist/scripts/claude-code-hook.js +41 -0
  261. package/dist/scripts/claude-code-stop-hook.js +47 -0
  262. package/dist/senses/attention-queue.js +116 -0
  263. package/dist/senses/bluebubbles/attachment-cache.js +53 -0
  264. package/dist/senses/bluebubbles/attachment-download.js +137 -0
  265. package/dist/senses/{bluebubbles-client.js → bluebubbles/client.js} +260 -9
  266. package/dist/senses/bluebubbles/entry.js +73 -0
  267. package/dist/senses/bluebubbles/inbound-log.js +113 -0
  268. package/dist/senses/bluebubbles/index.js +1620 -0
  269. package/dist/senses/{bluebubbles-media.js → bluebubbles/media.js} +121 -70
  270. package/dist/senses/{bluebubbles-model.js → bluebubbles/model.js} +33 -12
  271. package/dist/senses/{bluebubbles-mutation-log.js → bluebubbles/mutation-log.js} +45 -3
  272. package/dist/senses/bluebubbles/replay.js +129 -0
  273. package/dist/senses/bluebubbles/runtime-state.js +109 -0
  274. package/dist/senses/{bluebubbles-session-cleanup.js → bluebubbles/session-cleanup.js} +1 -1
  275. package/dist/senses/cli/bracketed-paste.js +82 -0
  276. package/dist/senses/cli/image-paste.js +287 -0
  277. package/dist/senses/cli/image-ref-navigation.js +75 -0
  278. package/dist/senses/cli/ink-app.js +156 -0
  279. package/dist/senses/cli/inline-diff.js +64 -0
  280. package/dist/senses/cli/input-keys.js +174 -0
  281. package/dist/senses/cli/kill-ring.js +86 -0
  282. package/dist/senses/cli/message-list.js +51 -0
  283. package/dist/senses/cli/ouro-tui.js +605 -0
  284. package/dist/senses/cli/spinner-imperative.js +135 -0
  285. package/dist/senses/cli/spinner.js +101 -0
  286. package/dist/senses/cli/status-line.js +60 -0
  287. package/dist/senses/cli/streaming-markdown.js +526 -0
  288. package/dist/senses/cli/tool-display.js +83 -0
  289. package/dist/senses/cli/tool-render.js +85 -0
  290. package/dist/senses/cli/tui-store.js +240 -0
  291. package/dist/senses/cli/virtual-list.js +35 -0
  292. package/dist/senses/cli-entry.js +60 -8
  293. package/dist/senses/cli-layout.js +187 -0
  294. package/dist/senses/cli.js +516 -211
  295. package/dist/senses/commands.js +66 -3
  296. package/dist/senses/habit-turn-message.js +108 -0
  297. package/dist/senses/inner-dialog-worker.js +102 -19
  298. package/dist/senses/inner-dialog.js +597 -95
  299. package/dist/senses/pipeline.js +533 -72
  300. package/dist/senses/proactive-content-guard.js +51 -0
  301. package/dist/senses/shared-turn.js +205 -0
  302. package/dist/senses/surface-tool.js +68 -0
  303. package/dist/senses/teams-entry.js +60 -8
  304. package/dist/senses/teams.js +413 -163
  305. package/dist/senses/trust-gate.js +5 -5
  306. package/package.json +32 -7
  307. package/skills/agent-commerce.md +106 -0
  308. package/skills/browser-navigation.md +117 -0
  309. package/skills/commerce-setup-guide.md +116 -0
  310. package/skills/commerce-setup.md +84 -0
  311. package/skills/configure-dev-tools.md +101 -0
  312. package/skills/travel-planning.md +138 -0
  313. package/dist/heart/daemon/ouro-path-installer.js +0 -178
  314. package/dist/heart/daemon/subagent-installer.js +0 -134
  315. package/dist/mind/associative-recall.js +0 -209
  316. package/dist/senses/bluebubbles-entry.js +0 -11
  317. package/dist/senses/bluebubbles.js +0 -854
  318. package/dist/senses/debug-activity.js +0 -127
  319. package/subagents/README.md +0 -60
  320. package/subagents/work-doer.md +0 -235
  321. package/subagents/work-merger.md +0 -618
  322. package/subagents/work-planner.md +0 -382
  323. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/basilisk.md +0 -0
  324. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jafar.md +0 -0
  325. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jormungandr.md +0 -0
  326. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/kaa.md +0 -0
  327. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/medusa.md +0 -0
  328. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/monty.md +0 -0
  329. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/nagini.md +0 -0
  330. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/ouroboros.md +0 -0
  331. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/python.md +0 -0
  332. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/quetzalcoatl.md +0 -0
  333. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/sir-hiss.md +0 -0
  334. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-snake.md +0 -0
  335. /package/dist/heart/{daemon → hatch}/hatch-animation.js +0 -0
  336. /package/dist/heart/{daemon → hatch}/specialist-orchestrator.js +0 -0
  337. /package/dist/heart/{daemon → versioning}/ouro-uti.js +0 -0
  338. /package/dist/heart/{daemon → versioning}/wrapper-publish-guard.js +0 -0
@@ -0,0 +1,161 @@
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.SAFE_MODE_OVERRIDE_FILENAME = void 0;
37
+ exports.detectSafeMode = detectSafeMode;
38
+ exports.pruneOldCrashes = pruneOldCrashes;
39
+ const fs = __importStar(require("fs"));
40
+ const path = __importStar(require("path"));
41
+ const runtime_1 = require("../../nerves/runtime");
42
+ exports.SAFE_MODE_OVERRIDE_FILENAME = "safe-mode-override.json";
43
+ /** 3+ crashes within this window triggers safe mode */
44
+ const CRASH_WINDOW_MS = 5 * 60 * 1000; // 5 minutes
45
+ const CRASH_THRESHOLD = 3;
46
+ /**
47
+ * Reads crash history from the tombstone file and determines if safe mode should be active.
48
+ * Returns active if 3+ crashes occurred within 5 minutes.
49
+ *
50
+ * Safe mode is bypassed when:
51
+ * - devMode is true
52
+ * - A safe-mode override file exists (written by `ouro up --force`)
53
+ */
54
+ function detectSafeMode(tombstonePath, options) {
55
+ const inactive = { active: false, reason: "", enteredAt: "" };
56
+ // Dev mode: never enter safe mode
57
+ if (options?.devMode) {
58
+ return inactive;
59
+ }
60
+ // Check for override file (--force)
61
+ const overridePath = path.join(path.dirname(tombstonePath), exports.SAFE_MODE_OVERRIDE_FILENAME);
62
+ try {
63
+ if (fs.existsSync(overridePath)) {
64
+ return inactive;
65
+ }
66
+ }
67
+ catch {
68
+ // Best-effort check
69
+ }
70
+ // Read tombstone
71
+ let parsed;
72
+ try {
73
+ const raw = fs.readFileSync(tombstonePath, "utf-8");
74
+ parsed = JSON.parse(raw);
75
+ }
76
+ catch {
77
+ return inactive;
78
+ }
79
+ // Extract recentCrashes
80
+ if (!Array.isArray(parsed.recentCrashes)) {
81
+ return inactive;
82
+ }
83
+ const nowMs = options?.now ? options.now() : Date.now();
84
+ const windowStart = nowMs - CRASH_WINDOW_MS;
85
+ // Filter to valid string timestamps within the crash window
86
+ const recentInWindow = parsed.recentCrashes.filter((entry) => {
87
+ if (typeof entry !== "string")
88
+ return false;
89
+ const ts = new Date(entry).getTime();
90
+ if (isNaN(ts))
91
+ return false;
92
+ return ts >= windowStart;
93
+ });
94
+ if (recentInWindow.length < CRASH_THRESHOLD) {
95
+ return inactive;
96
+ }
97
+ const result = {
98
+ active: true,
99
+ reason: `crash loop detected: ${recentInWindow.length} crashes in last 5 minutes`,
100
+ enteredAt: new Date(nowMs).toISOString(),
101
+ };
102
+ (0, runtime_1.emitNervesEvent)({
103
+ level: "error",
104
+ component: "daemon",
105
+ event: "daemon.safe_mode_entered",
106
+ message: result.reason,
107
+ meta: {
108
+ crashCount: recentInWindow.length,
109
+ windowMs: CRASH_WINDOW_MS,
110
+ tombstonePath,
111
+ },
112
+ });
113
+ return result;
114
+ }
115
+ /**
116
+ * Prunes crash entries older than 5 minutes from the tombstone's recentCrashes.
117
+ * Also removes the safe-mode override file if present.
118
+ * Called after successful startup (uptime > stability threshold).
119
+ */
120
+ function pruneOldCrashes(tombstonePath, options) {
121
+ // Remove override file
122
+ const overridePath = path.join(path.dirname(tombstonePath), exports.SAFE_MODE_OVERRIDE_FILENAME);
123
+ try {
124
+ if (fs.existsSync(overridePath)) {
125
+ fs.unlinkSync(overridePath);
126
+ }
127
+ }
128
+ catch {
129
+ // Best-effort
130
+ }
131
+ // Read existing tombstone
132
+ let parsed;
133
+ try {
134
+ const raw = fs.readFileSync(tombstonePath, "utf-8");
135
+ parsed = JSON.parse(raw);
136
+ }
137
+ catch {
138
+ return;
139
+ }
140
+ if (!Array.isArray(parsed.recentCrashes)) {
141
+ return;
142
+ }
143
+ const nowMs = options?.now ? options.now() : Date.now();
144
+ const windowStart = nowMs - CRASH_WINDOW_MS;
145
+ // Keep only entries within the window
146
+ const pruned = parsed.recentCrashes.filter((entry) => {
147
+ if (typeof entry !== "string")
148
+ return false;
149
+ const ts = new Date(entry).getTime();
150
+ if (isNaN(ts))
151
+ return false;
152
+ return ts >= windowStart;
153
+ });
154
+ parsed.recentCrashes = pruned;
155
+ try {
156
+ fs.writeFileSync(tombstonePath, JSON.stringify(parsed, null, 2) + "\n", "utf-8");
157
+ }
158
+ catch {
159
+ // Best-effort
160
+ }
161
+ }
@@ -39,11 +39,14 @@ const os = __importStar(require("os"));
39
39
  const path = __importStar(require("path"));
40
40
  const runtime_1 = require("../../nerves/runtime");
41
41
  const identity_1 = require("../identity");
42
+ const runtime_credentials_1 = require("../runtime-credentials");
42
43
  const sense_truth_1 = require("../sense-truth");
44
+ const machine_identity_1 = require("../machine-identity");
43
45
  const process_manager_1 = require("./process-manager");
44
46
  const DEFAULT_TEAMS_PORT = 3978;
45
47
  const DEFAULT_BLUEBUBBLES_PORT = 18790;
46
48
  const DEFAULT_BLUEBUBBLES_WEBHOOK_PATH = "/bluebubbles-webhook";
49
+ const BLUEBUBBLES_RUNTIME_FRESHNESS_WINDOW_MS = 90_000;
47
50
  function defaultSenses() {
48
51
  return {
49
52
  cli: { ...identity_1.DEFAULT_AGENT_SENSES.cli },
@@ -86,22 +89,6 @@ function readAgentSenses(agentJsonPath) {
86
89
  }
87
90
  return defaults;
88
91
  }
89
- function readSecretsPayload(secretsPath) {
90
- try {
91
- const raw = fs.readFileSync(secretsPath, "utf-8");
92
- const parsed = JSON.parse(raw);
93
- if (!parsed || typeof parsed !== "object" || Array.isArray(parsed)) {
94
- return { payload: {}, error: "invalid secrets.json object" };
95
- }
96
- return { payload: parsed, error: null };
97
- }
98
- catch (error) {
99
- return {
100
- payload: {},
101
- error: error instanceof Error ? error.message : String(error),
102
- };
103
- }
104
- }
105
92
  function textField(record, key) {
106
93
  const value = record?.[key];
107
94
  return typeof value === "string" ? value.trim() : "";
@@ -110,17 +97,34 @@ function numberField(record, key, fallback) {
110
97
  const value = record?.[key];
111
98
  return typeof value === "number" && Number.isFinite(value) ? value : fallback;
112
99
  }
113
- function senseFactsFromSecrets(agent, senses, secretsPath) {
100
+ function compactRuntimeConfigError(agent, error) {
101
+ const compact = error.replace(/\s+/g, " ").trim();
102
+ if (/credential vault is locked|vault locked|vault is locked/i.test(compact)) {
103
+ return `vault locked; run 'ouro vault unlock --agent ${agent}' if you have the saved secret, or 'ouro vault replace --agent ${agent}' if none was saved`;
104
+ }
105
+ return compact || "unavailable";
106
+ }
107
+ function runtimeConfigUnavailableDetail(agent, runtimeConfig) {
108
+ if (runtimeConfig.ok)
109
+ return "";
110
+ const itemName = /^vault:[^:]+:(.+)$/.exec(runtimeConfig.itemPath)?.[1] ?? "runtime/config";
111
+ if (runtimeConfig.reason === "missing")
112
+ return `missing vault ${itemName} (${agent})`;
113
+ return `vault ${itemName} unavailable (${compactRuntimeConfigError(agent, runtimeConfig.error)})`;
114
+ }
115
+ function senseFactsFromRuntimeConfig(agent, senses, runtimeConfig, machineRuntimeConfig = (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent)) {
114
116
  const base = {
115
117
  cli: { configured: true, detail: "local interactive terminal" },
116
118
  teams: { configured: false, detail: "not enabled in agent.json" },
117
119
  bluebubbles: { configured: false, detail: "not enabled in agent.json" },
118
120
  };
119
- const { payload, error } = readSecretsPayload(secretsPath);
121
+ const payload = runtimeConfig.ok ? runtimeConfig.config : {};
122
+ const unavailableDetail = runtimeConfigUnavailableDetail(agent, runtimeConfig);
120
123
  const teams = payload.teams;
121
124
  const teamsChannel = payload.teamsChannel;
122
- const bluebubbles = payload.bluebubbles;
123
- const bluebubblesChannel = payload.bluebubblesChannel;
125
+ const machinePayload = machineRuntimeConfig.ok ? machineRuntimeConfig.config : {};
126
+ const bluebubbles = machinePayload.bluebubbles;
127
+ const bluebubblesChannel = machinePayload.bluebubblesChannel;
124
128
  if (senses.teams.enabled) {
125
129
  const missing = [];
126
130
  if (!textField(teams, "clientId"))
@@ -136,9 +140,9 @@ function senseFactsFromSecrets(agent, senses, secretsPath) {
136
140
  }
137
141
  : {
138
142
  configured: false,
139
- detail: error && !fs.existsSync(secretsPath)
140
- ? `missing secrets.json (${agent})`
141
- : `missing ${missing.join("/")}`,
143
+ detail: runtimeConfig.ok
144
+ ? `missing ${missing.join("/")}`
145
+ : unavailableDetail,
142
146
  };
143
147
  }
144
148
  if (senses.bluebubbles.enabled) {
@@ -154,13 +158,25 @@ function senseFactsFromSecrets(agent, senses, secretsPath) {
154
158
  }
155
159
  : {
156
160
  configured: false,
157
- detail: error && !fs.existsSync(secretsPath)
158
- ? `missing secrets.json (${agent})`
159
- : `missing ${missing.join("/")}`,
161
+ optional: !machineRuntimeConfig.ok && machineRuntimeConfig.reason === "missing",
162
+ detail: !machineRuntimeConfig.ok && machineRuntimeConfig.reason === "missing"
163
+ ? "not attached on this machine"
164
+ : machineRuntimeConfig.ok
165
+ ? `missing ${missing.join("/")}`
166
+ : runtimeConfigUnavailableDetail(agent, machineRuntimeConfig),
160
167
  };
161
168
  }
162
169
  return base;
163
170
  }
171
+ function senseRepairHint(agent, sense) {
172
+ if (sense === "teams") {
173
+ return `Run 'ouro vault config set --agent ${agent} --key teams.clientId', teams.clientSecret, and teams.tenantId; then run 'ouro up' again.`;
174
+ }
175
+ return `Run 'ouro connect bluebubbles --agent ${agent}' to attach BlueBubbles on this machine; then run 'ouro up' again.`;
176
+ }
177
+ function currentMachineId() {
178
+ return (0, machine_identity_1.loadOrCreateMachineIdentity)({ homeDir: os.homedir() }).machineId;
179
+ }
164
180
  function parseSenseSnapshotName(name) {
165
181
  const parts = name.split(":");
166
182
  if (parts.length !== 2)
@@ -175,30 +191,112 @@ function runtimeInfoFor(status) {
175
191
  return { runtime: "running" };
176
192
  return { runtime: "error" };
177
193
  }
194
+ function blueBubblesRuntimeStateIsFresh(lastCheckedAt, now = Date.now()) {
195
+ if (!lastCheckedAt) {
196
+ return false;
197
+ }
198
+ const checkedAt = Date.parse(lastCheckedAt);
199
+ if (!Number.isFinite(checkedAt)) {
200
+ return false;
201
+ }
202
+ return checkedAt >= now - BLUEBUBBLES_RUNTIME_FRESHNESS_WINDOW_MS;
203
+ }
204
+ function readBlueBubblesRuntimeJson(runtimePath) {
205
+ try {
206
+ const raw = fs.readFileSync(runtimePath, "utf-8");
207
+ const parsed = JSON.parse(raw);
208
+ /* v8 ignore start -- branches: ternary fallbacks for missing/malformed BB runtime fields @preserve */
209
+ return {
210
+ upstreamStatus: parsed.upstreamStatus === "ok" || parsed.upstreamStatus === "error"
211
+ ? parsed.upstreamStatus
212
+ : "unknown",
213
+ detail: typeof parsed.detail === "string" && parsed.detail.trim()
214
+ ? parsed.detail
215
+ : "startup health probe pending",
216
+ lastCheckedAt: typeof parsed.lastCheckedAt === "string" ? parsed.lastCheckedAt : undefined,
217
+ };
218
+ /* v8 ignore stop */
219
+ /* v8 ignore start -- defensive: catch for missing/corrupt BB runtime state file @preserve */
220
+ }
221
+ catch {
222
+ return { upstreamStatus: "unknown", detail: "startup health probe pending" };
223
+ }
224
+ /* v8 ignore stop */
225
+ }
226
+ function readBlueBubblesRuntimeFacts(agent, bundlesRoot, snapshot) {
227
+ const agentRoot = path.join(bundlesRoot, `${agent}.ouro`);
228
+ const runtimePath = path.join(agentRoot, "state", "senses", "bluebubbles", "runtime.json");
229
+ if (!fs.existsSync(runtimePath)) {
230
+ return { runtime: snapshot?.runtime };
231
+ }
232
+ const state = readBlueBubblesRuntimeJson(runtimePath);
233
+ if (!blueBubblesRuntimeStateIsFresh(state.lastCheckedAt)) {
234
+ return { runtime: snapshot?.runtime };
235
+ }
236
+ if (state.upstreamStatus === "error") {
237
+ return {
238
+ runtime: "error",
239
+ detail: state.detail,
240
+ };
241
+ }
242
+ if (state.upstreamStatus === "ok") {
243
+ return { runtime: "running" };
244
+ }
245
+ return { runtime: snapshot?.runtime };
246
+ }
178
247
  class DaemonSenseManager {
179
248
  processManager;
180
249
  contexts;
250
+ bundlesRoot;
181
251
  constructor(options) {
182
252
  const bundlesRoot = options.bundlesRoot ?? path.join(os.homedir(), "AgentBundles");
183
- const secretsRoot = options.secretsRoot ?? path.join(os.homedir(), ".agentsecrets");
253
+ this.bundlesRoot = bundlesRoot;
184
254
  this.contexts = new Map(options.agents.map((agent) => {
185
255
  const senses = readAgentSenses(path.join(bundlesRoot, `${agent}.ouro`, "agent.json"));
186
- const facts = senseFactsFromSecrets(agent, senses, path.join(secretsRoot, agent, "secrets.json"));
256
+ const facts = senseFactsFromRuntimeConfig(agent, senses, (0, runtime_credentials_1.readRuntimeCredentialConfig)(agent), (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent));
187
257
  return [agent, { senses, facts }];
188
258
  }));
189
259
  const managedSenseAgents = [...this.contexts.entries()].flatMap(([agent, context]) => {
190
260
  return ["teams", "bluebubbles"]
191
- .filter((sense) => context.senses[sense].enabled && context.facts[sense].configured)
261
+ .filter((sense) => context.senses[sense].enabled)
192
262
  .map((sense) => ({
193
263
  name: `${agent}:${sense}`,
194
264
  agentArg: agent,
195
- entry: sense === "teams" ? "senses/teams-entry.js" : "senses/bluebubbles-entry.js",
265
+ entry: sense === "teams" ? "senses/teams-entry.js" : "senses/bluebubbles/entry.js",
196
266
  channel: sense,
197
267
  autoStart: true,
198
268
  }));
199
269
  });
200
270
  this.processManager = options.processManager ?? new process_manager_1.DaemonProcessManager({
201
271
  agents: managedSenseAgents,
272
+ configCheck: async (name) => {
273
+ const parsed = parseSenseSnapshotName(name);
274
+ if (!parsed)
275
+ return { ok: true };
276
+ const context = this.contexts.get(parsed.agent);
277
+ if (!context)
278
+ return { ok: true };
279
+ const refreshed = await (0, runtime_credentials_1.refreshRuntimeCredentialConfig)(parsed.agent, { preserveCachedOnFailure: true });
280
+ const machineRefreshed = parsed.sense === "bluebubbles"
281
+ ? await (0, runtime_credentials_1.refreshMachineRuntimeCredentialConfig)(parsed.agent, currentMachineId(), { preserveCachedOnFailure: true })
282
+ : (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(parsed.agent);
283
+ context.facts = senseFactsFromRuntimeConfig(parsed.agent, context.senses, refreshed, machineRefreshed);
284
+ const fact = context.facts[parsed.sense];
285
+ if (fact.configured)
286
+ return { ok: true };
287
+ if (fact.optional) {
288
+ return {
289
+ ok: false,
290
+ skip: true,
291
+ error: `${parsed.sense} is enabled for ${parsed.agent} but not attached on this machine`,
292
+ };
293
+ }
294
+ return {
295
+ ok: false,
296
+ error: `${parsed.sense} is enabled for ${parsed.agent} but runtime credentials are not ready: ${fact.detail}`,
297
+ fix: senseRepairHint(parsed.agent, parsed.sense),
298
+ };
299
+ },
202
300
  });
203
301
  (0, runtime_1.emitNervesEvent)({
204
302
  component: "channels",
@@ -216,6 +314,13 @@ class DaemonSenseManager {
216
314
  async stopAll() {
217
315
  await this.processManager.stopAll();
218
316
  }
317
+ /* v8 ignore start -- pid collection for orphan cleanup pidfile @preserve */
318
+ listManagedPids() {
319
+ return this.processManager.listAgentSnapshots()
320
+ .map((s) => s.pid)
321
+ .filter((pid) => pid !== null && pid !== undefined);
322
+ }
323
+ /* v8 ignore stop */
219
324
  listSenseRows() {
220
325
  const runtime = new Map();
221
326
  for (const snapshot of this.processManager.listAgentSnapshots()) {
@@ -227,6 +332,8 @@ class DaemonSenseManager {
227
332
  runtime.set(parsed.agent, current);
228
333
  }
229
334
  const rows = [...this.contexts.entries()].flatMap(([agent, context]) => {
335
+ context.facts = senseFactsFromRuntimeConfig(agent, context.senses, (0, runtime_credentials_1.readRuntimeCredentialConfig)(agent), (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent));
336
+ const blueBubblesRuntimeFacts = readBlueBubblesRuntimeFacts(agent, this.bundlesRoot, runtime.get(agent)?.bluebubbles);
230
337
  const runtimeInfo = {
231
338
  cli: { configured: true },
232
339
  teams: {
@@ -235,7 +342,8 @@ class DaemonSenseManager {
235
342
  },
236
343
  bluebubbles: {
237
344
  configured: context.facts.bluebubbles.configured,
238
- ...(runtime.get(agent)?.bluebubbles ?? {}),
345
+ optional: context.facts.bluebubbles.optional,
346
+ ...blueBubblesRuntimeFacts,
239
347
  },
240
348
  };
241
349
  const inventory = (0, sense_truth_1.getSenseInventory)({ senses: context.senses }, runtimeInfo);
@@ -245,7 +353,12 @@ class DaemonSenseManager {
245
353
  label: entry.label,
246
354
  enabled: entry.enabled,
247
355
  status: entry.status,
248
- detail: entry.enabled ? context.facts[entry.sense].detail : "not enabled in agent.json",
356
+ detail: entry.enabled
357
+ ? entry.sense === "bluebubbles"
358
+ ? blueBubblesRuntimeFacts.detail
359
+ ?? context.facts[entry.sense].detail
360
+ : context.facts[entry.sense].detail
361
+ : "not enabled in agent.json",
249
362
  }));
250
363
  });
251
364
  (0, runtime_1.emitNervesEvent)({
@@ -0,0 +1,131 @@
1
+ "use strict";
2
+ // Pluggable session ID resolver for MCP conversations.
3
+ // Tries tool-specific methods first, falls back to UUID.
4
+ //
5
+ // Claude Code: walks parent PID chain -> reads ~/.claude/sessions/{pid}.json
6
+ // Codex: reads thread_id from env (future)
7
+ // Fallback: generates UUID
8
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
9
+ if (k2 === undefined) k2 = k;
10
+ var desc = Object.getOwnPropertyDescriptor(m, k);
11
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
12
+ desc = { enumerable: true, get: function() { return m[k]; } };
13
+ }
14
+ Object.defineProperty(o, k2, desc);
15
+ }) : (function(o, m, k, k2) {
16
+ if (k2 === undefined) k2 = k;
17
+ o[k2] = m[k];
18
+ }));
19
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
20
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
21
+ }) : function(o, v) {
22
+ o["default"] = v;
23
+ });
24
+ var __importStar = (this && this.__importStar) || (function () {
25
+ var ownKeys = function(o) {
26
+ ownKeys = Object.getOwnPropertyNames || function (o) {
27
+ var ar = [];
28
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
29
+ return ar;
30
+ };
31
+ return ownKeys(o);
32
+ };
33
+ return function (mod) {
34
+ if (mod && mod.__esModule) return mod;
35
+ var result = {};
36
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
37
+ __setModuleDefault(result, mod);
38
+ return result;
39
+ };
40
+ })();
41
+ Object.defineProperty(exports, "__esModule", { value: true });
42
+ exports.resolveSessionId = resolveSessionId;
43
+ const fs = __importStar(require("fs"));
44
+ const path = __importStar(require("path"));
45
+ const os = __importStar(require("os"));
46
+ const crypto_1 = require("crypto");
47
+ const runtime_1 = require("../../nerves/runtime");
48
+ const DEFAULT_CLAUDE_SESSIONS_DIR = path.join(os.homedir(), ".claude", "sessions");
49
+ const MAX_PID_WALK_DEPTH = 10;
50
+ /**
51
+ * Try to read a Claude Code session ID from a PID-keyed session file.
52
+ * Returns the sessionId if found, null otherwise.
53
+ */
54
+ function tryReadClaudeSession(sessionsDir, pid) {
55
+ const sessionFile = path.join(sessionsDir, `${pid}.json`);
56
+ if (!fs.existsSync(sessionFile))
57
+ return null;
58
+ try {
59
+ const raw = fs.readFileSync(sessionFile, "utf-8");
60
+ const data = JSON.parse(raw);
61
+ if (typeof data.sessionId === "string" && data.sessionId.length > 0) {
62
+ return data.sessionId;
63
+ }
64
+ return null;
65
+ }
66
+ catch {
67
+ return null;
68
+ }
69
+ }
70
+ /**
71
+ * Walk the parent PID chain looking for a Claude Code session file.
72
+ * Starts at the current process PID, walks up to parent, grandparent, etc.
73
+ * Returns the session ID from the first matching file, or null.
74
+ */
75
+ function walkPidChain(sessionsDir) {
76
+ let currentPid = process.pid;
77
+ for (let depth = 0; depth < MAX_PID_WALK_DEPTH; depth++) {
78
+ const sessionId = tryReadClaudeSession(sessionsDir, currentPid);
79
+ if (sessionId) {
80
+ (0, runtime_1.emitNervesEvent)({
81
+ component: "daemon",
82
+ event: "daemon.session_id_pid_walk_hit",
83
+ message: "found Claude session via PID walk",
84
+ meta: { pid: currentPid, depth, sessionId },
85
+ });
86
+ return sessionId;
87
+ }
88
+ // Walk to parent PID
89
+ // On macOS/Linux, process.ppid gives the parent. For deeper ancestry,
90
+ // we'd need /proc/{pid}/stat or ps -o ppid=. For now, we check current + parent.
91
+ if (depth === 0) {
92
+ currentPid = process.ppid;
93
+ }
94
+ else {
95
+ // Cannot walk further without OS-specific process tree APIs
96
+ break;
97
+ }
98
+ }
99
+ return null;
100
+ }
101
+ /**
102
+ * Resolve a session ID for the current MCP connection.
103
+ * Returns a stable identifier that ties MCP tool calls to a conversation session.
104
+ *
105
+ * Resolution order:
106
+ * 1. Claude Code PID walk: check ~/.claude/sessions/{pid}.json for current + parent PID
107
+ * 2. UUID fallback: generate a random UUID
108
+ */
109
+ function resolveSessionId(options) {
110
+ const sessionsDir = options?.claudeSessionsDir ?? DEFAULT_CLAUDE_SESSIONS_DIR;
111
+ // Try Claude Code PID walk
112
+ const claudeSessionId = walkPidChain(sessionsDir);
113
+ if (claudeSessionId) {
114
+ (0, runtime_1.emitNervesEvent)({
115
+ component: "daemon",
116
+ event: "daemon.session_id_resolved",
117
+ message: "session ID resolved via Claude Code PID walk",
118
+ meta: { sessionId: claudeSessionId, method: "claude-pid-walk" },
119
+ });
120
+ return claudeSessionId;
121
+ }
122
+ // Fallback: UUID
123
+ const sessionId = (0, crypto_1.randomUUID)();
124
+ (0, runtime_1.emitNervesEvent)({
125
+ component: "daemon",
126
+ event: "daemon.session_id_resolved",
127
+ message: "session ID resolved via UUID fallback",
128
+ meta: { sessionId, method: "uuid-fallback" },
129
+ });
130
+ return sessionId;
131
+ }
@@ -0,0 +1,94 @@
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.ensureSkillManagement = ensureSkillManagement;
37
+ const fs = __importStar(require("fs"));
38
+ const path = __importStar(require("path"));
39
+ const runtime_1 = require("../../nerves/runtime");
40
+ const identity_1 = require("../identity");
41
+ const SKILL_MANAGEMENT_URL = "https://raw.githubusercontent.com/ouroborosbot/ouroboros-skills/main/skills/skill-management/SKILL.md";
42
+ async function ensureSkillManagement() {
43
+ const bundlesRoot = (0, identity_1.getAgentBundlesRoot)();
44
+ if (!fs.existsSync(bundlesRoot))
45
+ return;
46
+ // Find all agent bundles
47
+ const entries = fs.readdirSync(bundlesRoot).filter(e => e.endsWith(".ouro"));
48
+ if (entries.length === 0)
49
+ return;
50
+ // Check if ANY bundle is missing the skill
51
+ const missing = entries.filter(e => {
52
+ const targetPath = path.join(bundlesRoot, e, "skills", "skill-management.md");
53
+ return !fs.existsSync(targetPath);
54
+ });
55
+ if (missing.length === 0)
56
+ return;
57
+ // eslint-disable-next-line no-console -- terminal UX: visible install status
58
+ console.log("installing skill-management from ouroboros-skills...");
59
+ try {
60
+ const response = await fetch(SKILL_MANAGEMENT_URL);
61
+ if (!response.ok) {
62
+ // eslint-disable-next-line no-console -- terminal UX: visible install status
63
+ console.error(`✗ failed to fetch skill-management (HTTP ${response.status})`);
64
+ (0, runtime_1.emitNervesEvent)({
65
+ level: "warn",
66
+ component: "daemon",
67
+ event: "daemon.skill_management_install_error",
68
+ message: "failed to fetch skill-management from GitHub",
69
+ meta: { status: response.status, url: SKILL_MANAGEMENT_URL },
70
+ });
71
+ return;
72
+ }
73
+ const content = await response.text();
74
+ for (const bundle of missing) {
75
+ const skillsDir = path.join(bundlesRoot, bundle, "skills");
76
+ const targetPath = path.join(skillsDir, "skill-management.md");
77
+ fs.mkdirSync(skillsDir, { recursive: true });
78
+ fs.writeFileSync(targetPath, content, "utf-8");
79
+ }
80
+ // eslint-disable-next-line no-console -- terminal UX: visible install status
81
+ console.log(`✓ installed skill-management (${missing.length} agent${missing.length > 1 ? "s" : ""})`);
82
+ }
83
+ catch (error) {
84
+ // eslint-disable-next-line no-console -- terminal UX: visible install status
85
+ console.error(`✗ failed to install skill-management: ${error instanceof Error ? error.message : String(error)}`);
86
+ (0, runtime_1.emitNervesEvent)({
87
+ level: "warn",
88
+ component: "daemon",
89
+ event: "daemon.skill_management_install_error",
90
+ message: "failed to install skill-management skill",
91
+ meta: { error: error instanceof Error ? error.message : String(error) },
92
+ });
93
+ }
94
+ }