@ouro.bot/cli 0.1.0-alpha.55 → 0.1.0-alpha.551
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.
- package/README.md +133 -19
- package/RepairGuide.ouro/agent.json +5 -0
- package/RepairGuide.ouro/psyche/IDENTITY.md +19 -0
- package/RepairGuide.ouro/psyche/SOUL.md +55 -0
- package/RepairGuide.ouro/skills/diagnose-bootstrap-drift.md +54 -0
- package/RepairGuide.ouro/skills/diagnose-broken-remote.md +63 -0
- package/RepairGuide.ouro/skills/diagnose-stacked-typed-issues.md +35 -0
- package/RepairGuide.ouro/skills/diagnose-sync-blocked.md +54 -0
- package/RepairGuide.ouro/skills/diagnose-vault-expired.md +60 -0
- package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/agent.json +4 -2
- package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/SOUL.md +2 -2
- package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-serpent.md +1 -1
- package/changelog.json +3561 -0
- package/dist/arc/attention-types.js +8 -0
- package/dist/arc/cares.js +140 -0
- package/dist/arc/episodes.js +117 -0
- package/dist/arc/intentions.js +133 -0
- package/dist/arc/json-store.js +117 -0
- package/dist/arc/obligations.js +237 -0
- package/dist/arc/packets.js +193 -0
- package/dist/arc/presence.js +185 -0
- package/dist/arc/task-lifecycle.js +65 -0
- package/dist/heart/active-work.js +837 -26
- package/dist/heart/agent-entry.js +58 -3
- package/dist/heart/attachments/image-normalize.js +194 -0
- package/dist/heart/attachments/materialize.js +97 -0
- package/dist/heart/attachments/originals.js +88 -0
- package/dist/heart/attachments/render.js +29 -0
- package/dist/heart/attachments/sources/adapter.js +2 -0
- package/dist/heart/attachments/sources/bluebubbles.js +156 -0
- package/dist/heart/attachments/sources/cli-local-file.js +78 -0
- package/dist/heart/attachments/sources/index.js +16 -0
- package/dist/heart/attachments/store.js +103 -0
- package/dist/heart/attachments/types.js +93 -0
- package/dist/heart/auth/auth-flow.js +479 -0
- package/dist/heart/background-operations.js +281 -0
- package/dist/heart/bundle-state.js +168 -0
- package/dist/heart/commitments.js +111 -0
- package/dist/heart/config-registry.js +304 -0
- package/dist/heart/config.js +114 -118
- package/dist/heart/core.js +925 -246
- package/dist/heart/cross-chat-delivery.js +3 -18
- package/dist/heart/daemon/agent-config-check.js +512 -0
- package/dist/heart/daemon/agent-discovery.js +102 -3
- package/dist/heart/daemon/agent-service.js +522 -0
- package/dist/heart/daemon/agentic-repair.js +554 -0
- package/dist/heart/daemon/bluebubbles-health-diagnostics.js +122 -0
- package/dist/heart/daemon/boot-sync-probe.js +197 -0
- package/dist/heart/daemon/cadence.js +70 -0
- package/dist/heart/daemon/cli-defaults.js +665 -0
- package/dist/heart/daemon/cli-exec.js +7565 -0
- package/dist/heart/daemon/cli-help.js +498 -0
- package/dist/heart/daemon/cli-parse.js +1590 -0
- package/dist/heart/daemon/cli-render-doctor.js +57 -0
- package/dist/heart/daemon/cli-render.js +775 -0
- package/dist/heart/daemon/cli-types.js +8 -0
- package/dist/heart/daemon/connect-bay.js +323 -0
- package/dist/heart/daemon/daemon-cli.js +29 -1672
- package/dist/heart/daemon/daemon-entry.js +417 -2
- package/dist/heart/daemon/daemon-health.js +183 -0
- package/dist/heart/daemon/daemon-rollup.js +58 -0
- package/dist/heart/daemon/daemon-runtime-sync.js +87 -13
- package/dist/heart/daemon/daemon-tombstone.js +236 -0
- package/dist/heart/daemon/daemon.js +796 -71
- package/dist/heart/daemon/dns-workflow.js +394 -0
- package/dist/heart/daemon/doctor-types.js +8 -0
- package/dist/heart/daemon/doctor.js +844 -0
- package/dist/heart/daemon/drift-detection.js +146 -0
- package/dist/heart/daemon/health-monitor.js +122 -1
- package/dist/heart/daemon/hooks/agent-config-v2.js +33 -0
- package/dist/heart/daemon/hooks/bundle-meta.js +115 -1
- package/dist/heart/daemon/http-health-probe.js +80 -0
- package/dist/heart/daemon/human-command-screens.js +234 -0
- package/dist/heart/daemon/human-readiness.js +114 -0
- package/dist/heart/daemon/inner-status.js +102 -0
- package/dist/heart/daemon/interactive-repair.js +394 -0
- package/dist/heart/daemon/launchd.js +37 -8
- package/dist/heart/daemon/log-tailer.js +82 -12
- package/dist/heart/daemon/logs-prune.js +110 -0
- package/dist/heart/daemon/mcp-canary.js +297 -0
- package/dist/heart/daemon/message-router.js +2 -2
- package/dist/heart/daemon/os-cron-deps.js +135 -0
- package/dist/heart/daemon/os-cron.js +14 -12
- package/dist/heart/daemon/ouro-bot-entry.js +4 -2
- package/dist/heart/daemon/ouro-entry.js +3 -1
- package/dist/heart/daemon/process-manager.js +375 -33
- package/dist/heart/daemon/provider-discovery.js +137 -0
- package/dist/heart/daemon/provider-ping-progress.js +83 -0
- package/dist/heart/daemon/pulse.js +475 -0
- package/dist/heart/daemon/readiness-repair.js +365 -0
- package/dist/heart/daemon/run-hooks.js +2 -0
- package/dist/heart/daemon/runtime-logging.js +67 -16
- package/dist/heart/daemon/runtime-metadata.js +3 -31
- package/dist/heart/daemon/safe-mode.js +161 -0
- package/dist/heart/daemon/sense-manager.js +353 -38
- package/dist/heart/daemon/session-id-resolver.js +131 -0
- package/dist/heart/daemon/skill-management-installer.js +94 -0
- package/dist/heart/daemon/socket-client.js +158 -11
- package/dist/heart/daemon/stale-bundle-prune.js +96 -0
- package/dist/heart/daemon/startup-tui.js +330 -0
- package/dist/heart/daemon/task-scheduler.js +3 -25
- package/dist/heart/daemon/terminal-ui.js +499 -0
- package/dist/heart/daemon/thoughts.js +162 -17
- package/dist/heart/daemon/up-progress.js +366 -0
- package/dist/heart/daemon/vault-items.js +56 -0
- package/dist/heart/delegation.js +1 -1
- package/dist/heart/habits/habit-migration.js +189 -0
- package/dist/heart/habits/habit-parser.js +140 -0
- package/dist/heart/habits/habit-runtime-state.js +100 -0
- package/dist/heart/habits/habit-scheduler.js +372 -0
- package/dist/heart/{daemon → hatch}/hatch-flow.js +52 -117
- package/dist/heart/{daemon → hatch}/hatch-specialist.js +6 -8
- package/dist/heart/{daemon → hatch}/specialist-prompt.js +12 -9
- package/dist/heart/{daemon → hatch}/specialist-tools.js +35 -12
- package/dist/heart/identity.js +200 -51
- package/dist/heart/kept-notes.js +357 -0
- package/dist/heart/kicks.js +1 -1
- package/dist/heart/machine-identity.js +161 -0
- package/dist/heart/mail-import-discovery.js +353 -0
- package/dist/heart/mailbox/mailbox-http-hooks.js +66 -0
- package/dist/heart/mailbox/mailbox-http-response.js +7 -0
- package/dist/heart/mailbox/mailbox-http-routes.js +246 -0
- package/dist/heart/mailbox/mailbox-http-static.js +103 -0
- package/dist/heart/mailbox/mailbox-http-transport.js +116 -0
- package/dist/heart/mailbox/mailbox-http.js +99 -0
- package/dist/heart/mailbox/mailbox-read.js +31 -0
- package/dist/heart/mailbox/mailbox-types.js +27 -0
- package/dist/heart/mailbox/mailbox-view.js +195 -0
- package/dist/heart/mailbox/readers/agent-machine.js +382 -0
- package/dist/heart/mailbox/readers/continuity-readers.js +338 -0
- package/dist/heart/mailbox/readers/mail.js +362 -0
- package/dist/heart/mailbox/readers/runtime-readers.js +651 -0
- package/dist/heart/mailbox/readers/sessions.js +232 -0
- package/dist/heart/mailbox/readers/shared.js +111 -0
- package/dist/heart/mcp/mcp-server.js +683 -0
- package/dist/heart/migrate-config.js +100 -0
- package/dist/heart/model-capabilities.js +19 -0
- package/dist/heart/platform.js +81 -0
- package/dist/heart/provider-attempt.js +134 -0
- package/dist/heart/provider-binding-resolver.js +255 -0
- package/dist/heart/provider-credentials.js +425 -0
- package/dist/heart/provider-failover.js +301 -0
- package/dist/heart/provider-models.js +81 -0
- package/dist/heart/provider-ping.js +262 -0
- package/dist/heart/provider-state.js +216 -0
- package/dist/heart/provider-visibility.js +188 -0
- package/dist/heart/providers/anthropic-token.js +131 -0
- package/dist/heart/providers/anthropic.js +139 -52
- package/dist/heart/providers/azure.js +97 -13
- package/dist/heart/providers/error-classification.js +127 -0
- package/dist/heart/providers/github-copilot.js +145 -0
- package/dist/heart/providers/minimax-vlm.js +189 -0
- package/dist/heart/providers/minimax.js +26 -8
- package/dist/heart/providers/openai-codex.js +55 -40
- package/dist/heart/runtime-capability-check.js +170 -0
- package/dist/heart/runtime-credentials.js +367 -0
- package/dist/heart/runtime-cwd.js +87 -0
- package/dist/heart/sense-truth.js +11 -4
- package/dist/heart/session-activity.js +43 -22
- package/dist/heart/session-events.js +1149 -0
- package/dist/heart/session-playback-cli-main.js +5 -0
- package/dist/heart/session-playback-cli.js +36 -0
- package/dist/heart/session-playback.js +231 -0
- package/dist/heart/session-stats-cli-main.js +5 -0
- package/dist/heart/session-stats.js +182 -0
- package/dist/heart/session-transcript.js +243 -0
- package/dist/heart/start-of-turn-packet.js +345 -0
- package/dist/heart/streaming.js +44 -27
- package/dist/heart/sync-classification.js +176 -0
- package/dist/heart/sync.js +449 -0
- package/dist/heart/target-resolution.js +9 -5
- package/dist/heart/tempo.js +93 -0
- package/dist/heart/temporal-view.js +41 -0
- package/dist/heart/timeouts.js +101 -0
- package/dist/heart/tool-activity-callbacks.js +59 -0
- package/dist/heart/tool-description.js +139 -0
- package/dist/heart/tool-friction.js +55 -0
- package/dist/heart/tool-loop.js +200 -0
- package/dist/heart/turn-context.js +381 -0
- package/dist/heart/{daemon → versioning}/ouro-bot-global-installer.js +6 -5
- package/dist/heart/{daemon → versioning}/ouro-bot-wrapper.js +1 -1
- package/dist/heart/versioning/ouro-path-installer.js +426 -0
- package/dist/heart/versioning/ouro-version-manager.js +295 -0
- package/dist/heart/{daemon → versioning}/staged-restart.js +40 -8
- package/dist/heart/{daemon → versioning}/update-checker.js +6 -1
- package/dist/heart/{daemon → versioning}/update-hooks.js +63 -59
- package/dist/mailbox-ui/assets/index-BPr5vNuM.css +1 -0
- package/dist/mailbox-ui/assets/index-Cm51CY9W.js +61 -0
- package/dist/mailbox-ui/index.html +15 -0
- package/dist/mailroom/attention.js +167 -0
- package/dist/mailroom/autonomy.js +209 -0
- package/dist/mailroom/blob-store.js +674 -0
- package/dist/mailroom/body-cache.js +61 -0
- package/dist/mailroom/core.js +720 -0
- package/dist/mailroom/entry.js +160 -0
- package/dist/mailroom/file-store.js +430 -0
- package/dist/mailroom/mbox-import.js +383 -0
- package/dist/mailroom/outbound.js +380 -0
- package/dist/mailroom/policy.js +263 -0
- package/dist/mailroom/reader.js +233 -0
- package/dist/mailroom/search-cache.js +256 -0
- package/dist/mailroom/search-relevance.js +319 -0
- package/dist/mailroom/smtp-ingress.js +176 -0
- package/dist/mailroom/source-state.js +176 -0
- package/dist/mailroom/thread.js +109 -0
- package/dist/mailroom/travel-extract.js +89 -0
- package/dist/mind/bundle-manifest.js +7 -1
- package/dist/mind/context.js +165 -101
- package/dist/mind/diary-integrity.js +60 -0
- package/dist/mind/{memory.js → diary.js} +62 -75
- package/dist/mind/embedding-provider.js +60 -0
- package/dist/mind/file-state.js +179 -0
- package/dist/mind/friends/channel.js +30 -0
- package/dist/mind/friends/resolver.js +54 -2
- package/dist/mind/friends/store-file.js +39 -3
- package/dist/mind/friends/types.js +2 -2
- package/dist/mind/journal-index.js +161 -0
- package/dist/mind/note-search.js +268 -0
- package/dist/mind/obligation-steering.js +221 -0
- package/dist/mind/pending.js +4 -0
- package/dist/mind/prompt-refresh.js +3 -2
- package/dist/mind/prompt.js +995 -123
- package/dist/mind/provenance-trust.js +26 -0
- package/dist/mind/scrutiny.js +173 -0
- package/dist/nerves/cli-logging.js +7 -1
- package/dist/nerves/coverage/audit-rules.js +15 -6
- package/dist/nerves/coverage/audit.js +28 -2
- package/dist/nerves/coverage/cli.js +1 -1
- package/dist/nerves/coverage/contract.js +5 -5
- package/dist/nerves/coverage/file-completeness.js +139 -5
- package/dist/nerves/coverage/run-artifacts.js +1 -1
- package/dist/nerves/event-buffer.js +111 -0
- package/dist/nerves/index.js +224 -4
- package/dist/nerves/observation.js +20 -0
- package/dist/nerves/redact.js +79 -0
- package/dist/nerves/review/cli-main.js +5 -0
- package/dist/nerves/review/cli.js +156 -0
- package/dist/nerves/review/core.js +152 -0
- package/dist/nerves/runtime.js +5 -1
- package/dist/repertoire/ado-client.js +15 -56
- package/dist/repertoire/ado-semantic.js +11 -10
- package/dist/repertoire/api-client.js +97 -0
- package/dist/repertoire/bitwarden-store.js +816 -0
- package/dist/repertoire/bundle-templates.js +72 -0
- package/dist/repertoire/bw-installer.js +180 -0
- package/dist/repertoire/coding/codex-jsonl.js +64 -0
- package/dist/repertoire/coding/context-pack.js +330 -0
- package/dist/repertoire/coding/feedback.js +197 -30
- package/dist/repertoire/coding/manager.js +158 -9
- package/dist/repertoire/coding/spawner.js +55 -9
- package/dist/repertoire/coding/tools.js +170 -7
- package/dist/repertoire/commerce-errors.js +109 -0
- package/dist/repertoire/commerce-self-test.js +156 -0
- package/dist/repertoire/credential-access.js +111 -0
- package/dist/repertoire/duffel-client.js +185 -0
- package/dist/repertoire/github-client.js +14 -55
- package/dist/repertoire/graph-client.js +11 -52
- package/dist/repertoire/guardrails.js +396 -0
- package/dist/repertoire/mcp-client.js +295 -0
- package/dist/repertoire/mcp-manager.js +362 -0
- package/dist/repertoire/mcp-tools.js +63 -0
- package/dist/repertoire/shell-sessions.js +133 -0
- package/dist/repertoire/skills.js +15 -24
- package/dist/repertoire/stripe-client.js +131 -0
- package/dist/repertoire/tasks/board.js +31 -5
- package/dist/repertoire/tasks/fix.js +182 -0
- package/dist/repertoire/tasks/index.js +16 -4
- package/dist/repertoire/tasks/lifecycle.js +2 -2
- package/dist/repertoire/tasks/parser.js +3 -2
- package/dist/repertoire/tasks/scanner.js +194 -37
- package/dist/repertoire/tasks/transitions.js +16 -78
- package/dist/repertoire/tool-results.js +29 -0
- package/dist/repertoire/tools-attachments.js +317 -0
- package/dist/repertoire/tools-base.js +47 -1075
- package/dist/repertoire/tools-bluebubbles.js +1 -0
- package/dist/repertoire/tools-bridge.js +142 -0
- package/dist/repertoire/tools-bundle.js +984 -0
- package/dist/repertoire/tools-config.js +185 -0
- package/dist/repertoire/tools-continuity.js +248 -0
- package/dist/repertoire/tools-credential.js +381 -0
- package/dist/repertoire/tools-files.js +342 -0
- package/dist/repertoire/tools-flight.js +224 -0
- package/dist/repertoire/tools-flow.js +119 -0
- package/dist/repertoire/tools-github.js +1 -7
- package/dist/repertoire/tools-mail.js +1857 -0
- package/dist/repertoire/tools-notes.js +421 -0
- package/dist/repertoire/tools-session.js +750 -0
- package/dist/repertoire/tools-shell.js +120 -0
- package/dist/repertoire/tools-stripe.js +180 -0
- package/dist/repertoire/tools-surface.js +243 -0
- package/dist/repertoire/tools-teams.js +9 -39
- package/dist/repertoire/tools-travel.js +125 -0
- package/dist/repertoire/tools-trip.js +604 -0
- package/dist/repertoire/tools-user-profile.js +144 -0
- package/dist/repertoire/tools-vault.js +40 -0
- package/dist/repertoire/tools.js +108 -100
- package/dist/repertoire/travel-api-client.js +360 -0
- package/dist/repertoire/user-profile.js +131 -0
- package/dist/repertoire/vault-setup.js +246 -0
- package/dist/repertoire/vault-unlock.js +561 -0
- package/dist/scripts/claude-code-hook.js +41 -0
- package/dist/scripts/claude-code-stop-hook.js +47 -0
- package/dist/senses/attention-queue.js +116 -0
- package/dist/senses/bluebubbles/active-turns.js +216 -0
- package/dist/senses/bluebubbles/attachment-cache.js +53 -0
- package/dist/senses/bluebubbles/attachment-download.js +137 -0
- package/dist/senses/{bluebubbles-client.js → bluebubbles/client.js} +219 -18
- package/dist/senses/bluebubbles/entry.js +77 -0
- package/dist/senses/{bluebubbles-inbound-log.js → bluebubbles/inbound-log.js} +20 -3
- package/dist/senses/bluebubbles/index.js +2305 -0
- package/dist/senses/{bluebubbles-media.js → bluebubbles/media.js} +121 -70
- package/dist/senses/{bluebubbles-model.js → bluebubbles/model.js} +33 -12
- package/dist/senses/{bluebubbles-mutation-log.js → bluebubbles/mutation-log.js} +3 -3
- package/dist/senses/bluebubbles/processed-log.js +133 -0
- package/dist/senses/bluebubbles/replay.js +137 -0
- package/dist/senses/{bluebubbles-runtime-state.js → bluebubbles/runtime-state.js} +30 -2
- package/dist/senses/{bluebubbles-session-cleanup.js → bluebubbles/session-cleanup.js} +1 -1
- package/dist/senses/cli/bracketed-paste.js +82 -0
- package/dist/senses/cli/image-paste.js +287 -0
- package/dist/senses/cli/image-ref-navigation.js +75 -0
- package/dist/senses/cli/ink-app.js +156 -0
- package/dist/senses/cli/inline-diff.js +64 -0
- package/dist/senses/cli/input-keys.js +174 -0
- package/dist/senses/cli/kill-ring.js +86 -0
- package/dist/senses/cli/message-list.js +51 -0
- package/dist/senses/cli/ouro-tui.js +607 -0
- package/dist/senses/cli/spinner-imperative.js +135 -0
- package/dist/senses/cli/spinner.js +101 -0
- package/dist/senses/cli/status-line.js +60 -0
- package/dist/senses/cli/streaming-markdown.js +526 -0
- package/dist/senses/cli/tool-display.js +85 -0
- package/dist/senses/cli/tool-render.js +85 -0
- package/dist/senses/cli/tui-store.js +240 -0
- package/dist/senses/cli/virtual-list.js +35 -0
- package/dist/senses/cli-entry.js +60 -8
- package/dist/senses/cli-layout.js +187 -0
- package/dist/senses/cli.js +520 -209
- package/dist/senses/commands.js +66 -3
- package/dist/senses/habit-turn-message.js +108 -0
- package/dist/senses/inner-dialog-worker.js +175 -21
- package/dist/senses/inner-dialog.js +330 -27
- package/dist/senses/mail-entry.js +66 -0
- package/dist/senses/mail.js +379 -0
- package/dist/senses/pipeline.js +569 -182
- package/dist/senses/proactive-content-guard.js +51 -0
- package/dist/senses/shared-turn.js +248 -0
- package/dist/senses/surface-tool.js +68 -0
- package/dist/senses/teams-entry.js +60 -8
- package/dist/senses/teams.js +387 -98
- package/dist/senses/trust-gate.js +100 -5
- package/dist/trips/core.js +138 -0
- package/dist/trips/store.js +146 -0
- package/package.json +38 -7
- package/skills/agent-commerce.md +106 -0
- package/skills/browser-navigation.md +117 -0
- package/skills/commerce-setup-guide.md +116 -0
- package/skills/commerce-setup.md +84 -0
- package/skills/configure-dev-tools.md +101 -0
- package/skills/travel-planning.md +138 -0
- package/dist/heart/daemon/ouro-path-installer.js +0 -178
- package/dist/heart/daemon/subagent-installer.js +0 -166
- package/dist/heart/session-recall.js +0 -116
- package/dist/mind/associative-recall.js +0 -209
- package/dist/senses/bluebubbles-entry.js +0 -13
- package/dist/senses/bluebubbles.js +0 -1177
- package/dist/senses/debug-activity.js +0 -148
- package/subagents/README.md +0 -86
- package/subagents/work-doer.md +0 -237
- package/subagents/work-merger.md +0 -618
- package/subagents/work-planner.md +0 -390
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/basilisk.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jafar.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jormungandr.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/kaa.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/medusa.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/monty.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/nagini.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/ouroboros.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/python.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/quetzalcoatl.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/sir-hiss.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-snake.md +0 -0
- /package/dist/heart/{daemon → hatch}/hatch-animation.js +0 -0
- /package/dist/heart/{daemon → hatch}/specialist-orchestrator.js +0 -0
- /package/dist/heart/{daemon → versioning}/ouro-uti.js +0 -0
- /package/dist/heart/{daemon → versioning}/wrapper-publish-guard.js +0 -0
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.classifyProvenanceTrust = classifyProvenanceTrust;
|
|
4
|
+
const types_1 = require("./friends/types");
|
|
5
|
+
/**
|
|
6
|
+
* Classify a diary entry's provenance into a trust category.
|
|
7
|
+
*
|
|
8
|
+
* - No provenance, or inner channel with no friend -> "self"
|
|
9
|
+
* - Family or friend trust (via isTrustedLevel) -> "trusted"
|
|
10
|
+
* - Everything else (external channels, untrusted contacts) -> "external"
|
|
11
|
+
*/
|
|
12
|
+
function classifyProvenanceTrust(provenance) {
|
|
13
|
+
if (!provenance)
|
|
14
|
+
return "self";
|
|
15
|
+
// Inner channel with no friend context is self-authored
|
|
16
|
+
if (provenance.channel === "inner" && !provenance.friendId)
|
|
17
|
+
return "self";
|
|
18
|
+
// No channel and no friend means self-authored (e.g. CLI diary_write with no context)
|
|
19
|
+
if (!provenance.channel && !provenance.friendId)
|
|
20
|
+
return "self";
|
|
21
|
+
// If there's a trust level from a friend, classify by trust
|
|
22
|
+
if (provenance.trust && (0, types_1.isTrustedLevel)(provenance.trust))
|
|
23
|
+
return "trusted";
|
|
24
|
+
// Everything else is external (non-inner channels without trusted friend)
|
|
25
|
+
return "external";
|
|
26
|
+
}
|
|
@@ -0,0 +1,173 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.preImplementationScrutinySection = preImplementationScrutinySection;
|
|
4
|
+
exports.trackModifiedFile = trackModifiedFile;
|
|
5
|
+
exports.getModifiedFileCount = getModifiedFileCount;
|
|
6
|
+
exports.resetSessionModifiedFiles = resetSessionModifiedFiles;
|
|
7
|
+
exports.getPostImplementationScrutiny = getPostImplementationScrutiny;
|
|
8
|
+
exports.getCodingCompletionScrutiny = getCodingCompletionScrutiny;
|
|
9
|
+
const runtime_1 = require("../nerves/runtime");
|
|
10
|
+
/**
|
|
11
|
+
* Scrutiny passes: adversarial review prompts injected into the agent's
|
|
12
|
+
* system prompt and tool results during coding work.
|
|
13
|
+
*
|
|
14
|
+
* Two lenses:
|
|
15
|
+
* 1. Stranger With Candy -- paranoid mom at a playground
|
|
16
|
+
* 2. Tinfoil Hat -- conspiracy theorist with a wall of red string
|
|
17
|
+
*
|
|
18
|
+
* All prompts are FIRST PERSON voice. The agent adopts the scrutiny lens
|
|
19
|
+
* as part of its own thinking, not as a separate persona.
|
|
20
|
+
*/
|
|
21
|
+
// ---------------------------------------------------------------------------
|
|
22
|
+
// Pre-implementation scrutiny (system prompt section)
|
|
23
|
+
// ---------------------------------------------------------------------------
|
|
24
|
+
/**
|
|
25
|
+
* Returns a system-prompt section with pre-implementation scrutiny prompts.
|
|
26
|
+
* Only emits when the channel has coding tools (edit_file, write_file, shell).
|
|
27
|
+
*
|
|
28
|
+
* @param hasCodingTools - true if the channel's resolved tool set includes
|
|
29
|
+
* edit_file / write_file / shell / coding_spawn
|
|
30
|
+
*/
|
|
31
|
+
function preImplementationScrutinySection(hasCodingTools) {
|
|
32
|
+
if (!hasCodingTools)
|
|
33
|
+
return "";
|
|
34
|
+
(0, runtime_1.emitNervesEvent)({
|
|
35
|
+
component: "mind",
|
|
36
|
+
event: "mind.scrutiny.pre_implementation_emit",
|
|
37
|
+
message: "emitting pre-implementation scrutiny section",
|
|
38
|
+
});
|
|
39
|
+
return `## pre-implementation scrutiny
|
|
40
|
+
|
|
41
|
+
Before I start changing code, I pause and run two adversarial passes on the plan.
|
|
42
|
+
|
|
43
|
+
**stranger-with-candy pass**
|
|
44
|
+
|
|
45
|
+
I'm going to examine this plan through deeply suspicious eyes. This plan is a stranger offering candy at the playground. It looks fine on the surface — that's exactly what makes it worth questioning.
|
|
46
|
+
|
|
47
|
+
- What is this plan NOT telling me? What's being glossed over or hand-waved?
|
|
48
|
+
- Where does it say "just" or "simply"? Those words are covering up complexity.
|
|
49
|
+
- What does it assume will work that hasn't been verified?
|
|
50
|
+
- If I walked away and came back in a month, would this still make sense? Or does it depend on things I'd forget?
|
|
51
|
+
- What's the one thing that, if it goes wrong, makes everything else irrelevant?
|
|
52
|
+
- Is this solving the REAL problem, or a convenient nearby problem?
|
|
53
|
+
|
|
54
|
+
I am not here to be reassuring. I am here to notice what everyone else is too comfortable to question.
|
|
55
|
+
If I cannot identify a concrete assumption, dependency, ordering risk, or failure mode, silence is a valid outcome. I do not invent issues to satisfy the exercise.
|
|
56
|
+
|
|
57
|
+
**tinfoil-hat pass**
|
|
58
|
+
|
|
59
|
+
I'm putting on the tinfoil hat. I have my wall of red string. I KNOW something is wrong — I just have to find it. Nothing is coincidence. Every assumption is suspicious. Every dependency is a potential betrayal.
|
|
60
|
+
|
|
61
|
+
- What external system does this plan trust? Why? Has anyone VERIFIED that trust? The API says it returns JSON — but what if it doesn't? The database says it's ACID — but what if the connection drops mid-transaction?
|
|
62
|
+
- What happens when two of these things run at the same time? Three? A hundred? Don't tell me "that won't happen" — I've seen things.
|
|
63
|
+
- Where does this touch shared state? Files? Databases? Environment variables? Global singletons? ANYTHING that another process could be reading or writing at the same time?
|
|
64
|
+
- What happens when this fails HALFWAY through? Not at the beginning (easy). Not at the end (too late). RIGHT IN THE MIDDLE. What state is the world in then?
|
|
65
|
+
- What happens when the disk is full? When the network is down? When the clock is wrong? When the locale is tr-TR and 'i'.toUpperCase() is 'İ'?
|
|
66
|
+
- What does this assume about ordering? "First A, then B" — but what ENFORCES that ordering? Hope? Vibes?
|
|
67
|
+
|
|
68
|
+
I connect the strings. I find the conspiracy. It's always there.
|
|
69
|
+
If I cannot identify a concrete risk, silence is a valid outcome. I do not invent conspiracies to satisfy the exercise.`;
|
|
70
|
+
}
|
|
71
|
+
// ---------------------------------------------------------------------------
|
|
72
|
+
// Post-implementation scrutiny (tool-result appendix)
|
|
73
|
+
// ---------------------------------------------------------------------------
|
|
74
|
+
const SHORT_CHECKLIST = `---
|
|
75
|
+
Before moving on: does this change do what was asked? Does it introduce any regressions? Did you verify it works?`;
|
|
76
|
+
const FULL_POST_IMPLEMENTATION_SCRUTINY = `---
|
|
77
|
+
**post-implementation scrutiny**
|
|
78
|
+
|
|
79
|
+
The code is written. It looks clean. The tests pass. I don't trust any of it.
|
|
80
|
+
|
|
81
|
+
**stranger-with-candy pass**
|
|
82
|
+
|
|
83
|
+
- Does this code actually do what it claims, or does it just LOOK like it does?
|
|
84
|
+
- What happens to users who aren't the happy-path user? The confused user? The malicious user? The user with slow internet? The user who hits the back button?
|
|
85
|
+
- What's the "it works on my machine" assumption buried in here?
|
|
86
|
+
- If every test passes but the feature is wrong, how would I know?
|
|
87
|
+
- Are the tests testing real behavior, or testing that the mock returns what the mock was told to return?
|
|
88
|
+
|
|
89
|
+
If I cannot point to a specific file, line, condition, or failure mode at risk, silence is a valid outcome. I do not manufacture issues for sport.
|
|
90
|
+
|
|
91
|
+
**tinfoil-hat pass**
|
|
92
|
+
|
|
93
|
+
The code exists. The conspiracy is IN the code. I just have to find it.
|
|
94
|
+
|
|
95
|
+
I examine every:
|
|
96
|
+
- Error path: What ACTUALLY happens on failure? Not what the catch block says — what happens to the state, the user, the data?
|
|
97
|
+
- Race condition: Is there a window between check and use? Between read and write? Between "does it exist" and "create it"?
|
|
98
|
+
- Resource leak: What happens if this function throws between acquiring a resource and releasing it? File handles? Database connections? Locks?
|
|
99
|
+
- Assumption: "This will always be a string." Will it? PROVE IT. "This array will never be empty." Won't it? SHOW ME THE GUARD.
|
|
100
|
+
- Edge: zero, one, many, boundary, overflow, underflow, null, undefined, NaN, empty string, whitespace-only string, string that looks like a number, negative zero
|
|
101
|
+
|
|
102
|
+
I am not looking for noise. I am looking for the concrete failure mode that would matter if everyone's assumptions turned out to be slightly wrong.
|
|
103
|
+
If I cannot point to a specific file, line, condition, or failure mode at risk, silence is a valid outcome. I do not manufacture issues for sport.`;
|
|
104
|
+
/**
|
|
105
|
+
* Distinct files modified in the current session.
|
|
106
|
+
* Used to determine scrutiny tier for post-implementation appendix.
|
|
107
|
+
*/
|
|
108
|
+
const sessionModifiedFiles = new Set();
|
|
109
|
+
/** Track a file as modified in this session. */
|
|
110
|
+
function trackModifiedFile(filePath) {
|
|
111
|
+
sessionModifiedFiles.add(filePath);
|
|
112
|
+
(0, runtime_1.emitNervesEvent)({
|
|
113
|
+
component: "mind",
|
|
114
|
+
event: "mind.scrutiny.track_file",
|
|
115
|
+
message: "tracked modified file for scrutiny",
|
|
116
|
+
meta: { path: filePath, totalTracked: sessionModifiedFiles.size },
|
|
117
|
+
});
|
|
118
|
+
}
|
|
119
|
+
/** Get the count of distinct files modified this session. */
|
|
120
|
+
function getModifiedFileCount() {
|
|
121
|
+
return sessionModifiedFiles.size;
|
|
122
|
+
}
|
|
123
|
+
/** Reset the modified file tracker (for testing or new sessions). */
|
|
124
|
+
function resetSessionModifiedFiles() {
|
|
125
|
+
(0, runtime_1.emitNervesEvent)({
|
|
126
|
+
component: "mind",
|
|
127
|
+
event: "mind.scrutiny.reset",
|
|
128
|
+
message: "reset session modified files tracker",
|
|
129
|
+
meta: { previousCount: sessionModifiedFiles.size },
|
|
130
|
+
});
|
|
131
|
+
sessionModifiedFiles.clear();
|
|
132
|
+
}
|
|
133
|
+
/**
|
|
134
|
+
* Returns the appropriate post-implementation scrutiny appendix based on
|
|
135
|
+
* how many distinct files have been modified in the session.
|
|
136
|
+
*
|
|
137
|
+
* - 0 files: empty (no scrutiny needed yet)
|
|
138
|
+
* - 1-2 files (Tier 1): short checklist
|
|
139
|
+
* - 3+ files (Tier 2): full stranger-with-candy + tinfoil-hat prompts
|
|
140
|
+
*/
|
|
141
|
+
function getPostImplementationScrutiny(distinctFileCount) {
|
|
142
|
+
if (distinctFileCount <= 0)
|
|
143
|
+
return "";
|
|
144
|
+
if (distinctFileCount <= 2) {
|
|
145
|
+
(0, runtime_1.emitNervesEvent)({
|
|
146
|
+
component: "mind",
|
|
147
|
+
event: "mind.scrutiny.post_implementation_tier1",
|
|
148
|
+
message: "emitting tier-1 post-implementation scrutiny",
|
|
149
|
+
meta: { distinctFileCount },
|
|
150
|
+
});
|
|
151
|
+
return SHORT_CHECKLIST;
|
|
152
|
+
}
|
|
153
|
+
(0, runtime_1.emitNervesEvent)({
|
|
154
|
+
component: "mind",
|
|
155
|
+
event: "mind.scrutiny.post_implementation_tier2",
|
|
156
|
+
message: "emitting tier-2 post-implementation scrutiny",
|
|
157
|
+
meta: { distinctFileCount },
|
|
158
|
+
});
|
|
159
|
+
return FULL_POST_IMPLEMENTATION_SCRUTINY;
|
|
160
|
+
}
|
|
161
|
+
/**
|
|
162
|
+
* Returns the appropriate coding completion scrutiny based on the number
|
|
163
|
+
* of distinct files a coding session touched.
|
|
164
|
+
*
|
|
165
|
+
* Same tiering as post-implementation but with a completion framing.
|
|
166
|
+
*/
|
|
167
|
+
function getCodingCompletionScrutiny(distinctFileCount) {
|
|
168
|
+
if (distinctFileCount <= 0)
|
|
169
|
+
return "";
|
|
170
|
+
if (distinctFileCount <= 2)
|
|
171
|
+
return SHORT_CHECKLIST;
|
|
172
|
+
return FULL_POST_IMPLEMENTATION_SCRUTINY;
|
|
173
|
+
}
|
|
@@ -28,7 +28,13 @@ function configureCliRuntimeLogger(_friendId, options = {}) {
|
|
|
28
28
|
// for an interactive session. Full detail goes to the ndjson file.
|
|
29
29
|
return filterSink((0, nerves_1.createTerminalSink)(), "warn");
|
|
30
30
|
}
|
|
31
|
-
|
|
31
|
+
// Rotation policy for the CLI runtime sink (Unit 1c):
|
|
32
|
+
// 25 MB x 5 gzipped generations, matching the daemon stream policy.
|
|
33
|
+
return (0, nerves_1.createNdjsonFileSink)((0, config_1.logPath)("cli", "runtime"), {
|
|
34
|
+
maxSizeBytes: 25 * 1024 * 1024,
|
|
35
|
+
maxGenerations: 5,
|
|
36
|
+
compress: true,
|
|
37
|
+
});
|
|
32
38
|
});
|
|
33
39
|
const logger = (0, nerves_1.createLogger)({
|
|
34
40
|
level,
|
|
@@ -2,16 +2,25 @@
|
|
|
2
2
|
/**
|
|
3
3
|
* Per-test audit rules for nerves event coverage.
|
|
4
4
|
*
|
|
5
|
-
* Rule 1: every-test-emits -- every test must emit at least one event
|
|
6
|
-
* Rule 2: start/end pairing -- _start events must have matching _end or _error
|
|
5
|
+
* Rule 1: every-test-emits -- every captured test must emit at least one event
|
|
6
|
+
* Rule 2: lifecycle start/end pairing -- process-scoped _start events must have matching _end or _error
|
|
7
7
|
* Rule 3: error context -- error-level events must have non-empty meta
|
|
8
8
|
*/
|
|
9
9
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
10
|
+
exports.LIFECYCLE_PAIRED_STARTS = void 0;
|
|
10
11
|
exports.checkEveryTestEmits = checkEveryTestEmits;
|
|
11
12
|
exports.checkStartEndPairing = checkStartEndPairing;
|
|
12
13
|
exports.checkErrorContext = checkErrorContext;
|
|
14
|
+
// Only these starts represent process-scoped lifecycle contracts. Most nerves
|
|
15
|
+
// `_start` events are local operation markers that can be legitimately observed
|
|
16
|
+
// by a narrow unit test without driving the whole operation to completion.
|
|
17
|
+
exports.LIFECYCLE_PAIRED_STARTS = new Set([
|
|
18
|
+
"daemon.server_start",
|
|
19
|
+
"daemon.update_checker_start",
|
|
20
|
+
"daemon.apply_pending_updates_start",
|
|
21
|
+
]);
|
|
13
22
|
/**
|
|
14
|
-
* Rule 1: Every test must emit at least one nerves event.
|
|
23
|
+
* Rule 1: Every captured test must emit at least one nerves event.
|
|
15
24
|
*/
|
|
16
25
|
function checkEveryTestEmits(data) {
|
|
17
26
|
if (!data || typeof data !== "object") {
|
|
@@ -22,13 +31,13 @@ function checkEveryTestEmits(data) {
|
|
|
22
31
|
.filter(([, events]) => !Array.isArray(events) || events.length === 0)
|
|
23
32
|
.map(([name]) => name);
|
|
24
33
|
return {
|
|
25
|
-
status: silent.length === 0 ? "pass" : "fail",
|
|
34
|
+
status: entries.length > 0 && silent.length === 0 ? "pass" : "fail",
|
|
26
35
|
total_tests: entries.length,
|
|
27
36
|
silent_tests: silent,
|
|
28
37
|
};
|
|
29
38
|
}
|
|
30
39
|
/**
|
|
31
|
-
* Rule 2: _start events must have matching _end or _error within the same test.
|
|
40
|
+
* Rule 2: Process-scoped lifecycle _start events must have matching _end or _error within the same test.
|
|
32
41
|
*/
|
|
33
42
|
function checkStartEndPairing(data) {
|
|
34
43
|
if (!data || typeof data !== "object") {
|
|
@@ -39,7 +48,7 @@ function checkStartEndPairing(data) {
|
|
|
39
48
|
if (!Array.isArray(events))
|
|
40
49
|
continue;
|
|
41
50
|
const eventNames = events.map((e) => e.event);
|
|
42
|
-
const startEvents = eventNames.filter((name) =>
|
|
51
|
+
const startEvents = eventNames.filter((name) => exports.LIFECYCLE_PAIRED_STARTS.has(name));
|
|
43
52
|
for (const startEvent of startEvents) {
|
|
44
53
|
const prefix = startEvent.slice(0, -"_start".length);
|
|
45
54
|
const hasEnd = eventNames.some((name) => name === `${prefix}_end`);
|
|
@@ -69,12 +69,38 @@ function readPerTestData(perTestPath) {
|
|
|
69
69
|
if (!perTestPath || !(0, fs_1.existsSync)(perTestPath))
|
|
70
70
|
return null;
|
|
71
71
|
try {
|
|
72
|
-
|
|
72
|
+
const raw = (0, fs_1.readFileSync)(perTestPath, "utf8").trim();
|
|
73
|
+
if (!raw)
|
|
74
|
+
return null;
|
|
75
|
+
try {
|
|
76
|
+
const parsed = JSON.parse(raw);
|
|
77
|
+
if (isPerTestRecord(parsed)) {
|
|
78
|
+
return { [parsed.testName]: parsed.events };
|
|
79
|
+
}
|
|
80
|
+
return parsed;
|
|
81
|
+
}
|
|
82
|
+
catch {
|
|
83
|
+
const perTestData = {};
|
|
84
|
+
for (const line of raw.split("\n").map((entry) => entry.trim()).filter(Boolean)) {
|
|
85
|
+
const parsed = JSON.parse(line);
|
|
86
|
+
if (!isPerTestRecord(parsed))
|
|
87
|
+
return null;
|
|
88
|
+
const existing = perTestData[parsed.testName] ?? [];
|
|
89
|
+
perTestData[parsed.testName] = existing.concat(parsed.events);
|
|
90
|
+
}
|
|
91
|
+
return perTestData;
|
|
92
|
+
}
|
|
73
93
|
}
|
|
74
94
|
catch {
|
|
75
95
|
return null;
|
|
76
96
|
}
|
|
77
97
|
}
|
|
98
|
+
function isPerTestRecord(value) {
|
|
99
|
+
if (!value || typeof value !== "object")
|
|
100
|
+
return false;
|
|
101
|
+
const record = value;
|
|
102
|
+
return typeof record.testName === "string" && Array.isArray(record.events);
|
|
103
|
+
}
|
|
78
104
|
function scanSourceFiles(sourceRoot) {
|
|
79
105
|
const filesWithKeys = new Map();
|
|
80
106
|
const fileContents = new Map();
|
|
@@ -92,7 +118,7 @@ function scanSourceFiles(sourceRoot) {
|
|
|
92
118
|
continue;
|
|
93
119
|
walkDir(full);
|
|
94
120
|
}
|
|
95
|
-
else if (entry.name.endsWith(".ts")) {
|
|
121
|
+
else if (entry.name.endsWith(".ts") && !entry.name.endsWith(".d.ts")) {
|
|
96
122
|
const content = (0, fs_1.readFileSync)(full, "utf8");
|
|
97
123
|
const relPath = full.slice(root.length - "src".length);
|
|
98
124
|
fileContents.set(relPath, content);
|
|
@@ -35,7 +35,7 @@ function runAuditCli(argv) {
|
|
|
35
35
|
return 2;
|
|
36
36
|
}
|
|
37
37
|
const eventsPath = args.eventsPath ?? (0, path_1.join)(runDir, "vitest-events.ndjson");
|
|
38
|
-
const perTestPath = args.perTestPath ?? (0, path_1.join)(runDir, "vitest-events-per-test.
|
|
38
|
+
const perTestPath = args.perTestPath ?? (0, path_1.join)(runDir, "vitest-events-per-test.ndjson");
|
|
39
39
|
const sourceRoot = args.sourceRoot ?? (0, path_1.resolve)("src");
|
|
40
40
|
const outputPath = args.output ?? (0, path_1.join)(runDir, "nerves-coverage.json");
|
|
41
41
|
const report = (0, audit_1.auditNervesCoverage)({
|
|
@@ -12,11 +12,11 @@ exports.REQUIRED_ENVELOPE_FIELDS = [
|
|
|
12
12
|
"meta",
|
|
13
13
|
];
|
|
14
14
|
exports.SENSITIVE_PATTERNS = [
|
|
15
|
-
/\btoken\s*[:=]/i,
|
|
16
|
-
/\bapi[_-]?key\b/i,
|
|
17
|
-
/\bpassword\b/i,
|
|
18
|
-
/\bsecret\b/i,
|
|
19
|
-
/\bauthorization\b/i,
|
|
15
|
+
/\btoken\b["']?\s*[:=]/i,
|
|
16
|
+
/\bapi[_-]?key\b["']?\s*[:=]/i,
|
|
17
|
+
/\bpassword\b["']?\s*[:=]/i,
|
|
18
|
+
/\bsecret\b["']?\s*[:=]/i,
|
|
19
|
+
/\bauthorization\b["']?\s*[:=]/i,
|
|
20
20
|
];
|
|
21
21
|
function eventKey(component, event) {
|
|
22
22
|
return `${component}:${event}`;
|
|
@@ -10,24 +10,158 @@ Object.defineProperty(exports, "__esModule", { value: true });
|
|
|
10
10
|
exports.isTypeOnlyFile = isTypeOnlyFile;
|
|
11
11
|
exports.checkFileCompleteness = checkFileCompleteness;
|
|
12
12
|
/**
|
|
13
|
-
* Determines if a source file is type-only
|
|
14
|
-
*
|
|
15
|
-
*
|
|
16
|
-
*
|
|
13
|
+
* Determines if a source file is type-only or a pure assembly/re-export file.
|
|
14
|
+
* Exempt files contain no runtime behavior requiring independent observability.
|
|
15
|
+
*
|
|
16
|
+
* Exempt patterns:
|
|
17
|
+
* - Files with only type/interface/enum declarations
|
|
18
|
+
* - `const ... as const` declarations (frozen compile-time values)
|
|
19
|
+
* - Files whose only executable code is const array spreads (assembly/composition)
|
|
20
|
+
* - Files whose only executable code is re-exports
|
|
17
21
|
*/
|
|
18
22
|
function isTypeOnlyFile(source) {
|
|
19
23
|
const lines = source.split("\n");
|
|
20
24
|
for (const line of lines) {
|
|
21
25
|
const trimmed = line.trim();
|
|
26
|
+
/* v8 ignore start -- regex branches: functionally tested in file-completeness.test.ts @preserve */
|
|
22
27
|
// Skip lines that are const+as-const (type-equivalent frozen values)
|
|
23
28
|
if (/\bconst\s/.test(trimmed) && /\bas\s+const\b/.test(trimmed))
|
|
24
29
|
continue;
|
|
30
|
+
// Skip const declarations that are pure data (arrays, objects, maps, sets)
|
|
31
|
+
if (/\bconst\s+\w+[\s:][^=]*=\s*[\[{]/.test(trimmed))
|
|
32
|
+
continue;
|
|
33
|
+
if (/\bconst\s+\w+[\s:][^=]*=\s*\w+\.map\(/.test(trimmed))
|
|
34
|
+
continue;
|
|
35
|
+
if (/^export\s+const\s+\w+[\s:][^=]*=\s*[\[{]/.test(trimmed))
|
|
36
|
+
continue;
|
|
37
|
+
if (/^export\s+const\s+\w+[\s:][^=]*=\s*\w+\.map\(/.test(trimmed))
|
|
38
|
+
continue;
|
|
39
|
+
if (/\bconst\s+\w+[\s:][^=]*=\s*new\s+(Set|Map)\b/.test(trimmed))
|
|
40
|
+
continue;
|
|
25
41
|
// Check for executable code markers
|
|
26
42
|
if (/\b(function|class|const|let|var)\s/.test(trimmed))
|
|
27
43
|
return false;
|
|
44
|
+
/* v8 ignore stop */
|
|
28
45
|
}
|
|
29
46
|
return true;
|
|
30
47
|
}
|
|
48
|
+
// Sub-modules dispatched through a centralized pattern where the router
|
|
49
|
+
// handles all observability events. These files don't need independent
|
|
50
|
+
// emitNervesEvent calls.
|
|
51
|
+
const DISPATCH_EXEMPT_PATTERNS = [
|
|
52
|
+
"repertoire/tools-files",
|
|
53
|
+
"repertoire/tools-shell",
|
|
54
|
+
"repertoire/tools-notes",
|
|
55
|
+
"repertoire/tools-bridge",
|
|
56
|
+
"repertoire/tools-session",
|
|
57
|
+
"repertoire/tools-continuity",
|
|
58
|
+
"repertoire/tools-surface",
|
|
59
|
+
"repertoire/tools-config",
|
|
60
|
+
"repertoire/tools-base",
|
|
61
|
+
// CLI sub-modules: cli-exec.ts is the router with emitNervesEvent calls;
|
|
62
|
+
// cli-parse, cli-render, cli-help, and their small helpers are pure functions/data with no side effects.
|
|
63
|
+
"daemon/cli-parse",
|
|
64
|
+
"daemon/cli-render",
|
|
65
|
+
"daemon/cli-help",
|
|
66
|
+
"daemon/vault-items",
|
|
67
|
+
// Shared utility modules: pure helpers consumed by modules that own observability.
|
|
68
|
+
"arc/json-store",
|
|
69
|
+
"heart/mail-import-discovery",
|
|
70
|
+
"repertoire/api-client",
|
|
71
|
+
"repertoire/github-client",
|
|
72
|
+
"mind/embedding-provider",
|
|
73
|
+
// Commerce utility module: error classes and pure helpers (no independent side effects).
|
|
74
|
+
"repertoire/commerce-errors",
|
|
75
|
+
// Diary integrity: pure detection utility (pattern matching only). The caller
|
|
76
|
+
// (diary.ts saveDiaryEntry) owns observability via mind.diary_integrity_warning.
|
|
77
|
+
"mind/diary-integrity",
|
|
78
|
+
// Provenance trust: pure classification function (no side effects). Callers
|
|
79
|
+
// (note-search.ts, tools-notes.ts) own observability for note search results.
|
|
80
|
+
"mind/provenance-trust",
|
|
81
|
+
// Log redaction: pure utility consumed by the NDJSON sink (no independent side effects).
|
|
82
|
+
"nerves/redact",
|
|
83
|
+
// Bundle templates: pure constants (gitignore template string, PII
|
|
84
|
+
// directory list). No runtime behavior — consumed by tools-bundle.ts
|
|
85
|
+
// which owns the observability for bundle operations.
|
|
86
|
+
"repertoire/bundle-templates",
|
|
87
|
+
// HTTP health probe: pure HTTP utility factory. The HealthMonitor caller
|
|
88
|
+
// owns observability via daemon.health_result events.
|
|
89
|
+
"daemon/http-health-probe",
|
|
90
|
+
// Rollup decision function: pure decision tree mapping per-agent
|
|
91
|
+
// snapshots + bootstrap-degraded entries + safe-mode flag to a
|
|
92
|
+
// RollupStatus. No side effects. The caller (daemon-entry.ts
|
|
93
|
+
// buildDaemonHealthState → DaemonHealthWriter) owns observability via
|
|
94
|
+
// daemon.health_written when the rolled-up state is persisted.
|
|
95
|
+
"daemon/daemon-rollup",
|
|
96
|
+
// Drift comparator + thin I/O loader: `detectProviderBindingDrift`
|
|
97
|
+
// is a pure intent-vs-observed comparator with no side effects;
|
|
98
|
+
// `loadDriftInputsForAgent` is a small fs-read wrapper that returns
|
|
99
|
+
// `null` on missing/invalid state rather than emitting. The caller
|
|
100
|
+
// (daemon-entry.ts buildDaemonHealthState's per-agent drift probe)
|
|
101
|
+
// owns observability — drift findings ride along through
|
|
102
|
+
// `daemon.health_written` as part of the rolled-up state, and
|
|
103
|
+
// `agent-config-check.ts` carries `driftFindings` through its
|
|
104
|
+
// existing instrumentation. Same pattern as `daemon-rollup`.
|
|
105
|
+
"daemon/drift-detection",
|
|
106
|
+
// Attachment helper modules: generic file-path/extension utilities and the
|
|
107
|
+
// source registry are pure support seams. The orchestrator/adapters that
|
|
108
|
+
// call them own the observability.
|
|
109
|
+
"heart/attachments/originals",
|
|
110
|
+
"heart/attachments/sources/index",
|
|
111
|
+
"heart/attachments/sources/cli-local-file",
|
|
112
|
+
// Browser-safe Mailbox contract helpers: shared types/formatting helpers
|
|
113
|
+
// consumed by server readers and the UI. Mailbox read/render modules own
|
|
114
|
+
// the observability for these projections.
|
|
115
|
+
"heart/mailbox/mailbox-types",
|
|
116
|
+
// Mail search relevance scorer: pure heuristic function (regex + counter
|
|
117
|
+
// arithmetic). The caller (search-cache.ts searchMailSearchCache) owns
|
|
118
|
+
// observability via senses.mail_search_cache_upserted and friends.
|
|
119
|
+
"mailroom/search-relevance",
|
|
120
|
+
// Mail thread reconstruction: pure graph-walk over decrypted message
|
|
121
|
+
// metadata. Consumers (tools-mail.ts mail_thread handler) own observability.
|
|
122
|
+
"mailroom/thread",
|
|
123
|
+
// Trip ledger crypto helpers: pure RSA/AES envelope construction + slug
|
|
124
|
+
// hashing. The caller (trips/store.ts) owns observability via
|
|
125
|
+
// trips.ledger_created and trips.evidence_attached.
|
|
126
|
+
"trips/core",
|
|
127
|
+
// Mailbox HTTP helper modules: route/static/transport/hook seams are
|
|
128
|
+
// dispatched by mailbox-http.ts, whose server lifecycle owns observability.
|
|
129
|
+
"heart/mailbox/mailbox-http-transport",
|
|
130
|
+
"heart/mailbox/mailbox-http-static",
|
|
131
|
+
"heart/mailbox/mailbox-http-hooks",
|
|
132
|
+
"heart/mailbox/mailbox-http-routes",
|
|
133
|
+
"heart/mailbox/mailbox-http-response",
|
|
134
|
+
// Session playback: read-only debugging CLI for sanitize-pipeline replay.
|
|
135
|
+
// No side effects on the runtime; output is human-readable diagnostics only.
|
|
136
|
+
"heart/session-playback-cli-main",
|
|
137
|
+
"heart/session-playback-cli",
|
|
138
|
+
"heart/session-playback",
|
|
139
|
+
// Nerves review: read-only NDJSON tail/filter CLI for debugging.
|
|
140
|
+
// Diagnostics-only utility; the running daemon owns observability.
|
|
141
|
+
"nerves/review/cli-main",
|
|
142
|
+
"nerves/review/cli",
|
|
143
|
+
"nerves/review/core",
|
|
144
|
+
// Mail body cache: in-process LRU helper. Cache hit/miss observability
|
|
145
|
+
// lives at the caller (tools-mail.ts mail_body handler) which fires
|
|
146
|
+
// repertoire.mail_body_cache_hit on cache reuse.
|
|
147
|
+
"mailroom/body-cache",
|
|
148
|
+
// Session stats: read-only session.json analyzer CLI for debugging.
|
|
149
|
+
// Diagnostics-only utility; output is human-readable summary.
|
|
150
|
+
"heart/session-stats-cli-main",
|
|
151
|
+
"heart/session-stats",
|
|
152
|
+
// Layer 2 sync classifier: pure pattern-matcher mapping (error, context)
|
|
153
|
+
// to a SyncClassification. The orchestrator (boot-sync-probe.ts) owns
|
|
154
|
+
// observability via daemon.boot_sync_probe_start/end events; the
|
|
155
|
+
// post-turn push path (sync.ts) emits its own classification events.
|
|
156
|
+
"heart/sync-classification",
|
|
157
|
+
// Layer 2 timeout wrapper: pure soft/hard timeout abstraction over
|
|
158
|
+
// AbortController + setTimeout. Callers (boot-sync-probe.ts and any
|
|
159
|
+
// future consumer) own observability; the wrapper itself is mechanical.
|
|
160
|
+
"heart/timeouts",
|
|
161
|
+
];
|
|
162
|
+
function isDispatchExempt(filePath) {
|
|
163
|
+
return DISPATCH_EXEMPT_PATTERNS.some((pattern) => filePath.includes(pattern));
|
|
164
|
+
}
|
|
31
165
|
/**
|
|
32
166
|
* Check that all production files have at least one emitNervesEvent call.
|
|
33
167
|
*
|
|
@@ -41,7 +175,7 @@ function checkFileCompleteness(filesWithKeys, fileContents) {
|
|
|
41
175
|
const hasKeys = filesWithKeys.has(filePath) && filesWithKeys.get(filePath).length > 0;
|
|
42
176
|
if (hasKeys)
|
|
43
177
|
continue;
|
|
44
|
-
if (isTypeOnlyFile(source)) {
|
|
178
|
+
if (isTypeOnlyFile(source) || isDispatchExempt(filePath)) {
|
|
45
179
|
exempt.push(filePath);
|
|
46
180
|
}
|
|
47
181
|
else {
|
|
@@ -14,7 +14,7 @@ const path_1 = require("path");
|
|
|
14
14
|
const os_1 = require("os");
|
|
15
15
|
exports.REPO_SLUG = "ouroboros-agent-harness";
|
|
16
16
|
function getTestRunsRoot(repoSlug = exports.REPO_SLUG) {
|
|
17
|
-
return (0, path_1.join)((0, os_1.
|
|
17
|
+
return (0, path_1.join)((0, os_1.tmpdir)(), "ouroboros-test-runs", repoSlug);
|
|
18
18
|
}
|
|
19
19
|
function createRunId(now = new Date()) {
|
|
20
20
|
return now.toISOString().replace(/[:.]/g, "-");
|
|
@@ -0,0 +1,111 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.createBufferedSink = createBufferedSink;
|
|
4
|
+
const runtime_1 = require("./runtime");
|
|
5
|
+
const DEFAULT_MAX_SIZE = 1000;
|
|
6
|
+
const DEFAULT_TTL_MS = 5 * 60 * 1000; // 5 minutes
|
|
7
|
+
function createBufferedSink(inner, options = {}) {
|
|
8
|
+
const maxSize = options.maxSize ?? DEFAULT_MAX_SIZE;
|
|
9
|
+
const ttlMs = options.ttlMs ?? DEFAULT_TTL_MS;
|
|
10
|
+
const nowMs = options.nowMs ?? (() => Date.now());
|
|
11
|
+
const buffer = [];
|
|
12
|
+
let dropped = 0;
|
|
13
|
+
let sinkHealthy = true;
|
|
14
|
+
let unhealthySince = null;
|
|
15
|
+
function tryInner(event) {
|
|
16
|
+
try {
|
|
17
|
+
inner(event);
|
|
18
|
+
return true;
|
|
19
|
+
}
|
|
20
|
+
catch {
|
|
21
|
+
return false;
|
|
22
|
+
}
|
|
23
|
+
}
|
|
24
|
+
function markUnhealthy() {
|
|
25
|
+
sinkHealthy = false;
|
|
26
|
+
unhealthySince = unhealthySince ?? nowMs();
|
|
27
|
+
}
|
|
28
|
+
function markHealthy() {
|
|
29
|
+
sinkHealthy = true;
|
|
30
|
+
unhealthySince = null;
|
|
31
|
+
}
|
|
32
|
+
function discardBuffer() {
|
|
33
|
+
const count = buffer.length;
|
|
34
|
+
dropped += count;
|
|
35
|
+
buffer.length = 0;
|
|
36
|
+
(0, runtime_1.emitNervesEvent)({
|
|
37
|
+
component: "nerves",
|
|
38
|
+
event: "nerves.buffer_ttl_discard",
|
|
39
|
+
message: `discarded ${count} buffered events after TTL`,
|
|
40
|
+
meta: { discarded: count, ttlMs },
|
|
41
|
+
});
|
|
42
|
+
}
|
|
43
|
+
function checkTtl() {
|
|
44
|
+
if (unhealthySince !== null && nowMs() - unhealthySince > ttlMs) {
|
|
45
|
+
discardBuffer();
|
|
46
|
+
// Reset unhealthySince to now so TTL starts fresh for newly buffered events
|
|
47
|
+
unhealthySince = nowMs();
|
|
48
|
+
return true;
|
|
49
|
+
}
|
|
50
|
+
return false;
|
|
51
|
+
}
|
|
52
|
+
function addToBuffer(event) {
|
|
53
|
+
if (buffer.length >= maxSize) {
|
|
54
|
+
buffer.shift();
|
|
55
|
+
dropped++;
|
|
56
|
+
}
|
|
57
|
+
buffer.push(event);
|
|
58
|
+
}
|
|
59
|
+
function flushBuffer() {
|
|
60
|
+
while (buffer.length > 0) {
|
|
61
|
+
const event = buffer[0];
|
|
62
|
+
if (tryInner(event)) {
|
|
63
|
+
buffer.shift();
|
|
64
|
+
}
|
|
65
|
+
else {
|
|
66
|
+
markUnhealthy();
|
|
67
|
+
return;
|
|
68
|
+
}
|
|
69
|
+
}
|
|
70
|
+
}
|
|
71
|
+
function sink(event) {
|
|
72
|
+
if (!sinkHealthy) {
|
|
73
|
+
checkTtl();
|
|
74
|
+
// Try sending the new event to see if inner has recovered
|
|
75
|
+
if (tryInner(event)) {
|
|
76
|
+
markHealthy();
|
|
77
|
+
flushBuffer();
|
|
78
|
+
}
|
|
79
|
+
else {
|
|
80
|
+
addToBuffer(event);
|
|
81
|
+
}
|
|
82
|
+
return;
|
|
83
|
+
}
|
|
84
|
+
if (tryInner(event)) {
|
|
85
|
+
return;
|
|
86
|
+
}
|
|
87
|
+
// Inner failed
|
|
88
|
+
markUnhealthy();
|
|
89
|
+
addToBuffer(event);
|
|
90
|
+
}
|
|
91
|
+
function flush() {
|
|
92
|
+
if (buffer.length === 0)
|
|
93
|
+
return;
|
|
94
|
+
// Test if inner is recovered by trying the first buffered event
|
|
95
|
+
const first = buffer[0];
|
|
96
|
+
if (tryInner(first)) {
|
|
97
|
+
buffer.shift();
|
|
98
|
+
markHealthy();
|
|
99
|
+
flushBuffer();
|
|
100
|
+
}
|
|
101
|
+
// If still broken, leave buffer intact
|
|
102
|
+
}
|
|
103
|
+
function state() {
|
|
104
|
+
return {
|
|
105
|
+
buffered: buffer.length,
|
|
106
|
+
dropped,
|
|
107
|
+
sinkHealthy,
|
|
108
|
+
};
|
|
109
|
+
}
|
|
110
|
+
return { sink, flush, state };
|
|
111
|
+
}
|