@ouro.bot/cli 0.1.0-alpha.32 → 0.1.0-alpha.321

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 (308) hide show
  1. package/README.md +188 -190
  2. package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/agent.json +3 -2
  3. package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/SOUL.md +1 -1
  4. package/changelog.json +1924 -0
  5. package/dist/arc/attention-types.js +8 -0
  6. package/dist/arc/cares.js +140 -0
  7. package/dist/arc/episodes.js +117 -0
  8. package/dist/arc/intentions.js +133 -0
  9. package/dist/arc/json-store.js +117 -0
  10. package/dist/arc/obligations.js +237 -0
  11. package/dist/arc/packets.js +193 -0
  12. package/dist/arc/presence.js +185 -0
  13. package/dist/arc/task-lifecycle.js +65 -0
  14. package/dist/heart/active-work.js +832 -0
  15. package/dist/heart/agent-entry.js +37 -2
  16. package/dist/heart/attachments/image-normalize.js +194 -0
  17. package/dist/heart/attachments/materialize.js +97 -0
  18. package/dist/heart/attachments/originals.js +88 -0
  19. package/dist/heart/attachments/render.js +29 -0
  20. package/dist/heart/attachments/sources/adapter.js +2 -0
  21. package/dist/heart/attachments/sources/bluebubbles.js +156 -0
  22. package/dist/heart/attachments/sources/cli-local-file.js +78 -0
  23. package/dist/heart/attachments/sources/index.js +16 -0
  24. package/dist/heart/attachments/store.js +103 -0
  25. package/dist/heart/attachments/types.js +93 -0
  26. package/dist/heart/auth/auth-flow.js +456 -0
  27. package/dist/heart/bridges/manager.js +358 -0
  28. package/dist/heart/bridges/state-machine.js +135 -0
  29. package/dist/heart/bridges/store.js +123 -0
  30. package/dist/heart/bundle-state.js +168 -0
  31. package/dist/heart/commitments.js +111 -0
  32. package/dist/heart/config-registry.js +304 -0
  33. package/dist/heart/config.js +63 -30
  34. package/dist/heart/core.js +669 -195
  35. package/dist/heart/cross-chat-delivery.js +131 -0
  36. package/dist/heart/daemon/agent-config-check.js +149 -0
  37. package/dist/heart/daemon/agent-discovery.js +79 -3
  38. package/dist/heart/daemon/agent-service.js +360 -0
  39. package/dist/heart/daemon/agentic-repair.js +170 -0
  40. package/dist/heart/daemon/cadence.js +70 -0
  41. package/dist/heart/daemon/cli-defaults.js +596 -0
  42. package/dist/heart/daemon/cli-exec.js +2238 -0
  43. package/dist/heart/daemon/cli-help.js +306 -0
  44. package/dist/heart/daemon/cli-parse.js +824 -0
  45. package/dist/heart/daemon/cli-render-doctor.js +57 -0
  46. package/dist/heart/daemon/cli-render.js +506 -0
  47. package/dist/heart/daemon/cli-types.js +8 -0
  48. package/dist/heart/daemon/daemon-cli.js +29 -1171
  49. package/dist/heart/daemon/daemon-entry.js +333 -3
  50. package/dist/heart/daemon/daemon-health.js +137 -0
  51. package/dist/heart/daemon/daemon-runtime-sync.js +153 -12
  52. package/dist/heart/daemon/daemon-tombstone.js +236 -0
  53. package/dist/heart/daemon/daemon.js +751 -58
  54. package/dist/heart/daemon/doctor-types.js +8 -0
  55. package/dist/heart/daemon/doctor.js +322 -0
  56. package/dist/heart/daemon/health-monitor.js +66 -0
  57. package/dist/heart/daemon/hooks/agent-config-v2.js +33 -0
  58. package/dist/heart/daemon/hooks/bundle-meta.js +115 -1
  59. package/dist/heart/daemon/http-health-probe.js +80 -0
  60. package/dist/heart/daemon/inner-status.js +89 -0
  61. package/dist/heart/daemon/interactive-repair.js +69 -0
  62. package/dist/heart/daemon/launchd.js +46 -9
  63. package/dist/heart/daemon/log-tailer.js +82 -12
  64. package/dist/heart/daemon/logs-prune.js +105 -0
  65. package/dist/heart/daemon/message-router.js +17 -8
  66. package/dist/heart/daemon/os-cron-deps.js +134 -0
  67. package/dist/heart/daemon/ouro-bot-entry.js +1 -1
  68. package/dist/heart/daemon/process-manager.js +201 -0
  69. package/dist/heart/daemon/provider-discovery.js +105 -0
  70. package/dist/heart/daemon/pulse.js +463 -0
  71. package/dist/heart/daemon/run-hooks.js +2 -0
  72. package/dist/heart/daemon/runtime-logging.js +67 -16
  73. package/dist/heart/daemon/runtime-metadata.js +101 -0
  74. package/dist/heart/daemon/runtime-mode.js +67 -0
  75. package/dist/heart/daemon/safe-mode.js +161 -0
  76. package/dist/heart/daemon/sense-manager.js +72 -3
  77. package/dist/heart/daemon/session-id-resolver.js +131 -0
  78. package/dist/heart/daemon/skill-management-installer.js +94 -0
  79. package/dist/heart/daemon/socket-client.js +307 -0
  80. package/dist/heart/daemon/stale-bundle-prune.js +96 -0
  81. package/dist/heart/daemon/startup-tui.js +227 -0
  82. package/dist/heart/daemon/task-scheduler.js +3 -25
  83. package/dist/heart/daemon/thoughts.js +510 -0
  84. package/dist/heart/daemon/up-progress.js +135 -0
  85. package/dist/heart/delegation.js +62 -0
  86. package/dist/heart/habits/habit-migration.js +181 -0
  87. package/dist/heart/habits/habit-parser.js +140 -0
  88. package/dist/heart/habits/habit-scheduler.js +371 -0
  89. package/dist/heart/{daemon → hatch}/hatch-flow.js +30 -120
  90. package/dist/heart/{daemon → hatch}/hatch-specialist.js +3 -3
  91. package/dist/heart/{daemon → hatch}/specialist-prompt.js +10 -7
  92. package/dist/heart/{daemon → hatch}/specialist-tools.js +49 -3
  93. package/dist/heart/identity.js +163 -60
  94. package/dist/heart/kicks.js +2 -20
  95. package/dist/heart/mcp/mcp-server.js +653 -0
  96. package/dist/heart/migrate-config.js +127 -0
  97. package/dist/heart/model-capabilities.js +59 -0
  98. package/dist/heart/outlook/outlook-http.js +439 -0
  99. package/dist/heart/outlook/outlook-read.js +28 -0
  100. package/dist/heart/outlook/outlook-render.js +1032 -0
  101. package/dist/heart/outlook/outlook-types.js +27 -0
  102. package/dist/heart/outlook/outlook-view.js +194 -0
  103. package/dist/heart/outlook/readers/agent-machine.js +355 -0
  104. package/dist/heart/outlook/readers/continuity-readers.js +332 -0
  105. package/dist/heart/outlook/readers/runtime-readers.js +660 -0
  106. package/dist/heart/outlook/readers/sessions.js +231 -0
  107. package/dist/heart/outlook/readers/shared.js +111 -0
  108. package/dist/heart/progress-story.js +42 -0
  109. package/dist/heart/provider-failover.js +88 -0
  110. package/dist/heart/provider-ping.js +162 -0
  111. package/dist/heart/providers/anthropic-token.js +163 -0
  112. package/dist/heart/providers/anthropic.js +169 -46
  113. package/dist/heart/providers/azure.js +98 -11
  114. package/dist/heart/providers/error-classification.js +63 -0
  115. package/dist/heart/providers/github-copilot.js +136 -0
  116. package/dist/heart/providers/minimax-vlm.js +189 -0
  117. package/dist/heart/providers/minimax.js +23 -5
  118. package/dist/heart/providers/openai-codex.js +33 -22
  119. package/dist/heart/session-activity.js +190 -0
  120. package/dist/heart/session-events.js +726 -0
  121. package/dist/heart/session-recall.js +162 -0
  122. package/dist/heart/start-of-turn-packet.js +341 -0
  123. package/dist/heart/streaming.js +36 -27
  124. package/dist/heart/sync.js +332 -0
  125. package/dist/heart/target-resolution.js +127 -0
  126. package/dist/heart/tempo.js +93 -0
  127. package/dist/heart/temporal-view.js +41 -0
  128. package/dist/heart/tool-activity-callbacks.js +36 -0
  129. package/dist/heart/tool-description.js +135 -0
  130. package/dist/heart/tool-friction.js +55 -0
  131. package/dist/heart/tool-loop.js +200 -0
  132. package/dist/heart/turn-context.js +358 -0
  133. package/dist/heart/turn-coordinator.js +28 -0
  134. package/dist/heart/{daemon → versioning}/ouro-bot-global-installer.js +1 -1
  135. package/dist/heart/{daemon → versioning}/ouro-bot-wrapper.js +1 -1
  136. package/dist/heart/{daemon → versioning}/ouro-path-installer.js +78 -35
  137. package/dist/heart/versioning/ouro-version-manager.js +295 -0
  138. package/dist/heart/{daemon → versioning}/staged-restart.js +40 -8
  139. package/dist/heart/{daemon → versioning}/update-checker.js +12 -2
  140. package/dist/heart/{daemon → versioning}/update-hooks.js +63 -59
  141. package/dist/mind/associative-recall.js +137 -66
  142. package/dist/mind/bundle-manifest.js +8 -1
  143. package/dist/mind/context.js +89 -93
  144. package/dist/mind/diary-integrity.js +60 -0
  145. package/dist/mind/{memory.js → diary.js} +84 -96
  146. package/dist/mind/embedding-provider.js +60 -0
  147. package/dist/mind/file-state.js +179 -0
  148. package/dist/mind/first-impressions.js +14 -1
  149. package/dist/mind/friends/channel.js +56 -0
  150. package/dist/mind/friends/group-context.js +144 -0
  151. package/dist/mind/friends/resolver.js +37 -0
  152. package/dist/mind/friends/store-file.js +58 -3
  153. package/dist/mind/friends/trust-explanation.js +74 -0
  154. package/dist/mind/friends/types.js +8 -0
  155. package/dist/mind/journal-index.js +161 -0
  156. package/dist/mind/obligation-steering.js +221 -0
  157. package/dist/mind/pending.js +76 -9
  158. package/dist/mind/prompt.js +950 -113
  159. package/dist/mind/provenance-trust.js +26 -0
  160. package/dist/mind/scrutiny.js +173 -0
  161. package/dist/mind/token-estimate.js +8 -12
  162. package/dist/nerves/cli-logging.js +7 -1
  163. package/dist/nerves/coverage/audit.js +1 -1
  164. package/dist/nerves/coverage/file-completeness.js +76 -5
  165. package/dist/nerves/coverage/run-artifacts.js +1 -1
  166. package/dist/nerves/event-buffer.js +111 -0
  167. package/dist/nerves/index.js +224 -4
  168. package/dist/nerves/observation.js +20 -0
  169. package/dist/nerves/redact.js +79 -0
  170. package/dist/nerves/runtime.js +5 -1
  171. package/dist/outlook-ui/assets/index-IuR4F6y6.js +61 -0
  172. package/dist/outlook-ui/assets/index-LwChZTgL.css +1 -0
  173. package/dist/outlook-ui/index.html +15 -0
  174. package/dist/repertoire/ado-client.js +15 -56
  175. package/dist/repertoire/ado-semantic.js +11 -10
  176. package/dist/repertoire/api-client.js +97 -0
  177. package/dist/repertoire/bitwarden-store.js +319 -0
  178. package/dist/repertoire/bundle-templates.js +72 -0
  179. package/dist/repertoire/bw-installer.js +79 -0
  180. package/dist/repertoire/coding/codex-jsonl.js +64 -0
  181. package/dist/repertoire/coding/context-pack.js +330 -0
  182. package/dist/repertoire/coding/feedback.js +197 -30
  183. package/dist/repertoire/coding/manager.js +159 -11
  184. package/dist/repertoire/coding/spawner.js +55 -9
  185. package/dist/repertoire/coding/tools.js +170 -7
  186. package/dist/repertoire/commerce-errors.js +109 -0
  187. package/dist/repertoire/commerce-self-test.js +156 -0
  188. package/dist/repertoire/credential-access.js +527 -0
  189. package/dist/repertoire/duffel-client.js +185 -0
  190. package/dist/repertoire/github-client.js +14 -55
  191. package/dist/repertoire/graph-client.js +11 -52
  192. package/dist/repertoire/guardrails.js +375 -0
  193. package/dist/repertoire/mcp-client.js +255 -0
  194. package/dist/repertoire/mcp-manager.js +305 -0
  195. package/dist/repertoire/mcp-tools.js +63 -0
  196. package/dist/repertoire/shell-sessions.js +133 -0
  197. package/dist/repertoire/skills.js +14 -23
  198. package/dist/repertoire/stripe-client.js +131 -0
  199. package/dist/repertoire/tasks/board.js +43 -5
  200. package/dist/repertoire/tasks/fix.js +182 -0
  201. package/dist/repertoire/tasks/index.js +28 -10
  202. package/dist/repertoire/tasks/lifecycle.js +2 -2
  203. package/dist/repertoire/tasks/parser.js +3 -2
  204. package/dist/repertoire/tasks/scanner.js +194 -37
  205. package/dist/repertoire/tasks/transitions.js +16 -79
  206. package/dist/repertoire/tool-results.js +29 -0
  207. package/dist/repertoire/tools-attachments.js +316 -0
  208. package/dist/repertoire/tools-base.js +45 -771
  209. package/dist/repertoire/tools-bluebubbles.js +1 -0
  210. package/dist/repertoire/tools-bridge.js +141 -0
  211. package/dist/repertoire/tools-bundle.js +984 -0
  212. package/dist/repertoire/tools-config.js +185 -0
  213. package/dist/repertoire/tools-continuity.js +248 -0
  214. package/dist/repertoire/tools-credential.js +182 -0
  215. package/dist/repertoire/tools-files.js +342 -0
  216. package/dist/repertoire/tools-flight.js +224 -0
  217. package/dist/repertoire/tools-flow.js +105 -0
  218. package/dist/repertoire/tools-github.js +1 -7
  219. package/dist/repertoire/tools-memory.js +376 -0
  220. package/dist/repertoire/tools-session.js +739 -0
  221. package/dist/repertoire/tools-shell.js +120 -0
  222. package/dist/repertoire/tools-stripe.js +180 -0
  223. package/dist/repertoire/tools-surface.js +243 -0
  224. package/dist/repertoire/tools-teams.js +12 -62
  225. package/dist/repertoire/tools-travel.js +125 -0
  226. package/dist/repertoire/tools-user-profile.js +144 -0
  227. package/dist/repertoire/tools-vault.js +110 -0
  228. package/dist/repertoire/tools.js +144 -138
  229. package/dist/repertoire/travel-api-client.js +360 -0
  230. package/dist/repertoire/user-profile.js +118 -0
  231. package/dist/repertoire/vault-setup.js +241 -0
  232. package/dist/scripts/claude-code-hook.js +41 -0
  233. package/dist/scripts/claude-code-stop-hook.js +47 -0
  234. package/dist/senses/attention-queue.js +116 -0
  235. package/dist/senses/bluebubbles/attachment-cache.js +53 -0
  236. package/dist/senses/bluebubbles/attachment-download.js +137 -0
  237. package/dist/senses/{bluebubbles-client.js → bluebubbles/client.js} +143 -9
  238. package/dist/senses/bluebubbles/entry.js +13 -0
  239. package/dist/senses/bluebubbles/inbound-log.js +113 -0
  240. package/dist/senses/bluebubbles/index.js +1436 -0
  241. package/dist/senses/{bluebubbles-media.js → bluebubbles/media.js} +121 -70
  242. package/dist/senses/{bluebubbles-model.js → bluebubbles/model.js} +43 -12
  243. package/dist/senses/{bluebubbles-mutation-log.js → bluebubbles/mutation-log.js} +46 -6
  244. package/dist/senses/bluebubbles/replay.js +129 -0
  245. package/dist/senses/bluebubbles/runtime-state.js +109 -0
  246. package/dist/senses/{bluebubbles-session-cleanup.js → bluebubbles/session-cleanup.js} +1 -1
  247. package/dist/senses/cli/bracketed-paste.js +82 -0
  248. package/dist/senses/cli/image-paste.js +287 -0
  249. package/dist/senses/cli/image-ref-navigation.js +75 -0
  250. package/dist/senses/cli/ink-app.js +156 -0
  251. package/dist/senses/cli/inline-diff.js +64 -0
  252. package/dist/senses/cli/input-keys.js +174 -0
  253. package/dist/senses/cli/kill-ring.js +86 -0
  254. package/dist/senses/cli/message-list.js +51 -0
  255. package/dist/senses/cli/ouro-tui.js +605 -0
  256. package/dist/senses/cli/spinner-imperative.js +135 -0
  257. package/dist/senses/cli/spinner.js +101 -0
  258. package/dist/senses/cli/status-line.js +60 -0
  259. package/dist/senses/cli/streaming-markdown.js +526 -0
  260. package/dist/senses/cli/tool-display.js +83 -0
  261. package/dist/senses/cli/tool-render.js +85 -0
  262. package/dist/senses/cli/tui-store.js +240 -0
  263. package/dist/senses/cli/virtual-list.js +35 -0
  264. package/dist/senses/cli-entry.js +1 -1
  265. package/dist/senses/cli-layout.js +187 -0
  266. package/dist/senses/cli.js +595 -246
  267. package/dist/senses/commands.js +65 -1
  268. package/dist/senses/continuity.js +94 -0
  269. package/dist/senses/habit-turn-message.js +108 -0
  270. package/dist/senses/inner-dialog-worker.js +112 -19
  271. package/dist/senses/inner-dialog.js +633 -86
  272. package/dist/senses/pipeline.js +565 -0
  273. package/dist/senses/shared-turn.js +199 -0
  274. package/dist/senses/surface-tool.js +68 -0
  275. package/dist/senses/teams.js +666 -166
  276. package/dist/senses/trust-gate.js +112 -2
  277. package/package.json +27 -7
  278. package/skills/agent-commerce.md +106 -0
  279. package/skills/browser-navigation.md +110 -0
  280. package/skills/commerce-setup-guide.md +116 -0
  281. package/skills/commerce-setup.md +84 -0
  282. package/skills/configure-dev-tools.md +81 -0
  283. package/skills/travel-planning.md +138 -0
  284. package/dist/heart/daemon/subagent-installer.js +0 -134
  285. package/dist/senses/bluebubbles-entry.js +0 -11
  286. package/dist/senses/bluebubbles.js +0 -544
  287. package/dist/senses/debug-activity.js +0 -108
  288. package/subagents/README.md +0 -73
  289. package/subagents/work-doer.md +0 -235
  290. package/subagents/work-merger.md +0 -618
  291. package/subagents/work-planner.md +0 -382
  292. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/basilisk.md +0 -0
  293. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jafar.md +0 -0
  294. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jormungandr.md +0 -0
  295. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/kaa.md +0 -0
  296. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/medusa.md +0 -0
  297. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/monty.md +0 -0
  298. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/nagini.md +0 -0
  299. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/ouroboros.md +0 -0
  300. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/python.md +0 -0
  301. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/quetzalcoatl.md +0 -0
  302. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/sir-hiss.md +0 -0
  303. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-serpent.md +0 -0
  304. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-snake.md +0 -0
  305. /package/dist/heart/{daemon → hatch}/hatch-animation.js +0 -0
  306. /package/dist/heart/{daemon → hatch}/specialist-orchestrator.js +0 -0
  307. /package/dist/heart/{daemon → versioning}/ouro-uti.js +0 -0
  308. /package/dist/heart/{daemon → versioning}/wrapper-publish-guard.js +0 -0
@@ -1,15 +1,66 @@
1
1
  #!/usr/bin/env node
2
2
  "use strict";
3
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
4
+ if (k2 === undefined) k2 = k;
5
+ var desc = Object.getOwnPropertyDescriptor(m, k);
6
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
7
+ desc = { enumerable: true, get: function() { return m[k]; } };
8
+ }
9
+ Object.defineProperty(o, k2, desc);
10
+ }) : (function(o, m, k, k2) {
11
+ if (k2 === undefined) k2 = k;
12
+ o[k2] = m[k];
13
+ }));
14
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
15
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
16
+ }) : function(o, v) {
17
+ o["default"] = v;
18
+ });
19
+ var __importStar = (this && this.__importStar) || (function () {
20
+ var ownKeys = function(o) {
21
+ ownKeys = Object.getOwnPropertyNames || function (o) {
22
+ var ar = [];
23
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
24
+ return ar;
25
+ };
26
+ return ownKeys(o);
27
+ };
28
+ return function (mod) {
29
+ if (mod && mod.__esModule) return mod;
30
+ var result = {};
31
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
32
+ __setModuleDefault(result, mod);
33
+ return result;
34
+ };
35
+ })();
3
36
  Object.defineProperty(exports, "__esModule", { value: true });
37
+ const fs = __importStar(require("fs"));
38
+ const path = __importStar(require("path"));
4
39
  const process_manager_1 = require("./process-manager");
5
40
  const daemon_1 = require("./daemon");
6
41
  const runtime_1 = require("../../nerves/runtime");
42
+ const index_1 = require("../../nerves/index");
7
43
  const message_router_1 = require("./message-router");
8
44
  const health_monitor_1 = require("./health-monitor");
45
+ const daemon_health_1 = require("./daemon-health");
9
46
  const task_scheduler_1 = require("./task-scheduler");
10
47
  const runtime_logging_1 = require("./runtime-logging");
11
48
  const sense_manager_1 = require("./sense-manager");
12
49
  const agent_discovery_1 = require("./agent-discovery");
50
+ const identity_1 = require("../identity");
51
+ const runtime_mode_1 = require("./runtime-mode");
52
+ const habit_scheduler_1 = require("../habits/habit-scheduler");
53
+ const habit_migration_1 = require("../habits/habit-migration");
54
+ const os_cron_deps_1 = require("./os-cron-deps");
55
+ const os_cron_1 = require("./os-cron");
56
+ const daemon_tombstone_1 = require("./daemon-tombstone");
57
+ const os = __importStar(require("os"));
58
+ const agent_config_check_1 = require("./agent-config-check");
59
+ const pulse_1 = require("./pulse");
60
+ const socket_client_1 = require("./socket-client");
61
+ const bundle_manifest_1 = require("../../mind/bundle-manifest");
62
+ const http_health_probe_1 = require("./http-health-probe");
63
+ const config_1 = require("../config");
13
64
  function parseSocketPath(argv) {
14
65
  const socketIndex = argv.indexOf("--socket");
15
66
  if (socketIndex >= 0) {
@@ -21,12 +72,24 @@ function parseSocketPath(argv) {
21
72
  }
22
73
  const socketPath = parseSocketPath(process.argv);
23
74
  (0, runtime_logging_1.configureDaemonRuntimeLogger)("daemon");
75
+ const entryPath = path.resolve(__dirname, "daemon-entry.js");
76
+ const mode = (0, runtime_mode_1.detectRuntimeMode)((0, identity_1.getRepoRoot)());
24
77
  (0, runtime_1.emitNervesEvent)({
25
78
  component: "daemon",
26
79
  event: "daemon.entry_start",
27
80
  message: "starting daemon entrypoint",
28
- meta: { socketPath },
81
+ meta: { socketPath, entryPath, mode },
29
82
  });
83
+ /* v8 ignore next -- dev-mode indicator: false branch (production) tested in daemon-boot-updates.test.ts @preserve */
84
+ if (mode === "dev") {
85
+ const repoRoot = (0, identity_1.getRepoRoot)();
86
+ (0, runtime_1.emitNervesEvent)({
87
+ component: "daemon",
88
+ event: "daemon.dev_mode_indicator",
89
+ message: `[dev] running from ${repoRoot}`,
90
+ meta: { repoRoot },
91
+ });
92
+ }
30
93
  const managedAgents = (0, agent_discovery_1.listEnabledBundleAgents)();
31
94
  const processManager = new process_manager_1.DaemonProcessManager({
32
95
  agents: managedAgents.map((agent) => ({
@@ -35,6 +98,30 @@ const processManager = new process_manager_1.DaemonProcessManager({
35
98
  channel: "inner-dialog",
36
99
  autoStart: true,
37
100
  })),
101
+ existsSync: fs.existsSync,
102
+ /* v8 ignore next 4 -- wiring: delegates to checkAgentConfig which has full unit tests @preserve */
103
+ configCheck: (agent) => {
104
+ const bundlesRoot = (0, identity_1.getAgentBundlesRoot)();
105
+ const secretsRoot = path.join(os.homedir(), ".agentsecrets");
106
+ return (0, agent_config_check_1.checkAgentConfig)(agent, bundlesRoot, secretsRoot);
107
+ },
108
+ /* v8 ignore start -- pulse flush wiring: integration code; flushPulse itself has full unit tests @preserve */
109
+ onSnapshotChange: () => {
110
+ (0, pulse_1.flushPulse)({
111
+ snapshots: processManager.listAgentSnapshots(),
112
+ bundlesRoot: (0, identity_1.getAgentBundlesRoot)(),
113
+ daemonVersion: (0, bundle_manifest_1.getPackageVersion)(),
114
+ now: new Date(),
115
+ // Default I/O wired into pulse.ts (writePulse, readPulse, etc.)
116
+ // Wake recipient: send inner.wake over the daemon's own socket so
117
+ // the recipient agent runs an inner-dialog turn that picks up the
118
+ // pulse alert. Catch errors silently — pulse is best-effort.
119
+ fireInnerWake: (agent) => {
120
+ (0, socket_client_1.sendDaemonCommand)(socketPath, { kind: "inner.wake", agent }).catch(() => { });
121
+ },
122
+ });
123
+ },
124
+ /* v8 ignore stop */
38
125
  });
39
126
  const scheduler = new task_scheduler_1.TaskDrivenScheduler({
40
127
  agents: [...managedAgents],
@@ -43,9 +130,19 @@ const router = new message_router_1.FileMessageRouter();
43
130
  const senseManager = new sense_manager_1.DaemonSenseManager({
44
131
  agents: [...managedAgents],
45
132
  });
133
+ /* v8 ignore next 5 -- entry-point wiring: probe factory and HealthMonitor both have full unit tests @preserve */
134
+ let bbPort = 18790;
135
+ try {
136
+ bbPort = (0, config_1.getBlueBubblesChannelConfig)().port;
137
+ }
138
+ catch {
139
+ // Daemon runs without --agent; agent-scoped config may not be available.
140
+ }
141
+ const bbProbe = (0, http_health_probe_1.createHttpHealthProbe)("bluebubbles", bbPort);
46
142
  const healthMonitor = new health_monitor_1.HealthMonitor({
47
143
  processManager,
48
144
  scheduler,
145
+ senseProbes: [bbProbe],
49
146
  alertSink: (message) => {
50
147
  (0, runtime_1.emitNervesEvent)({
51
148
  level: "error",
@@ -55,6 +152,13 @@ const healthMonitor = new health_monitor_1.HealthMonitor({
55
152
  meta: { message },
56
153
  });
57
154
  },
155
+ /* v8 ignore next 3 -- wiring: delegates to processManager.restartAgent which has full unit tests @preserve */
156
+ onCriticalAgent: (agentName) => {
157
+ try {
158
+ processManager.restartAgent(agentName);
159
+ }
160
+ catch { /* recovery is best-effort */ }
161
+ },
58
162
  });
59
163
  const daemon = new daemon_1.OuroDaemon({
60
164
  socketPath,
@@ -63,21 +167,247 @@ const daemon = new daemon_1.OuroDaemon({
63
167
  scheduler,
64
168
  healthMonitor,
65
169
  router,
170
+ mode,
66
171
  });
67
- void daemon.start().catch(async () => {
172
+ const daemonStartedAt = new Date().toISOString();
173
+ const degradedComponents = [];
174
+ function buildDaemonHealthState() {
175
+ return {
176
+ status: degradedComponents.length > 0 ? "degraded" : "ok",
177
+ mode,
178
+ pid: process.pid,
179
+ startedAt: daemonStartedAt,
180
+ uptimeSeconds: Math.floor(process.uptime()),
181
+ safeMode: null,
182
+ degraded: degradedComponents.map((entry) => ({ ...entry })),
183
+ agents: {},
184
+ habits: {},
185
+ };
186
+ }
187
+ function recordRecoverableBootstrapFailure(options) {
188
+ const errorMessage = options.error instanceof Error ? options.error.message : String(options.error);
189
+ const existing = degradedComponents.find((entry) => entry.component === options.component);
190
+ const reason = `${errorMessage}. ${options.guidance}`;
191
+ if (existing) {
192
+ existing.reason = reason;
193
+ }
194
+ else {
195
+ degradedComponents.push({
196
+ component: options.component,
197
+ reason,
198
+ since: new Date().toISOString(),
199
+ });
200
+ }
201
+ (0, runtime_1.emitNervesEvent)({
202
+ level: "warn",
203
+ component: "daemon",
204
+ event: "daemon.bootstrap_degraded",
205
+ message: "recoverable daemon bootstrap failure; daemon remains available in degraded mode",
206
+ meta: {
207
+ agent: options.agent,
208
+ component: options.component,
209
+ habitsDir: options.habitsDir,
210
+ error: errorMessage,
211
+ guidance: options.guidance,
212
+ },
213
+ });
214
+ }
215
+ function emitHabitSetupError(agent, error) {
216
+ const normalized = error instanceof Error ? error : new Error(String(error));
217
+ (0, runtime_1.emitNervesEvent)({
218
+ level: "error",
219
+ component: "daemon",
220
+ event: "daemon.habit_setup_error",
221
+ message: `habit setup failed for agent ${agent}`,
222
+ meta: { agent, error: normalized.message },
223
+ });
224
+ }
225
+ /* v8 ignore start — daemon health writer wiring, tested via daemon-health.test.ts @preserve */
226
+ const healthWriter = new daemon_health_1.DaemonHealthWriter((0, daemon_health_1.getDefaultHealthPath)());
227
+ const healthSink = (0, daemon_health_1.createHealthNervesSink)(healthWriter, buildDaemonHealthState);
228
+ (0, index_1.registerGlobalLogSink)(healthSink);
229
+ /* v8 ignore stop */
230
+ const habitSchedulers = [];
231
+ /* v8 ignore start -- habit wiring: lambdas delegate to processManager/fs; tested via HabitScheduler unit tests @preserve */
232
+ void daemon.start().then(() => {
233
+ const bundlesRoot = (0, identity_1.getAgentBundlesRoot)();
234
+ const ouroPath = (0, os_cron_deps_1.resolveOuroBinaryPath)();
235
+ const osCronDeps = (0, os_cron_deps_1.createRealOsCronDeps)();
236
+ for (const agent of managedAgents) {
237
+ const bundleRoot = path.join(bundlesRoot, `${agent}.ouro`);
238
+ const habitsDir = path.join(bundleRoot, "habits");
239
+ const degradedComponent = `habits:${agent}`;
240
+ try {
241
+ // Migrate old tasks/habits/ to habits/ at bundle root
242
+ (0, habit_migration_1.migrateHabitsFromTaskSystem)(bundleRoot);
243
+ const osCronManager = new os_cron_1.LaunchdCronManager(osCronDeps);
244
+ const scheduler = new habit_scheduler_1.HabitScheduler({
245
+ agent,
246
+ habitsDir,
247
+ osCronManager,
248
+ onHabitFire: (habitName) => {
249
+ processManager.sendToAgent(agent, { type: "habit", habitName });
250
+ },
251
+ deps: {
252
+ readdir: (dir) => fs.readdirSync(dir),
253
+ readFile: (p, enc) => fs.readFileSync(p, enc),
254
+ writeFile: (p, c, enc) => fs.writeFileSync(p, c, enc),
255
+ existsSync: (p) => fs.existsSync(p),
256
+ now: () => Date.now(),
257
+ ouroPath,
258
+ watch: (dir, cb) => fs.watch(dir, cb),
259
+ },
260
+ });
261
+ try {
262
+ scheduler.start();
263
+ scheduler.startPeriodicReconciliation();
264
+ scheduler.watchForChanges();
265
+ habitSchedulers.push(scheduler);
266
+ }
267
+ catch (error) {
268
+ try {
269
+ scheduler.stopWatching();
270
+ scheduler.stop();
271
+ }
272
+ catch {
273
+ // Cleanup is best-effort for partially initialized schedulers.
274
+ }
275
+ emitHabitSetupError(agent, error);
276
+ recordRecoverableBootstrapFailure({
277
+ agent,
278
+ component: degradedComponent,
279
+ habitsDir,
280
+ error,
281
+ guidance: `fix ${agent} habits or cron setup and rerun ouro up to restore habit automation`,
282
+ });
283
+ }
284
+ }
285
+ catch (err) {
286
+ const error = err instanceof Error ? err : new Error(String(err));
287
+ emitHabitSetupError(agent, error);
288
+ recordRecoverableBootstrapFailure({
289
+ agent,
290
+ component: degradedComponent,
291
+ habitsDir,
292
+ error,
293
+ guidance: `fix ${agent} habits or cron setup and rerun ouro up to restore habit automation`,
294
+ });
295
+ }
296
+ }
297
+ healthMonitor.startPeriodicChecks(60_000);
298
+ /* v8 ignore start -- startup failure + signal handlers: call process.exit, untestable in vitest @preserve */
299
+ }).catch(async (err) => {
300
+ const error = err instanceof Error ? err : new Error(String(err));
301
+ _tombstoneWritten = true;
302
+ (0, daemon_tombstone_1.writeDaemonTombstone)("startupFailure", error);
68
303
  (0, runtime_1.emitNervesEvent)({
69
304
  level: "error",
70
305
  component: "daemon",
71
306
  event: "daemon.entry_error",
72
307
  message: "daemon entrypoint failed",
73
- meta: {},
308
+ meta: { error: error.message },
74
309
  });
310
+ setTimeout(() => process.exit(1), 5_000).unref();
75
311
  await daemon.stop();
76
312
  process.exit(1);
77
313
  });
78
314
  process.on("SIGINT", () => {
315
+ // ALWAYS write a tombstone, even on signal-driven shutdown. The previous
316
+ // behavior was to set _gracefulShutdown=true and skip the tombstone, which
317
+ // meant ANY external SIGINT/SIGTERM (launchd policy, OOM killer, manual
318
+ // kill, killOrphanProcesses from a sibling daemon) silently disappeared
319
+ // from the death log. The user lost weeks of visibility into why their
320
+ // daemon kept dying. Tombstones are informational — having a "sigint"
321
+ // tombstone is strictly better than silence.
322
+ _tombstoneWritten = true;
323
+ (0, daemon_tombstone_1.writeDaemonTombstone)("sigint", new Error("daemon received SIGINT"));
324
+ for (const s of habitSchedulers) {
325
+ s.stopWatching();
326
+ s.stop();
327
+ }
328
+ healthMonitor.stopPeriodicChecks();
329
+ setTimeout(() => process.exit(1), 5_000).unref();
79
330
  void daemon.stop().then(() => process.exit(0));
80
331
  });
81
332
  process.on("SIGTERM", () => {
333
+ _tombstoneWritten = true;
334
+ (0, daemon_tombstone_1.writeDaemonTombstone)("sigterm", new Error("daemon received SIGTERM"));
335
+ for (const s of habitSchedulers) {
336
+ s.stopWatching();
337
+ s.stop();
338
+ }
339
+ healthMonitor.stopPeriodicChecks();
340
+ setTimeout(() => process.exit(1), 5_000).unref();
82
341
  void daemon.stop().then(() => process.exit(0));
83
342
  });
343
+ /* v8 ignore stop */
344
+ // Suppress EPIPE on stdout/stderr — normal when detached daemon's parent exits
345
+ /* v8 ignore start -- EPIPE suppression: only fires when parent process exits @preserve */
346
+ process.stdout?.on?.("error", () => { });
347
+ process.stderr?.on?.("error", () => { });
348
+ /* v8 ignore stop */
349
+ /* v8 ignore start -- global exception handlers: genuinely untestable in vitest; exercised by real daemon crashes @preserve */
350
+ let _uncaughtCount = 0;
351
+ let _tombstoneWritten = false;
352
+ let _lastKnownCause = null;
353
+ const CIRCUIT_BREAKER_WINDOW_MS = 60_000;
354
+ const CIRCUIT_BREAKER_MAX = 10;
355
+ process.on("uncaughtException", (error) => {
356
+ // EPIPE is normal for detached daemon processes — parent closed the pipe
357
+ if (error.code === "EPIPE")
358
+ return;
359
+ _uncaughtCount++;
360
+ _lastKnownCause = error;
361
+ setTimeout(() => { _uncaughtCount--; }, CIRCUIT_BREAKER_WINDOW_MS).unref();
362
+ _tombstoneWritten = true;
363
+ (0, daemon_tombstone_1.writeDaemonTombstone)("uncaughtException", error);
364
+ (0, runtime_1.emitNervesEvent)({
365
+ level: "error",
366
+ component: "daemon",
367
+ event: "daemon.uncaught_exception",
368
+ message: "uncaught exception in daemon process (continuing)",
369
+ meta: { error: error.message, stack: error.stack ?? null, uncaughtCount: _uncaughtCount },
370
+ });
371
+ // Circuit breaker: if too many exceptions in a short window, the process
372
+ // is in a bad state — exit so launchd/self-spawn can restart fresh.
373
+ if (_uncaughtCount >= CIRCUIT_BREAKER_MAX) {
374
+ (0, runtime_1.emitNervesEvent)({
375
+ level: "error",
376
+ component: "daemon",
377
+ event: "daemon.circuit_breaker_exit",
378
+ message: `daemon exiting: ${_uncaughtCount} uncaught exceptions in ${CIRCUIT_BREAKER_WINDOW_MS / 1000}s`,
379
+ meta: { uncaughtCount: _uncaughtCount },
380
+ });
381
+ setTimeout(() => process.exit(1), 5_000).unref();
382
+ void daemon.stop().then(() => process.exit(1));
383
+ }
384
+ });
385
+ process.on("unhandledRejection", (reason) => {
386
+ const error = reason instanceof Error ? reason : new Error(String(reason));
387
+ _lastKnownCause = error;
388
+ _tombstoneWritten = true;
389
+ (0, daemon_tombstone_1.writeDaemonTombstone)("unhandledRejection", error);
390
+ (0, runtime_1.emitNervesEvent)({
391
+ level: "error",
392
+ component: "daemon",
393
+ event: "daemon.unhandled_rejection",
394
+ message: "unhandled promise rejection in daemon process",
395
+ meta: { reason: error.message, stack: error.stack ?? null },
396
+ });
397
+ });
398
+ // Catch-all: write tombstone on any exit where we didn't already record the cause.
399
+ // process.on('exit') is synchronous-only — writeDaemonTombstone uses writeFileSync, so it works.
400
+ //
401
+ // Previously this skipped writing if `_gracefulShutdown` was true, which made
402
+ // SIGINT/SIGTERM-driven exits invisible in the death log. The signal handlers
403
+ // above now always write their own tombstone before exiting, so this catch-all
404
+ // only runs for exits the signal handlers didn't reach (e.g. process.exit
405
+ // called from somewhere unexpected).
406
+ process.on("exit", (code) => {
407
+ if (_tombstoneWritten)
408
+ return;
409
+ const reason = code === 0 ? "unexpectedCleanExit" : "unexpectedExit";
410
+ const error = _lastKnownCause ?? new Error(`daemon exited with code ${code} (no specific cause captured)`);
411
+ (0, daemon_tombstone_1.writeDaemonTombstone)(reason, error);
412
+ });
413
+ /* v8 ignore stop */
@@ -0,0 +1,137 @@
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.HEALTH_TRACKED_EVENTS = exports.DaemonHealthWriter = void 0;
37
+ exports.getDefaultHealthPath = getDefaultHealthPath;
38
+ exports.createHealthNervesSink = createHealthNervesSink;
39
+ exports.readHealth = readHealth;
40
+ const fs = __importStar(require("fs"));
41
+ const os = __importStar(require("os"));
42
+ const path = __importStar(require("path"));
43
+ const runtime_1 = require("../../nerves/runtime");
44
+ class DaemonHealthWriter {
45
+ healthPath;
46
+ constructor(healthPath) {
47
+ this.healthPath = healthPath;
48
+ }
49
+ writeHealth(state) {
50
+ try {
51
+ const dir = path.dirname(this.healthPath);
52
+ fs.mkdirSync(dir, { recursive: true });
53
+ const tmpPath = `${this.healthPath}.tmp.${process.pid}`;
54
+ fs.writeFileSync(tmpPath, JSON.stringify(state, null, 2) + "\n", "utf-8");
55
+ fs.renameSync(tmpPath, this.healthPath);
56
+ (0, runtime_1.emitNervesEvent)({
57
+ component: "daemon",
58
+ event: "daemon.health_written",
59
+ message: "daemon health file written",
60
+ meta: { path: this.healthPath, status: state.status },
61
+ });
62
+ }
63
+ catch {
64
+ // Best-effort: if we can't write, don't crash the daemon.
65
+ }
66
+ }
67
+ }
68
+ exports.DaemonHealthWriter = DaemonHealthWriter;
69
+ function getDefaultHealthPath() {
70
+ return path.join(os.homedir(), ".ouro-cli", "daemon-health.json");
71
+ }
72
+ /** Events that trigger a debounced health file write */
73
+ exports.HEALTH_TRACKED_EVENTS = new Set([
74
+ "daemon.habit_cron_verification_failed",
75
+ "daemon.habit_fire",
76
+ "daemon.agent_exit",
77
+ "daemon.agent_started",
78
+ "daemon.agent_restart_exhausted",
79
+ "daemon.agent_permanent_failure",
80
+ "daemon.agent_cooldown_recovery",
81
+ "daemon.bootstrap_degraded",
82
+ "daemon.safe_mode_entered",
83
+ "daemon.habit_scheduler_start",
84
+ ]);
85
+ /**
86
+ * Creates a nerves LogSink that triggers debounced health writes on relevant events.
87
+ * Components don't know about the health writer — they just emit events.
88
+ */
89
+ function createHealthNervesSink(writer, getState) {
90
+ let debounceTimer = null;
91
+ return (entry) => {
92
+ if (!exports.HEALTH_TRACKED_EVENTS.has(entry.event)) {
93
+ return;
94
+ }
95
+ // Debounce: max once per second
96
+ if (debounceTimer !== null) {
97
+ clearTimeout(debounceTimer);
98
+ }
99
+ debounceTimer = setTimeout(() => {
100
+ debounceTimer = null;
101
+ const state = getState();
102
+ writer.writeHealth(state);
103
+ }, 1000);
104
+ };
105
+ }
106
+ function readHealth(healthPath) {
107
+ try {
108
+ const raw = fs.readFileSync(healthPath, "utf-8");
109
+ const parsed = JSON.parse(raw);
110
+ if (typeof parsed.status !== "string" ||
111
+ typeof parsed.mode !== "string" ||
112
+ typeof parsed.pid !== "number" ||
113
+ typeof parsed.startedAt !== "string" ||
114
+ typeof parsed.uptimeSeconds !== "number" ||
115
+ !Array.isArray(parsed.degraded) ||
116
+ typeof parsed.agents !== "object" ||
117
+ parsed.agents === null ||
118
+ typeof parsed.habits !== "object" ||
119
+ parsed.habits === null) {
120
+ return null;
121
+ }
122
+ return {
123
+ status: parsed.status,
124
+ mode: parsed.mode,
125
+ pid: parsed.pid,
126
+ startedAt: parsed.startedAt,
127
+ uptimeSeconds: parsed.uptimeSeconds,
128
+ safeMode: parsed.safeMode,
129
+ degraded: parsed.degraded,
130
+ agents: parsed.agents,
131
+ habits: parsed.habits,
132
+ };
133
+ }
134
+ catch {
135
+ return null;
136
+ }
137
+ }