@ouro.bot/cli 0.1.0-alpha.5 → 0.1.0-alpha.500

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 (380) hide show
  1. package/README.md +226 -183
  2. package/SerpentGuide.ouro/agent.json +82 -0
  3. package/SerpentGuide.ouro/psyche/SOUL.md +25 -0
  4. package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/monty.md +2 -2
  5. package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-serpent.md +1 -1
  6. package/assets/ouroboros.png +0 -0
  7. package/changelog.json +3418 -0
  8. package/dist/arc/attention-types.js +8 -0
  9. package/dist/arc/cares.js +140 -0
  10. package/dist/arc/episodes.js +117 -0
  11. package/dist/arc/intentions.js +133 -0
  12. package/dist/arc/json-store.js +117 -0
  13. package/dist/arc/obligations.js +237 -0
  14. package/dist/arc/packets.js +193 -0
  15. package/dist/arc/presence.js +185 -0
  16. package/dist/arc/task-lifecycle.js +65 -0
  17. package/dist/heart/active-work.js +989 -0
  18. package/dist/heart/agent-entry.js +58 -3
  19. package/dist/heart/attachments/image-normalize.js +194 -0
  20. package/dist/heart/attachments/materialize.js +97 -0
  21. package/dist/heart/attachments/originals.js +88 -0
  22. package/dist/heart/attachments/render.js +29 -0
  23. package/dist/heart/attachments/sources/adapter.js +2 -0
  24. package/dist/heart/attachments/sources/bluebubbles.js +156 -0
  25. package/dist/heart/attachments/sources/cli-local-file.js +78 -0
  26. package/dist/heart/attachments/sources/index.js +16 -0
  27. package/dist/heart/attachments/store.js +103 -0
  28. package/dist/heart/attachments/types.js +93 -0
  29. package/dist/heart/auth/auth-flow.js +426 -0
  30. package/dist/heart/background-operations.js +281 -0
  31. package/dist/heart/bridges/manager.js +358 -0
  32. package/dist/heart/bridges/state-machine.js +135 -0
  33. package/dist/heart/bridges/store.js +123 -0
  34. package/dist/heart/bundle-state.js +168 -0
  35. package/dist/heart/commitments.js +111 -0
  36. package/dist/heart/config-registry.js +304 -0
  37. package/dist/heart/config.js +193 -130
  38. package/dist/heart/core.js +1010 -261
  39. package/dist/heart/cross-chat-delivery.js +131 -0
  40. package/dist/heart/daemon/agent-config-check.js +490 -0
  41. package/dist/heart/daemon/agent-discovery.js +157 -0
  42. package/dist/heart/daemon/agent-service.js +360 -0
  43. package/dist/heart/daemon/agentic-repair.js +216 -0
  44. package/dist/heart/daemon/bluebubbles-health-diagnostics.js +122 -0
  45. package/dist/heart/daemon/cadence.js +70 -0
  46. package/dist/heart/daemon/cli-defaults.js +640 -0
  47. package/dist/heart/daemon/cli-exec.js +7239 -0
  48. package/dist/heart/daemon/cli-help.js +493 -0
  49. package/dist/heart/daemon/cli-parse.js +1533 -0
  50. package/dist/heart/daemon/cli-render-doctor.js +57 -0
  51. package/dist/heart/daemon/cli-render.js +561 -0
  52. package/dist/heart/daemon/cli-types.js +8 -0
  53. package/dist/heart/daemon/connect-bay.js +323 -0
  54. package/dist/heart/daemon/daemon-cli.js +30 -697
  55. package/dist/heart/daemon/daemon-entry.js +359 -8
  56. package/dist/heart/daemon/daemon-health.js +141 -0
  57. package/dist/heart/daemon/daemon-runtime-sync.js +268 -0
  58. package/dist/heart/daemon/daemon-tombstone.js +236 -0
  59. package/dist/heart/daemon/daemon.js +813 -19
  60. package/dist/heart/daemon/dns-workflow.js +394 -0
  61. package/dist/heart/daemon/doctor-types.js +8 -0
  62. package/dist/heart/daemon/doctor.js +615 -0
  63. package/dist/heart/daemon/health-monitor.js +92 -1
  64. package/dist/heart/daemon/hooks/agent-config-v2.js +33 -0
  65. package/dist/heart/daemon/hooks/bundle-meta.js +206 -0
  66. package/dist/heart/daemon/http-health-probe.js +80 -0
  67. package/dist/heart/daemon/human-command-screens.js +234 -0
  68. package/dist/heart/daemon/human-readiness.js +114 -0
  69. package/dist/heart/daemon/inner-status.js +89 -0
  70. package/dist/heart/daemon/interactive-repair.js +394 -0
  71. package/dist/heart/daemon/launchd.js +171 -0
  72. package/dist/heart/daemon/log-tailer.js +82 -12
  73. package/dist/heart/daemon/logs-prune.js +110 -0
  74. package/dist/heart/daemon/message-router.js +17 -8
  75. package/dist/heart/daemon/os-cron-deps.js +134 -0
  76. package/dist/heart/daemon/ouro-bot-entry.js +4 -2
  77. package/dist/heart/daemon/ouro-entry.js +3 -1
  78. package/dist/heart/daemon/process-manager.js +215 -1
  79. package/dist/heart/daemon/provider-discovery.js +137 -0
  80. package/dist/heart/daemon/provider-ping-progress.js +83 -0
  81. package/dist/heart/daemon/pulse.js +475 -0
  82. package/dist/heart/daemon/readiness-repair.js +365 -0
  83. package/dist/heart/daemon/run-hooks.js +39 -0
  84. package/dist/heart/daemon/runtime-logging.js +67 -16
  85. package/dist/heart/daemon/runtime-metadata.js +191 -0
  86. package/dist/heart/daemon/runtime-mode.js +67 -0
  87. package/dist/heart/daemon/safe-mode.js +161 -0
  88. package/dist/heart/daemon/sense-manager.js +431 -0
  89. package/dist/heart/daemon/session-id-resolver.js +131 -0
  90. package/dist/heart/daemon/skill-management-installer.js +94 -0
  91. package/dist/heart/daemon/socket-client.js +307 -0
  92. package/dist/heart/daemon/stale-bundle-prune.js +96 -0
  93. package/dist/heart/daemon/startup-tui.js +264 -0
  94. package/dist/heart/daemon/task-scheduler.js +3 -25
  95. package/dist/heart/daemon/terminal-ui.js +499 -0
  96. package/dist/heart/daemon/thoughts.js +524 -0
  97. package/dist/heart/daemon/up-progress.js +366 -0
  98. package/dist/heart/daemon/vault-items.js +56 -0
  99. package/dist/heart/delegation.js +62 -0
  100. package/dist/heart/habits/habit-migration.js +189 -0
  101. package/dist/heart/habits/habit-parser.js +140 -0
  102. package/dist/heart/habits/habit-runtime-state.js +100 -0
  103. package/dist/heart/habits/habit-scheduler.js +372 -0
  104. package/dist/heart/{daemon → hatch}/hatch-animation.js +10 -3
  105. package/dist/heart/{daemon → hatch}/hatch-flow.js +54 -136
  106. package/dist/heart/{daemon → hatch}/hatch-specialist.js +3 -3
  107. package/dist/heart/hatch/specialist-orchestrator.js +129 -0
  108. package/dist/heart/hatch/specialist-prompt.js +102 -0
  109. package/dist/heart/hatch/specialist-tools.js +306 -0
  110. package/dist/heart/identity.js +274 -61
  111. package/dist/heart/kept-notes.js +357 -0
  112. package/dist/heart/kicks.js +2 -20
  113. package/dist/heart/machine-identity.js +161 -0
  114. package/dist/heart/mail-import-discovery.js +353 -0
  115. package/dist/heart/mcp/mcp-server.js +653 -0
  116. package/dist/heart/migrate-config.js +100 -0
  117. package/dist/heart/model-capabilities.js +59 -0
  118. package/dist/heart/outlook/outlook-http-hooks.js +66 -0
  119. package/dist/heart/outlook/outlook-http-response.js +7 -0
  120. package/dist/heart/outlook/outlook-http-routes.js +244 -0
  121. package/dist/heart/outlook/outlook-http-static.js +103 -0
  122. package/dist/heart/outlook/outlook-http-transport.js +116 -0
  123. package/dist/heart/outlook/outlook-http.js +99 -0
  124. package/dist/heart/outlook/outlook-read.js +31 -0
  125. package/dist/heart/outlook/outlook-types.js +27 -0
  126. package/dist/heart/outlook/outlook-view.js +195 -0
  127. package/dist/heart/outlook/readers/agent-machine.js +382 -0
  128. package/dist/heart/outlook/readers/continuity-readers.js +336 -0
  129. package/dist/heart/outlook/readers/mail.js +362 -0
  130. package/dist/heart/outlook/readers/runtime-readers.js +644 -0
  131. package/dist/heart/outlook/readers/sessions.js +232 -0
  132. package/dist/heart/outlook/readers/shared.js +111 -0
  133. package/dist/heart/platform.js +81 -0
  134. package/dist/heart/progress-story.js +42 -0
  135. package/dist/heart/provider-attempt.js +134 -0
  136. package/dist/heart/provider-binding-resolver.js +255 -0
  137. package/dist/heart/provider-credentials.js +424 -0
  138. package/dist/heart/provider-failover.js +301 -0
  139. package/dist/heart/provider-models.js +81 -0
  140. package/dist/heart/provider-ping.js +262 -0
  141. package/dist/heart/provider-state.js +216 -0
  142. package/dist/heart/provider-visibility.js +188 -0
  143. package/dist/heart/providers/anthropic-token.js +131 -0
  144. package/dist/heart/providers/anthropic.js +202 -50
  145. package/dist/heart/providers/azure.js +104 -13
  146. package/dist/heart/providers/error-classification.js +63 -0
  147. package/dist/heart/providers/github-copilot.js +145 -0
  148. package/dist/heart/providers/minimax-vlm.js +189 -0
  149. package/dist/heart/providers/minimax.js +29 -7
  150. package/dist/heart/providers/openai-codex.js +63 -39
  151. package/dist/heart/runtime-capability-check.js +170 -0
  152. package/dist/heart/runtime-credentials.js +260 -0
  153. package/dist/heart/sense-truth.js +68 -0
  154. package/dist/heart/session-activity.js +190 -0
  155. package/dist/heart/session-events.js +1089 -0
  156. package/dist/heart/session-playback-cli-main.js +5 -0
  157. package/dist/heart/session-playback-cli.js +36 -0
  158. package/dist/heart/session-playback.js +231 -0
  159. package/dist/heart/session-transcript.js +167 -0
  160. package/dist/heart/start-of-turn-packet.js +345 -0
  161. package/dist/heart/streaming.js +129 -34
  162. package/dist/heart/sync.js +332 -0
  163. package/dist/heart/target-resolution.js +127 -0
  164. package/dist/heart/tempo.js +93 -0
  165. package/dist/heart/temporal-view.js +41 -0
  166. package/dist/heart/tool-activity-callbacks.js +36 -0
  167. package/dist/heart/tool-description.js +135 -0
  168. package/dist/heart/tool-friction.js +55 -0
  169. package/dist/heart/tool-loop.js +200 -0
  170. package/dist/heart/turn-context.js +372 -0
  171. package/dist/heart/turn-coordinator.js +28 -0
  172. package/dist/heart/versioning/ouro-bot-global-installer.js +128 -0
  173. package/dist/heart/{daemon → versioning}/ouro-bot-wrapper.js +1 -1
  174. package/dist/heart/versioning/ouro-path-installer.js +425 -0
  175. package/dist/heart/{daemon → versioning}/ouro-uti.js +11 -2
  176. package/dist/heart/versioning/ouro-version-manager.js +295 -0
  177. package/dist/heart/versioning/staged-restart.js +146 -0
  178. package/dist/heart/versioning/update-checker.js +115 -0
  179. package/dist/heart/versioning/update-hooks.js +142 -0
  180. package/dist/heart/versioning/wrapper-publish-guard.js +86 -0
  181. package/dist/mailroom/attention.js +167 -0
  182. package/dist/mailroom/autonomy.js +209 -0
  183. package/dist/mailroom/blob-store.js +606 -0
  184. package/dist/mailroom/core.js +672 -0
  185. package/dist/mailroom/entry.js +160 -0
  186. package/dist/mailroom/file-store.js +426 -0
  187. package/dist/mailroom/mbox-import.js +382 -0
  188. package/dist/mailroom/outbound.js +380 -0
  189. package/dist/mailroom/policy.js +263 -0
  190. package/dist/mailroom/reader.js +219 -0
  191. package/dist/mailroom/search-cache.js +182 -0
  192. package/dist/mailroom/search-relevance.js +319 -0
  193. package/dist/mailroom/smtp-ingress.js +176 -0
  194. package/dist/mailroom/source-state.js +176 -0
  195. package/dist/mailroom/thread.js +109 -0
  196. package/dist/mailroom/travel-extract.js +89 -0
  197. package/dist/mind/bundle-manifest.js +77 -1
  198. package/dist/mind/context.js +173 -94
  199. package/dist/mind/diary-integrity.js +60 -0
  200. package/dist/mind/{memory.js → diary.js} +84 -96
  201. package/dist/mind/embedding-provider.js +60 -0
  202. package/dist/mind/file-state.js +179 -0
  203. package/dist/mind/first-impressions.js +16 -2
  204. package/dist/mind/friends/channel.js +73 -0
  205. package/dist/mind/friends/group-context.js +144 -0
  206. package/dist/mind/friends/resolver.js +54 -2
  207. package/dist/mind/friends/store-file.js +58 -3
  208. package/dist/mind/friends/trust-explanation.js +74 -0
  209. package/dist/mind/friends/types.js +10 -2
  210. package/dist/mind/journal-index.js +161 -0
  211. package/dist/mind/note-search.js +268 -0
  212. package/dist/mind/obligation-steering.js +221 -0
  213. package/dist/mind/pending.js +76 -9
  214. package/dist/mind/phrases.js +1 -0
  215. package/dist/mind/prompt-refresh.js +3 -2
  216. package/dist/mind/prompt.js +1144 -117
  217. package/dist/mind/provenance-trust.js +26 -0
  218. package/dist/mind/scrutiny.js +173 -0
  219. package/dist/mind/token-estimate.js +8 -12
  220. package/dist/nerves/cli-logging.js +22 -3
  221. package/dist/nerves/coverage/audit-rules.js +15 -6
  222. package/dist/nerves/coverage/audit.js +28 -2
  223. package/dist/nerves/coverage/cli.js +1 -1
  224. package/dist/nerves/coverage/contract.js +5 -5
  225. package/dist/nerves/coverage/file-completeness.js +101 -5
  226. package/dist/nerves/coverage/run-artifacts.js +1 -1
  227. package/dist/nerves/event-buffer.js +111 -0
  228. package/dist/nerves/index.js +224 -4
  229. package/dist/nerves/observation.js +20 -0
  230. package/dist/nerves/redact.js +79 -0
  231. package/dist/nerves/runtime.js +5 -1
  232. package/dist/outlook-ui/assets/index-BPr5vNuM.css +1 -0
  233. package/dist/outlook-ui/assets/index-Cm51CY9W.js +61 -0
  234. package/dist/outlook-ui/index.html +15 -0
  235. package/dist/repertoire/ado-client.js +17 -56
  236. package/dist/repertoire/ado-semantic.js +11 -10
  237. package/dist/repertoire/api-client.js +97 -0
  238. package/dist/repertoire/bitwarden-store.js +774 -0
  239. package/dist/repertoire/bundle-templates.js +72 -0
  240. package/dist/repertoire/bw-installer.js +180 -0
  241. package/dist/repertoire/coding/codex-jsonl.js +64 -0
  242. package/dist/repertoire/coding/context-pack.js +330 -0
  243. package/dist/repertoire/coding/feedback.js +301 -0
  244. package/dist/repertoire/coding/index.js +4 -1
  245. package/dist/repertoire/coding/manager.js +220 -13
  246. package/dist/repertoire/coding/spawner.js +58 -12
  247. package/dist/repertoire/coding/tools.js +209 -7
  248. package/dist/repertoire/commerce-errors.js +109 -0
  249. package/dist/repertoire/commerce-self-test.js +156 -0
  250. package/dist/repertoire/credential-access.js +111 -0
  251. package/dist/repertoire/data/ado-endpoints.json +188 -0
  252. package/dist/repertoire/duffel-client.js +185 -0
  253. package/dist/repertoire/github-client.js +14 -55
  254. package/dist/repertoire/graph-client.js +11 -52
  255. package/dist/repertoire/guardrails.js +396 -0
  256. package/dist/repertoire/mcp-client.js +255 -0
  257. package/dist/repertoire/mcp-manager.js +305 -0
  258. package/dist/repertoire/mcp-tools.js +63 -0
  259. package/dist/repertoire/shell-sessions.js +133 -0
  260. package/dist/repertoire/skills.js +15 -24
  261. package/dist/repertoire/stripe-client.js +131 -0
  262. package/dist/repertoire/tasks/board.js +43 -5
  263. package/dist/repertoire/tasks/fix.js +182 -0
  264. package/dist/repertoire/tasks/index.js +39 -13
  265. package/dist/repertoire/tasks/lifecycle.js +2 -2
  266. package/dist/repertoire/tasks/parser.js +3 -2
  267. package/dist/repertoire/tasks/scanner.js +194 -37
  268. package/dist/repertoire/tasks/transitions.js +16 -79
  269. package/dist/repertoire/tool-results.js +29 -0
  270. package/dist/repertoire/tools-attachments.js +317 -0
  271. package/dist/repertoire/tools-base.js +49 -707
  272. package/dist/repertoire/tools-bluebubbles.js +94 -0
  273. package/dist/repertoire/tools-bridge.js +141 -0
  274. package/dist/repertoire/tools-bundle.js +984 -0
  275. package/dist/repertoire/tools-config.js +185 -0
  276. package/dist/repertoire/tools-continuity.js +248 -0
  277. package/dist/repertoire/tools-credential.js +381 -0
  278. package/dist/repertoire/tools-files.js +342 -0
  279. package/dist/repertoire/tools-flight.js +224 -0
  280. package/dist/repertoire/tools-flow.js +105 -0
  281. package/dist/repertoire/tools-github.js +1 -7
  282. package/dist/repertoire/tools-mail.js +1377 -0
  283. package/dist/repertoire/tools-notes.js +376 -0
  284. package/dist/repertoire/tools-session.js +749 -0
  285. package/dist/repertoire/tools-shell.js +120 -0
  286. package/dist/repertoire/tools-stripe.js +180 -0
  287. package/dist/repertoire/tools-surface.js +243 -0
  288. package/dist/repertoire/tools-teams.js +64 -61
  289. package/dist/repertoire/tools-travel.js +125 -0
  290. package/dist/repertoire/tools-trip.js +356 -0
  291. package/dist/repertoire/tools-user-profile.js +144 -0
  292. package/dist/repertoire/tools-vault.js +40 -0
  293. package/dist/repertoire/tools.js +149 -98
  294. package/dist/repertoire/travel-api-client.js +360 -0
  295. package/dist/repertoire/user-profile.js +131 -0
  296. package/dist/repertoire/vault-setup.js +246 -0
  297. package/dist/repertoire/vault-unlock.js +561 -0
  298. package/dist/scripts/claude-code-hook.js +41 -0
  299. package/dist/scripts/claude-code-stop-hook.js +47 -0
  300. package/dist/senses/attention-queue.js +116 -0
  301. package/dist/senses/bluebubbles/attachment-cache.js +53 -0
  302. package/dist/senses/bluebubbles/attachment-download.js +137 -0
  303. package/dist/senses/bluebubbles/client.js +685 -0
  304. package/dist/senses/bluebubbles/entry.js +73 -0
  305. package/dist/senses/bluebubbles/inbound-log.js +126 -0
  306. package/dist/senses/bluebubbles/index.js +1881 -0
  307. package/dist/senses/bluebubbles/media.js +389 -0
  308. package/dist/senses/bluebubbles/model.js +282 -0
  309. package/dist/senses/bluebubbles/mutation-log.js +116 -0
  310. package/dist/senses/bluebubbles/processed-log.js +111 -0
  311. package/dist/senses/bluebubbles/replay.js +129 -0
  312. package/dist/senses/bluebubbles/runtime-state.js +109 -0
  313. package/dist/senses/bluebubbles/session-cleanup.js +72 -0
  314. package/dist/senses/cli/bracketed-paste.js +82 -0
  315. package/dist/senses/cli/image-paste.js +287 -0
  316. package/dist/senses/cli/image-ref-navigation.js +75 -0
  317. package/dist/senses/cli/ink-app.js +156 -0
  318. package/dist/senses/cli/inline-diff.js +64 -0
  319. package/dist/senses/cli/input-keys.js +174 -0
  320. package/dist/senses/cli/kill-ring.js +86 -0
  321. package/dist/senses/cli/message-list.js +51 -0
  322. package/dist/senses/cli/ouro-tui.js +605 -0
  323. package/dist/senses/cli/spinner-imperative.js +135 -0
  324. package/dist/senses/cli/spinner.js +101 -0
  325. package/dist/senses/cli/status-line.js +60 -0
  326. package/dist/senses/cli/streaming-markdown.js +526 -0
  327. package/dist/senses/cli/tool-display.js +83 -0
  328. package/dist/senses/cli/tool-render.js +85 -0
  329. package/dist/senses/cli/tui-store.js +240 -0
  330. package/dist/senses/cli/virtual-list.js +35 -0
  331. package/dist/senses/cli-entry.js +60 -8
  332. package/dist/senses/cli-layout.js +187 -0
  333. package/dist/senses/cli.js +768 -264
  334. package/dist/senses/commands.js +66 -3
  335. package/dist/senses/continuity.js +94 -0
  336. package/dist/senses/habit-turn-message.js +108 -0
  337. package/dist/senses/inner-dialog-worker.js +199 -16
  338. package/dist/senses/inner-dialog.js +640 -91
  339. package/dist/senses/mail-entry.js +66 -0
  340. package/dist/senses/mail.js +379 -0
  341. package/dist/senses/pipeline.js +665 -0
  342. package/dist/senses/proactive-content-guard.js +51 -0
  343. package/dist/senses/shared-turn.js +248 -0
  344. package/dist/senses/surface-tool.js +68 -0
  345. package/dist/senses/teams-entry.js +60 -8
  346. package/dist/senses/teams.js +844 -197
  347. package/dist/senses/trust-gate.js +207 -2
  348. package/dist/trips/core.js +138 -0
  349. package/dist/trips/store.js +146 -0
  350. package/package.json +47 -6
  351. package/skills/agent-commerce.md +106 -0
  352. package/skills/browser-navigation.md +117 -0
  353. package/skills/commerce-setup-guide.md +116 -0
  354. package/skills/commerce-setup.md +84 -0
  355. package/skills/configure-dev-tools.md +101 -0
  356. package/skills/travel-planning.md +138 -0
  357. package/AdoptionSpecialist.ouro/agent.json +0 -20
  358. package/AdoptionSpecialist.ouro/psyche/SOUL.md +0 -22
  359. package/dist/heart/daemon/specialist-orchestrator.js +0 -160
  360. package/dist/heart/daemon/specialist-prompt.js +0 -40
  361. package/dist/heart/daemon/specialist-session.js +0 -142
  362. package/dist/heart/daemon/specialist-tools.js +0 -128
  363. package/dist/heart/daemon/subagent-installer.js +0 -125
  364. package/dist/inner-worker-entry.js +0 -4
  365. package/dist/mind/associative-recall.js +0 -197
  366. package/subagents/README.md +0 -73
  367. package/subagents/work-doer.md +0 -233
  368. package/subagents/work-merger.md +0 -624
  369. package/subagents/work-planner.md +0 -373
  370. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/basilisk.md +0 -0
  371. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jafar.md +0 -0
  372. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jormungandr.md +0 -0
  373. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/kaa.md +0 -0
  374. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/medusa.md +0 -0
  375. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/nagini.md +0 -0
  376. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/ouroboros.md +0 -0
  377. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/python.md +0 -0
  378. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/quetzalcoatl.md +0 -0
  379. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/sir-hiss.md +0 -0
  380. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-snake.md +0 -0
@@ -0,0 +1,135 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.createBridgeState = createBridgeState;
4
+ exports.bridgeStateLabel = bridgeStateLabel;
5
+ exports.activateBridge = activateBridge;
6
+ exports.beginBridgeProcessing = beginBridgeProcessing;
7
+ exports.queueBridgeFollowUp = queueBridgeFollowUp;
8
+ exports.advanceBridgeAfterTurn = advanceBridgeAfterTurn;
9
+ exports.suspendBridge = suspendBridge;
10
+ exports.completeBridge = completeBridge;
11
+ exports.cancelBridge = cancelBridge;
12
+ exports.reconcileBridgeState = reconcileBridgeState;
13
+ const runtime_1 = require("../../nerves/runtime");
14
+ function transition(state, next, action) {
15
+ (0, runtime_1.emitNervesEvent)({
16
+ component: "engine",
17
+ event: "engine.bridge_state_transition",
18
+ message: "bridge state transitioned",
19
+ meta: {
20
+ action,
21
+ from: bridgeStateLabel(state),
22
+ to: bridgeStateLabel(next),
23
+ },
24
+ });
25
+ return next;
26
+ }
27
+ function assertNonTerminal(state, action) {
28
+ if (state.lifecycle === "completed" || state.lifecycle === "cancelled") {
29
+ throw new Error(`cannot ${action} a terminal bridge`);
30
+ }
31
+ }
32
+ function createBridgeState() {
33
+ return {
34
+ lifecycle: "forming",
35
+ runtime: "idle",
36
+ };
37
+ }
38
+ function bridgeStateLabel(state) {
39
+ switch (state.lifecycle) {
40
+ case "forming":
41
+ return "forming";
42
+ case "suspended":
43
+ return "suspended";
44
+ case "completed":
45
+ return "completed";
46
+ case "cancelled":
47
+ return "cancelled";
48
+ case "active":
49
+ if (state.runtime === "processing")
50
+ return "active-processing";
51
+ if (state.runtime === "awaiting-follow-up")
52
+ return "awaiting-follow-up";
53
+ return "active-idle";
54
+ }
55
+ }
56
+ function activateBridge(state) {
57
+ assertNonTerminal(state, "activate");
58
+ if (state.lifecycle !== "forming" && state.lifecycle !== "suspended") {
59
+ throw new Error("cannot activate bridge from current state");
60
+ }
61
+ return transition(state, { lifecycle: "active", runtime: "idle" }, "activate");
62
+ }
63
+ function beginBridgeProcessing(state) {
64
+ assertNonTerminal(state, "process");
65
+ if (state.lifecycle !== "active" || state.runtime !== "idle") {
66
+ throw new Error("cannot process bridge from current state");
67
+ }
68
+ return transition(state, { lifecycle: "active", runtime: "processing" }, "begin-processing");
69
+ }
70
+ function queueBridgeFollowUp(state) {
71
+ assertNonTerminal(state, "queue");
72
+ if (state.lifecycle !== "active") {
73
+ throw new Error("cannot queue follow-up for non-active bridge");
74
+ }
75
+ if (state.runtime === "processing") {
76
+ return transition(state, { lifecycle: "active", runtime: "awaiting-follow-up" }, "queue-follow-up");
77
+ }
78
+ if (state.runtime === "awaiting-follow-up") {
79
+ return state;
80
+ }
81
+ throw new Error("cannot queue follow-up when bridge is not processing");
82
+ }
83
+ function advanceBridgeAfterTurn(state) {
84
+ assertNonTerminal(state, "advance");
85
+ if (state.lifecycle !== "active") {
86
+ throw new Error("cannot advance non-active bridge");
87
+ }
88
+ if (state.runtime === "processing") {
89
+ return transition(state, { lifecycle: "active", runtime: "idle" }, "finish-processing");
90
+ }
91
+ if (state.runtime === "awaiting-follow-up") {
92
+ return transition(state, { lifecycle: "active", runtime: "processing" }, "resume-follow-up");
93
+ }
94
+ throw new Error("cannot advance an idle bridge");
95
+ }
96
+ function suspendBridge(state) {
97
+ assertNonTerminal(state, "suspend");
98
+ if ((state.lifecycle !== "forming" && state.lifecycle !== "active") || state.runtime !== "idle") {
99
+ throw new Error("cannot suspend bridge from current state");
100
+ }
101
+ return transition(state, { lifecycle: "suspended", runtime: "idle" }, "suspend");
102
+ }
103
+ function completeBridge(state) {
104
+ assertNonTerminal(state, "complete");
105
+ if (state.runtime !== "idle") {
106
+ throw new Error("cannot complete a bridge mid-turn");
107
+ }
108
+ return transition(state, { lifecycle: "completed", runtime: "idle" }, "complete");
109
+ }
110
+ function cancelBridge(state) {
111
+ assertNonTerminal(state, "cancel");
112
+ if (state.runtime !== "idle") {
113
+ throw new Error("cannot cancel a bridge mid-turn");
114
+ }
115
+ return transition(state, { lifecycle: "cancelled", runtime: "idle" }, "cancel");
116
+ }
117
+ function reconcileBridgeState(state, input) {
118
+ if (state.lifecycle === "completed" || state.lifecycle === "cancelled") {
119
+ return state;
120
+ }
121
+ if (state.runtime !== "idle") {
122
+ return state;
123
+ }
124
+ const hasLiveSignal = input.hasAttachedSessionActivity || input.hasLiveTask || input.currentSessionAttached;
125
+ if (state.lifecycle === "suspended") {
126
+ return hasLiveSignal ? activateBridge(state) : state;
127
+ }
128
+ if (state.lifecycle === "forming") {
129
+ return hasLiveSignal ? activateBridge(state) : suspendBridge(state);
130
+ }
131
+ if (state.lifecycle === "active") {
132
+ return hasLiveSignal ? state : suspendBridge(state);
133
+ }
134
+ return state;
135
+ }
@@ -0,0 +1,123 @@
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.getBridgeStateRoot = getBridgeStateRoot;
37
+ exports.createBridgeStore = createBridgeStore;
38
+ const fs = __importStar(require("fs"));
39
+ const path = __importStar(require("path"));
40
+ const identity_1 = require("../identity");
41
+ const runtime_1 = require("../../nerves/runtime");
42
+ function sessionIdentityMatches(session, candidate) {
43
+ return (session.friendId === candidate.friendId
44
+ && session.channel === candidate.channel
45
+ && session.key === candidate.key);
46
+ }
47
+ function bridgeFilePath(rootDir, id) {
48
+ return path.join(rootDir, `${id}.json`);
49
+ }
50
+ function getBridgeStateRoot() {
51
+ return path.join((0, identity_1.getAgentStateRoot)(), "bridges");
52
+ }
53
+ function createBridgeStore(options = {}) {
54
+ const rootDir = options.rootDir ?? getBridgeStateRoot();
55
+ function ensureRoot() {
56
+ fs.mkdirSync(rootDir, { recursive: true });
57
+ }
58
+ return {
59
+ save(bridge) {
60
+ ensureRoot();
61
+ fs.writeFileSync(bridgeFilePath(rootDir, bridge.id), JSON.stringify(bridge, null, 2), "utf-8");
62
+ (0, runtime_1.emitNervesEvent)({
63
+ component: "engine",
64
+ event: "engine.bridge_store_save",
65
+ message: "saved bridge record",
66
+ meta: {
67
+ bridgeId: bridge.id,
68
+ rootDir,
69
+ },
70
+ });
71
+ return bridge;
72
+ },
73
+ get(id) {
74
+ const filePath = bridgeFilePath(rootDir, id);
75
+ try {
76
+ const raw = fs.readFileSync(filePath, "utf-8");
77
+ return JSON.parse(raw);
78
+ }
79
+ catch {
80
+ return null;
81
+ }
82
+ },
83
+ list() {
84
+ ensureRoot();
85
+ const files = fs.readdirSync(rootDir).filter((entry) => entry.endsWith(".json")).sort();
86
+ const bridges = files
87
+ .map((fileName) => {
88
+ try {
89
+ return JSON.parse(fs.readFileSync(path.join(rootDir, fileName), "utf-8"));
90
+ }
91
+ catch {
92
+ return null;
93
+ }
94
+ })
95
+ .filter((bridge) => bridge !== null);
96
+ (0, runtime_1.emitNervesEvent)({
97
+ component: "engine",
98
+ event: "engine.bridge_store_list",
99
+ message: "listed bridge records",
100
+ meta: {
101
+ rootDir,
102
+ count: bridges.length,
103
+ },
104
+ });
105
+ return bridges;
106
+ },
107
+ findBySession(session) {
108
+ const matches = this.list().filter((bridge) => bridge.attachedSessions.some((candidate) => sessionIdentityMatches(session, candidate)));
109
+ (0, runtime_1.emitNervesEvent)({
110
+ component: "engine",
111
+ event: "engine.bridge_store_find_by_session",
112
+ message: "located bridges for canonical session",
113
+ meta: {
114
+ friendId: session.friendId,
115
+ channel: session.channel,
116
+ key: session.key,
117
+ count: matches.length,
118
+ },
119
+ });
120
+ return matches;
121
+ },
122
+ };
123
+ }
@@ -0,0 +1,168 @@
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.detectBundleState = detectBundleState;
37
+ exports.renderBundleStateHint = renderBundleStateHint;
38
+ /**
39
+ * Bundle state detection: inspects an agent bundle's git + pending-sync
40
+ * state and returns a structured list of issues the agent needs to remediate
41
+ * via its `bundle_*` tools.
42
+ *
43
+ * Previously the sync pipeline surfaced a single free-form `syncFailure`
44
+ * string that was good for humans but hard for the agent to act on. The
45
+ * new `BundleStateIssue` enum lets the agent pattern-match on discrete
46
+ * cases and pick the correct remediation tool (shipping in PR 6).
47
+ *
48
+ * Detection never throws: every git call is wrapped in try/catch so a
49
+ * broken bundle or a missing git binary degrades to an empty array or a
50
+ * `not_a_git_repo` signal rather than exploding the turn pipeline.
51
+ *
52
+ * The classification returned here is detected from bundle state +
53
+ * the `classification` field of `state/pending-sync.json` (written by
54
+ * `postTurnPush` in sync.ts when it writes the pending-sync record).
55
+ *
56
+ * - `not_a_git_repo`: `.git` directory missing.
57
+ * - `no_remote_configured`: `git remote` returns empty.
58
+ * - `first_commit_never_happened`: `.git` exists but `git rev-parse HEAD`
59
+ * fails (fresh `git init` with nothing committed).
60
+ * - `pending_sync_exists`: `state/pending-sync.json` exists — the agent
61
+ * should inspect it and clear the pending state.
62
+ * - `remote_push_failed`: pending-sync classified as `push_rejected` —
63
+ * the remote advanced and a simple retry won't work.
64
+ * - `pull_rebase_conflict`: pending-sync classified as
65
+ * `pull_rebase_conflict` — the rebase left conflicted files the agent
66
+ * has to walk the user through resolving.
67
+ */
68
+ const child_process_1 = require("child_process");
69
+ const fs = __importStar(require("fs"));
70
+ const path = __importStar(require("path"));
71
+ const runtime_1 = require("../nerves/runtime");
72
+ function detectBundleState(agentRoot, deps = {}) {
73
+ const exec = deps.execFileSync ?? child_process_1.execFileSync;
74
+ const exists = deps.existsSync ?? fs.existsSync;
75
+ const readFile = deps.readFileSync ?? ((p, enc) => fs.readFileSync(p, enc));
76
+ const issues = [];
77
+ (0, runtime_1.emitNervesEvent)({
78
+ component: "heart",
79
+ event: "heart.bundle_state_detect_start",
80
+ message: "detecting bundle state",
81
+ meta: { agentRoot },
82
+ });
83
+ const gitDir = path.join(agentRoot, ".git");
84
+ const isGitRepo = exists(gitDir);
85
+ if (!isGitRepo) {
86
+ issues.push("not_a_git_repo");
87
+ }
88
+ else {
89
+ // Check remote presence
90
+ try {
91
+ const remoteOutput = exec("git", ["remote"], {
92
+ cwd: agentRoot,
93
+ stdio: "pipe",
94
+ timeout: 5000,
95
+ }).toString().trim();
96
+ if (remoteOutput.length === 0) {
97
+ issues.push("no_remote_configured");
98
+ }
99
+ }
100
+ catch {
101
+ // Git missing or broken — treat as no_remote_configured so the
102
+ // agent surfaces a "fix git" signal rather than a silent pass.
103
+ issues.push("no_remote_configured");
104
+ }
105
+ // Check for initial commit
106
+ try {
107
+ exec("git", ["rev-parse", "HEAD"], {
108
+ cwd: agentRoot,
109
+ stdio: "pipe",
110
+ timeout: 5000,
111
+ });
112
+ }
113
+ catch {
114
+ issues.push("first_commit_never_happened");
115
+ }
116
+ }
117
+ // Pending-sync file — independent of git state (could exist on a bundle
118
+ // that was a repo and then had its .git deleted). When present, also
119
+ // inspect the classification field to surface the specific failure
120
+ // mode as an additional issue.
121
+ const pendingSyncPath = path.join(agentRoot, "state", "pending-sync.json");
122
+ if (exists(pendingSyncPath)) {
123
+ issues.push("pending_sync_exists");
124
+ try {
125
+ const raw = readFile(pendingSyncPath, "utf-8");
126
+ const parsed = JSON.parse(raw);
127
+ if (parsed.classification === "push_rejected") {
128
+ issues.push("remote_push_failed");
129
+ }
130
+ else if (parsed.classification === "pull_rebase_conflict") {
131
+ issues.push("pull_rebase_conflict");
132
+ }
133
+ }
134
+ catch {
135
+ // Malformed or unreadable pending-sync.json — pending_sync_exists
136
+ // alone is enough signal for the agent to investigate.
137
+ }
138
+ }
139
+ (0, runtime_1.emitNervesEvent)({
140
+ component: "heart",
141
+ event: "heart.bundle_state_detect_end",
142
+ message: `bundle state detected: ${issues.length} issue(s)`,
143
+ meta: { agentRoot, issues },
144
+ });
145
+ return issues;
146
+ }
147
+ /**
148
+ * First-person remediation hint text for the start-of-turn packet. Reads
149
+ * as the agent's own voice per the declarative first-person note rule.
150
+ * Returns empty string when there are no issues (so the packet renderer
151
+ * can skip the section entirely).
152
+ */
153
+ function renderBundleStateHint(issues) {
154
+ if (issues.length === 0)
155
+ return "";
156
+ const labels = {
157
+ not_a_git_repo: "not a git repo",
158
+ no_remote_configured: "no remote configured",
159
+ first_commit_never_happened: "first commit never happened",
160
+ pending_sync_exists: "pending sync from a prior turn",
161
+ remote_push_failed: "remote push rejected (remote advanced)",
162
+ pull_rebase_conflict: "pull-rebase left conflicts i need to walk the user through",
163
+ };
164
+ const list = issues.map((issue) => labels[issue]).join(", ");
165
+ return (`my bundle has unresolved git state: ${list}. ` +
166
+ `i have the bundle_* tools available to fix this — i should run ` +
167
+ `bundle_check_sync_status first, then use the appropriate remediation tool.`);
168
+ }
@@ -0,0 +1,111 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.deriveCommitments = deriveCommitments;
4
+ exports.formatCommitments = formatCommitments;
5
+ const obligations_1 = require("../arc/obligations");
6
+ const runtime_1 = require("../nerves/runtime");
7
+ function describeActiveObligation(obligation) {
8
+ if (obligation.status === "pending") {
9
+ return `i owe ${obligation.origin.friendId}: ${obligation.content}`;
10
+ }
11
+ const surface = obligation.currentSurface?.label;
12
+ const statusText = obligation.status.replaceAll("_", " ");
13
+ if (surface) {
14
+ return `i owe ${obligation.origin.friendId}: ${obligation.content} (${statusText} in ${surface})`;
15
+ }
16
+ return `i owe ${obligation.origin.friendId}: ${obligation.content} (${statusText})`;
17
+ }
18
+ function deriveCommitments(activeWorkFrame, innerJob, pendingObligations) {
19
+ const committedTo = [];
20
+ const completionCriteria = [];
21
+ const safeToIgnore = [];
22
+ // Persistent obligations from the obligation store
23
+ // Sort by status priority: investigating/waiting/updating before pending
24
+ if (pendingObligations && pendingObligations.length > 0) {
25
+ const sorted = [...pendingObligations].sort((a, b) => {
26
+ const advancedA = a.status !== "pending" && a.status !== "fulfilled" ? 0 : 1;
27
+ const advancedB = b.status !== "pending" && b.status !== "fulfilled" ? 0 : 1;
28
+ return advancedA - advancedB;
29
+ });
30
+ let hasAdvancedObligation = false;
31
+ for (const ob of sorted) {
32
+ if (!(0, obligations_1.isOpenObligationStatus)(ob.status))
33
+ continue;
34
+ committedTo.push(describeActiveObligation(ob));
35
+ if (ob.status !== "pending")
36
+ hasAdvancedObligation = true;
37
+ }
38
+ completionCriteria.push("fulfill my outstanding obligations");
39
+ if (hasAdvancedObligation) {
40
+ completionCriteria.push("close my active obligation loops");
41
+ }
42
+ }
43
+ // Inner job
44
+ if (innerJob.status === "queued" || innerJob.status === "running") {
45
+ const contentSuffix = innerJob.content ? ` -- ${innerJob.content.slice(0, 60)}` : "";
46
+ committedTo.push(`i'm thinking through something privately${contentSuffix}`);
47
+ }
48
+ else if (innerJob.status === "surfaced") {
49
+ committedTo.push("i finished thinking about something and need to bring it back");
50
+ }
51
+ // mustResolveBeforeHandoff
52
+ if (activeWorkFrame.mustResolveBeforeHandoff) {
53
+ committedTo.push("i need to finish what i started before moving on");
54
+ completionCriteria.push("resolve the current thread before moving on");
55
+ }
56
+ // Bridges
57
+ for (const bridge of activeWorkFrame.bridges) {
58
+ committedTo.push(`i have shared work: ${bridge.summary || bridge.objective}`);
59
+ }
60
+ if (activeWorkFrame.bridges.length > 0) {
61
+ completionCriteria.push("keep shared work aligned across sessions");
62
+ }
63
+ // Tasks
64
+ for (const taskName of activeWorkFrame.taskPressure?.liveTaskNames ?? []) {
65
+ committedTo.push(`i'm tracking: ${taskName}`);
66
+ }
67
+ // Obligation completion criteria
68
+ if (innerJob.obligationStatus === "pending") {
69
+ const name = innerJob.origin?.friendName ?? innerJob.origin?.friendId ?? "them";
70
+ completionCriteria.push(`bring my answer back to ${name}`);
71
+ }
72
+ // Default completion criteria
73
+ if (completionCriteria.length === 0) {
74
+ completionCriteria.push("just be present in this conversation");
75
+ }
76
+ // Safe to ignore
77
+ if (innerJob.status === "idle" && !(activeWorkFrame.inner?.hasPending)) {
78
+ safeToIgnore.push("no private thinking in progress");
79
+ }
80
+ if (activeWorkFrame.bridges.length === 0) {
81
+ safeToIgnore.push("no shared work to coordinate");
82
+ }
83
+ if ((activeWorkFrame.taskPressure?.liveTaskNames ?? []).length === 0) {
84
+ safeToIgnore.push("no active tasks to track");
85
+ }
86
+ (0, runtime_1.emitNervesEvent)({
87
+ component: "engine",
88
+ event: "engine.commitments_derive",
89
+ message: "derived commitments frame",
90
+ meta: { committedCount: committedTo.length, criteriaCount: completionCriteria.length },
91
+ });
92
+ return { committedTo, completionCriteria, safeToIgnore };
93
+ }
94
+ function formatCommitments(commitments) {
95
+ const sections = [];
96
+ if (commitments.committedTo.length === 0) {
97
+ sections.push("i'm not holding anything specific right now. i'm free to be present.");
98
+ }
99
+ else {
100
+ sections.push("## what i'm holding right now");
101
+ sections.push("");
102
+ sections.push(commitments.committedTo.map((c) => `- ${c}`).join("\n"));
103
+ }
104
+ sections.push("");
105
+ sections.push("## what \"done\" looks like");
106
+ sections.push(commitments.completionCriteria.map((c) => `- ${c}`).join("\n"));
107
+ sections.push("");
108
+ sections.push("## what i can let go of");
109
+ sections.push(commitments.safeToIgnore.map((c) => `- ${c}`).join("\n"));
110
+ return sections.join("\n");
111
+ }