@cmetech/otto 1.1.1 → 1.2.4
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/dist/coworker/persona-commands.d.ts +1 -0
- package/dist/coworker/persona-commands.js +5 -0
- package/dist/coworker/persona-commands.test.d.ts +1 -0
- package/dist/coworker/persona-commands.test.js +45 -0
- package/dist/resources/.managed-resources-content-hash +1 -1
- package/dist/resources/extensions/_coworker-paths.js +8 -0
- package/dist/resources/extensions/coworker-artifacts/artifacts-command.js +31 -0
- package/dist/resources/extensions/coworker-artifacts/artifacts-singleton.js +17 -0
- package/dist/resources/extensions/coworker-artifacts/extension-manifest.json +13 -0
- package/dist/resources/extensions/coworker-artifacts/index.js +125 -0
- package/dist/resources/extensions/coworker-artifacts/list-tool.js +27 -0
- package/dist/resources/extensions/coworker-artifacts/open-tool.js +25 -0
- package/dist/resources/extensions/coworker-memory/extension-manifest.json +13 -0
- package/dist/resources/extensions/coworker-memory/index.js +219 -0
- package/dist/resources/extensions/coworker-memory/memorize-tool.js +10 -0
- package/dist/resources/extensions/coworker-memory/memory-command.js +157 -0
- package/dist/resources/extensions/coworker-memory/memory-singleton.js +55 -0
- package/dist/resources/extensions/coworker-memory/recall-tool.js +18 -0
- package/dist/resources/extensions/coworker-memory/session-hooks.js +45 -0
- package/dist/resources/extensions/coworker-scratchpad/attach-banners.js +53 -0
- package/dist/resources/extensions/coworker-scratchpad/extension-manifest.json +13 -0
- package/dist/resources/extensions/coworker-scratchpad/format-age.js +9 -0
- package/dist/resources/extensions/coworker-scratchpad/helpers.js +38 -0
- package/dist/resources/extensions/coworker-scratchpad/index.js +199 -0
- package/dist/resources/extensions/coworker-scratchpad/mime-bundle.js +20 -0
- package/dist/resources/extensions/coworker-scratchpad/scratchpad-tool.js +118 -0
- package/dist/resources/extensions/coworker-scratchpad/session-sidecar.js +60 -0
- package/dist/resources/extensions/coworker-scratchpad/sp-command.js +597 -0
- package/dist/resources/extensions/coworker-scratchpad/workspace-pointer.js +41 -0
- package/dist/resources/extensions/coworker-scratchpad/workspace-root.js +17 -0
- package/dist/resources/extensions/coworker-vault/audit-command.js +35 -0
- package/dist/resources/extensions/coworker-vault/connect-command.js +42 -0
- package/dist/resources/extensions/coworker-vault/datasource-command.js +50 -0
- package/dist/resources/extensions/coworker-vault/extension-manifest.json +12 -0
- package/dist/resources/extensions/coworker-vault/index.js +171 -0
- package/dist/resources/extensions/coworker-vault/test-helpers.js +86 -0
- package/dist/resources/extensions/coworker-vault/vault-singleton.js +24 -0
- package/dist/resources/extensions/otto/commands/release-notes/_data.js +71 -0
- package/dist/resources/extensions/otto/commands/release-notes/command.js +15 -4
- package/dist/resources/extensions/subagent/index.js +8 -1
- package/dist/resources/extensions/subagent/launch.js +37 -5
- package/dist/resources/extensions/subagent/run-store.js +1 -0
- package/dist/resources/extensions/workflow/bootstrap/register-extension.js +2 -0
- package/dist/resources/extensions/workflow/bootstrap/register-hooks.js +10 -0
- package/dist/resources/extensions/workflow/persona-status.js +87 -0
- package/package.json +25 -10
- package/packages/contracts/package.json +1 -1
- package/packages/coworker-artifacts/dist/artifact-store.d.ts +25 -0
- package/packages/coworker-artifacts/dist/artifact-store.js +187 -0
- package/packages/coworker-artifacts/dist/dir-snapshot.d.ts +7 -0
- package/packages/coworker-artifacts/dist/dir-snapshot.js +54 -0
- package/packages/coworker-artifacts/dist/errors.d.ts +18 -0
- package/packages/coworker-artifacts/dist/errors.js +37 -0
- package/packages/coworker-artifacts/dist/index.d.ts +7 -0
- package/packages/coworker-artifacts/dist/index.js +7 -0
- package/packages/coworker-artifacts/dist/readme-renderer.d.ts +5 -0
- package/packages/coworker-artifacts/dist/readme-renderer.js +47 -0
- package/packages/coworker-artifacts/dist/resolve-uri.d.ts +3 -0
- package/packages/coworker-artifacts/dist/resolve-uri.js +29 -0
- package/packages/coworker-artifacts/dist/slug.d.ts +4 -0
- package/packages/coworker-artifacts/dist/slug.js +32 -0
- package/packages/coworker-artifacts/dist/types.d.ts +52 -0
- package/packages/coworker-artifacts/dist/types.js +1 -0
- package/packages/coworker-artifacts/package.json +20 -0
- package/packages/coworker-artifacts/src/artifact-store.test.ts +188 -0
- package/packages/coworker-artifacts/src/artifact-store.ts +206 -0
- package/packages/coworker-artifacts/src/artifacts-integration.test.ts +109 -0
- package/packages/coworker-artifacts/src/dir-snapshot.test.ts +71 -0
- package/packages/coworker-artifacts/src/dir-snapshot.ts +52 -0
- package/packages/coworker-artifacts/src/errors.test.ts +37 -0
- package/packages/coworker-artifacts/src/errors.ts +28 -0
- package/packages/coworker-artifacts/src/index.test.ts +22 -0
- package/packages/coworker-artifacts/src/index.ts +7 -0
- package/packages/coworker-artifacts/src/readme-renderer.test.ts +72 -0
- package/packages/coworker-artifacts/src/readme-renderer.ts +56 -0
- package/packages/coworker-artifacts/src/resolve-uri.test.ts +46 -0
- package/packages/coworker-artifacts/src/resolve-uri.ts +29 -0
- package/packages/coworker-artifacts/src/slug.test.ts +47 -0
- package/packages/coworker-artifacts/src/slug.ts +31 -0
- package/packages/coworker-artifacts/src/types.ts +61 -0
- package/packages/coworker-artifacts/tsconfig.json +15 -0
- package/packages/coworker-artifacts/tsconfig.publish.json +4 -0
- package/packages/coworker-memory/dist/context-injection.d.ts +9 -0
- package/packages/coworker-memory/dist/context-injection.js +41 -0
- package/packages/coworker-memory/dist/errors.d.ts +25 -0
- package/packages/coworker-memory/dist/errors.js +51 -0
- package/packages/coworker-memory/dist/index.d.ts +12 -0
- package/packages/coworker-memory/dist/index.js +12 -0
- package/packages/coworker-memory/dist/layer-a-store.d.ts +16 -0
- package/packages/coworker-memory/dist/layer-a-store.js +78 -0
- package/packages/coworker-memory/dist/local-sqlite-backend.d.ts +28 -0
- package/packages/coworker-memory/dist/local-sqlite-backend.js +167 -0
- package/packages/coworker-memory/dist/memory-backend.d.ts +14 -0
- package/packages/coworker-memory/dist/memory-backend.js +1 -0
- package/packages/coworker-memory/dist/memory-recorder.d.ts +50 -0
- package/packages/coworker-memory/dist/memory-recorder.js +69 -0
- package/packages/coworker-memory/dist/migrations/001-init.sql +38 -0
- package/packages/coworker-memory/dist/migrations/002-artifact-kind.sql +50 -0
- package/packages/coworker-memory/dist/paste-detector.d.ts +5 -0
- package/packages/coworker-memory/dist/paste-detector.js +14 -0
- package/packages/coworker-memory/dist/persona-seed.d.ts +10 -0
- package/packages/coworker-memory/dist/persona-seed.js +38 -0
- package/packages/coworker-memory/dist/recall-formatter.d.ts +2 -0
- package/packages/coworker-memory/dist/recall-formatter.js +14 -0
- package/packages/coworker-memory/dist/scope-resolver.d.ts +9 -0
- package/packages/coworker-memory/dist/scope-resolver.js +10 -0
- package/packages/coworker-memory/dist/types.d.ts +51 -0
- package/packages/coworker-memory/dist/types.js +2 -0
- package/packages/coworker-memory/dist/workspace-id.d.ts +3 -0
- package/packages/coworker-memory/dist/workspace-id.js +54 -0
- package/packages/coworker-memory/package.json +35 -0
- package/packages/coworker-memory/src/activator-integration.test.ts +141 -0
- package/packages/coworker-memory/src/context-injection.test.ts +72 -0
- package/packages/coworker-memory/src/context-injection.ts +57 -0
- package/packages/coworker-memory/src/errors.test.ts +45 -0
- package/packages/coworker-memory/src/errors.ts +42 -0
- package/packages/coworker-memory/src/index.test.ts +21 -0
- package/packages/coworker-memory/src/index.ts +12 -0
- package/packages/coworker-memory/src/layer-a-store.test.ts +85 -0
- package/packages/coworker-memory/src/layer-a-store.ts +88 -0
- package/packages/coworker-memory/src/local-sqlite-backend.test.ts +110 -0
- package/packages/coworker-memory/src/local-sqlite-backend.ts +185 -0
- package/packages/coworker-memory/src/memory-backend.ts +10 -0
- package/packages/coworker-memory/src/memory-integration.test.ts +89 -0
- package/packages/coworker-memory/src/memory-recorder.test.ts +101 -0
- package/packages/coworker-memory/src/memory-recorder.ts +95 -0
- package/packages/coworker-memory/src/migrations/001-init.sql +38 -0
- package/packages/coworker-memory/src/migrations/002-artifact-kind.sql +50 -0
- package/packages/coworker-memory/src/paste-detector.test.ts +23 -0
- package/packages/coworker-memory/src/paste-detector.ts +18 -0
- package/packages/coworker-memory/src/persona-seed.test.ts +57 -0
- package/packages/coworker-memory/src/persona-seed.ts +46 -0
- package/packages/coworker-memory/src/recall-formatter.test.ts +34 -0
- package/packages/coworker-memory/src/recall-formatter.ts +15 -0
- package/packages/coworker-memory/src/scope-resolver.test.ts +23 -0
- package/packages/coworker-memory/src/scope-resolver.ts +18 -0
- package/packages/coworker-memory/src/types.ts +61 -0
- package/packages/coworker-memory/src/workspace-id.test.ts +48 -0
- package/packages/coworker-memory/src/workspace-id.ts +56 -0
- package/packages/coworker-memory/tsconfig.json +15 -0
- package/packages/coworker-memory/tsconfig.publish.json +4 -0
- package/packages/coworker-persona/dist/commands.d.ts +7 -0
- package/packages/coworker-persona/dist/commands.js +35 -0
- package/packages/coworker-persona/dist/defaults/manifest.yaml +12 -0
- package/packages/coworker-persona/dist/defaults/steering/identity.md +3 -0
- package/packages/coworker-persona/dist/index.d.ts +3 -0
- package/packages/coworker-persona/dist/index.js +3 -0
- package/packages/coworker-persona/dist/manifest.d.ts +24 -0
- package/packages/coworker-persona/dist/manifest.js +21 -0
- package/packages/coworker-persona/dist/registry.d.ts +22 -0
- package/packages/coworker-persona/dist/registry.js +142 -0
- package/packages/coworker-persona/package.json +28 -0
- package/packages/coworker-persona/scripts/copy-defaults.cjs +17 -0
- package/packages/coworker-persona/src/commands.ts +47 -0
- package/packages/coworker-persona/src/defaults/manifest.yaml +12 -0
- package/packages/coworker-persona/src/defaults/steering/identity.md +3 -0
- package/packages/coworker-persona/src/index.ts +3 -0
- package/packages/coworker-persona/src/manifest.test.ts +67 -0
- package/packages/coworker-persona/src/manifest.ts +49 -0
- package/packages/coworker-persona/src/registry.test.ts +89 -0
- package/packages/coworker-persona/src/registry.ts +147 -0
- package/packages/coworker-persona/tsconfig.json +15 -0
- package/packages/coworker-persona/tsconfig.publish.json +4 -0
- package/packages/coworker-scratchpad/dist/cell-archive.d.ts +39 -0
- package/packages/coworker-scratchpad/dist/cell-archive.js +77 -0
- package/packages/coworker-scratchpad/dist/cell-tree.d.ts +14 -0
- package/packages/coworker-scratchpad/dist/cell-tree.js +72 -0
- package/packages/coworker-scratchpad/dist/child-process-runtime.d.ts +129 -0
- package/packages/coworker-scratchpad/dist/child-process-runtime.js +427 -0
- package/packages/coworker-scratchpad/dist/collector-registry.d.ts +12 -0
- package/packages/coworker-scratchpad/dist/collector-registry.js +29 -0
- package/packages/coworker-scratchpad/dist/detect-kind.d.ts +3 -0
- package/packages/coworker-scratchpad/dist/detect-kind.js +19 -0
- package/packages/coworker-scratchpad/dist/file-collector.d.ts +15 -0
- package/packages/coworker-scratchpad/dist/file-collector.js +99 -0
- package/packages/coworker-scratchpad/dist/index.d.ts +13 -0
- package/packages/coworker-scratchpad/dist/index.js +13 -0
- package/packages/coworker-scratchpad/dist/kernel-bindings.d.ts +49 -0
- package/packages/coworker-scratchpad/dist/kernel-bindings.js +220 -0
- package/packages/coworker-scratchpad/dist/kernel-entry.d.ts +1 -0
- package/packages/coworker-scratchpad/dist/kernel-entry.js +355 -0
- package/packages/coworker-scratchpad/dist/kernel-protocol.d.ts +171 -0
- package/packages/coworker-scratchpad/dist/kernel-protocol.js +48 -0
- package/packages/coworker-scratchpad/dist/kernel-spawn.d.ts +3 -0
- package/packages/coworker-scratchpad/dist/kernel-spawn.js +54 -0
- package/packages/coworker-scratchpad/dist/namespace-codec.d.ts +22 -0
- package/packages/coworker-scratchpad/dist/namespace-codec.js +61 -0
- package/packages/coworker-scratchpad/dist/scratchpad-lock.d.ts +24 -0
- package/packages/coworker-scratchpad/dist/scratchpad-lock.js +86 -0
- package/packages/coworker-scratchpad/dist/scratchpad-manager.d.ts +193 -0
- package/packages/coworker-scratchpad/dist/scratchpad-manager.js +866 -0
- package/packages/coworker-scratchpad/dist/staleness-banner.d.ts +12 -0
- package/packages/coworker-scratchpad/dist/staleness-banner.js +27 -0
- package/packages/coworker-scratchpad/package.json +31 -0
- package/packages/coworker-scratchpad/src/cell-archive.test.ts +150 -0
- package/packages/coworker-scratchpad/src/cell-archive.ts +97 -0
- package/packages/coworker-scratchpad/src/cell-tree.test.ts +105 -0
- package/packages/coworker-scratchpad/src/cell-tree.ts +90 -0
- package/packages/coworker-scratchpad/src/child-process-runtime.test.ts +413 -0
- package/packages/coworker-scratchpad/src/child-process-runtime.ts +493 -0
- package/packages/coworker-scratchpad/src/collector-registry.test.ts +69 -0
- package/packages/coworker-scratchpad/src/collector-registry.ts +33 -0
- package/packages/coworker-scratchpad/src/detect-kind.test.ts +33 -0
- package/packages/coworker-scratchpad/src/detect-kind.ts +22 -0
- package/packages/coworker-scratchpad/src/file-collector.test.ts +109 -0
- package/packages/coworker-scratchpad/src/file-collector.ts +114 -0
- package/packages/coworker-scratchpad/src/index.ts +74 -0
- package/packages/coworker-scratchpad/src/kernel-bindings.test.ts +188 -0
- package/packages/coworker-scratchpad/src/kernel-bindings.ts +279 -0
- package/packages/coworker-scratchpad/src/kernel-entry.test.ts +123 -0
- package/packages/coworker-scratchpad/src/kernel-entry.ts +390 -0
- package/packages/coworker-scratchpad/src/kernel-protocol.test.ts +105 -0
- package/packages/coworker-scratchpad/src/kernel-protocol.ts +230 -0
- package/packages/coworker-scratchpad/src/kernel-spawn.test.ts +60 -0
- package/packages/coworker-scratchpad/src/kernel-spawn.ts +54 -0
- package/packages/coworker-scratchpad/src/namespace-codec.test.ts +102 -0
- package/packages/coworker-scratchpad/src/namespace-codec.ts +90 -0
- package/packages/coworker-scratchpad/src/scratchpad-lock.test.ts +98 -0
- package/packages/coworker-scratchpad/src/scratchpad-lock.ts +102 -0
- package/packages/coworker-scratchpad/src/scratchpad-manager.test.ts +1343 -0
- package/packages/coworker-scratchpad/src/scratchpad-manager.ts +891 -0
- package/packages/coworker-scratchpad/src/staleness-banner.test.ts +53 -0
- package/packages/coworker-scratchpad/src/staleness-banner.ts +33 -0
- package/packages/coworker-scratchpad/src/vault-integration.test.ts +221 -0
- package/packages/coworker-scratchpad/tsconfig.json +15 -0
- package/packages/coworker-scratchpad/tsconfig.publish.json +4 -0
- package/packages/coworker-types/dist/artifacts.d.ts +31 -0
- package/packages/coworker-types/dist/artifacts.js +2 -0
- package/packages/coworker-types/dist/contracts.d.ts +32 -0
- package/packages/coworker-types/dist/contracts.js +1 -0
- package/packages/coworker-types/dist/index.d.ts +5 -0
- package/packages/coworker-types/dist/index.js +5 -0
- package/packages/coworker-types/dist/memory.d.ts +61 -0
- package/packages/coworker-types/dist/memory.js +3 -0
- package/packages/coworker-types/dist/scratchpad.d.ts +43 -0
- package/packages/coworker-types/dist/scratchpad.js +2 -0
- package/packages/coworker-types/dist/vault.d.ts +34 -0
- package/packages/coworker-types/dist/vault.js +2 -0
- package/packages/coworker-types/package.json +24 -0
- package/packages/coworker-types/src/artifacts.test.ts +52 -0
- package/packages/coworker-types/src/artifacts.ts +35 -0
- package/packages/coworker-types/src/contracts.test.ts +43 -0
- package/packages/coworker-types/src/contracts.ts +36 -0
- package/packages/coworker-types/src/index.ts +5 -0
- package/packages/coworker-types/src/memory.test.ts +50 -0
- package/packages/coworker-types/src/memory.ts +79 -0
- package/packages/coworker-types/src/scratchpad.test.ts +46 -0
- package/packages/coworker-types/src/scratchpad.ts +51 -0
- package/packages/coworker-types/src/smoke.test.ts +34 -0
- package/packages/coworker-types/src/vault.test.ts +49 -0
- package/packages/coworker-types/src/vault.ts +40 -0
- package/packages/coworker-types/tsconfig.json +15 -0
- package/packages/coworker-types/tsconfig.publish.json +4 -0
- package/packages/coworker-utils/dist/audit-log.d.ts +34 -0
- package/packages/coworker-utils/dist/audit-log.js +88 -0
- package/packages/coworker-utils/dist/index.d.ts +6 -0
- package/packages/coworker-utils/dist/index.js +6 -0
- package/packages/coworker-utils/dist/lease.d.ts +7 -0
- package/packages/coworker-utils/dist/lease.js +67 -0
- package/packages/coworker-utils/dist/logger.d.ts +13 -0
- package/packages/coworker-utils/dist/logger.js +26 -0
- package/packages/coworker-utils/dist/migration-runner.d.ts +7 -0
- package/packages/coworker-utils/dist/migration-runner.js +36 -0
- package/packages/coworker-utils/dist/ndjson-channel.d.ts +3 -0
- package/packages/coworker-utils/dist/ndjson-channel.js +38 -0
- package/packages/coworker-utils/dist/secret-scanner.d.ts +10 -0
- package/packages/coworker-utils/dist/secret-scanner.js +42 -0
- package/packages/coworker-utils/package.json +24 -0
- package/packages/coworker-utils/src/audit-log.test.ts +140 -0
- package/packages/coworker-utils/src/audit-log.ts +107 -0
- package/packages/coworker-utils/src/index.ts +6 -0
- package/packages/coworker-utils/src/lease.test.ts +64 -0
- package/packages/coworker-utils/src/lease.ts +76 -0
- package/packages/coworker-utils/src/logger.test.ts +50 -0
- package/packages/coworker-utils/src/logger.ts +45 -0
- package/packages/coworker-utils/src/migration-runner.test.ts +65 -0
- package/packages/coworker-utils/src/migration-runner.ts +50 -0
- package/packages/coworker-utils/src/ndjson-channel.test.ts +76 -0
- package/packages/coworker-utils/src/ndjson-channel.ts +41 -0
- package/packages/coworker-utils/src/secret-scanner.test.ts +61 -0
- package/packages/coworker-utils/src/secret-scanner.ts +56 -0
- package/packages/coworker-utils/tsconfig.json +15 -0
- package/packages/coworker-utils/tsconfig.publish.json +4 -0
- package/packages/coworker-vault/dist/data-vault.d.ts +41 -0
- package/packages/coworker-vault/dist/data-vault.js +223 -0
- package/packages/coworker-vault/dist/engine-registry.d.ts +34 -0
- package/packages/coworker-vault/dist/engine-registry.js +90 -0
- package/packages/coworker-vault/dist/engines/jira.yaml +17 -0
- package/packages/coworker-vault/dist/errors.d.ts +28 -0
- package/packages/coworker-vault/dist/errors.js +57 -0
- package/packages/coworker-vault/dist/index.d.ts +6 -0
- package/packages/coworker-vault/dist/index.js +6 -0
- package/packages/coworker-vault/dist/injector.d.ts +19 -0
- package/packages/coworker-vault/dist/injector.js +77 -0
- package/packages/coworker-vault/dist/types.d.ts +28 -0
- package/packages/coworker-vault/dist/types.js +1 -0
- package/packages/coworker-vault/dist/vault-keep.d.ts +4 -0
- package/packages/coworker-vault/dist/vault-keep.js +21 -0
- package/packages/coworker-vault/package.json +29 -0
- package/packages/coworker-vault/src/data-vault.test.ts +199 -0
- package/packages/coworker-vault/src/data-vault.ts +257 -0
- package/packages/coworker-vault/src/engine-registry.test.ts +120 -0
- package/packages/coworker-vault/src/engine-registry.ts +107 -0
- package/packages/coworker-vault/src/engines/jira.yaml +17 -0
- package/packages/coworker-vault/src/errors.test.ts +58 -0
- package/packages/coworker-vault/src/errors.ts +50 -0
- package/packages/coworker-vault/src/index.test.ts +24 -0
- package/packages/coworker-vault/src/index.ts +6 -0
- package/packages/coworker-vault/src/injector.test.ts +109 -0
- package/packages/coworker-vault/src/injector.ts +98 -0
- package/packages/coworker-vault/src/types.ts +33 -0
- package/packages/coworker-vault/src/vault-keep.test.ts +49 -0
- package/packages/coworker-vault/src/vault-keep.ts +31 -0
- package/packages/coworker-vault/tsconfig.json +15 -0
- package/packages/coworker-vault/tsconfig.publish.json +4 -0
- package/packages/daemon/package.json +3 -3
- package/packages/mcp-server/package.json +3 -3
- package/packages/mcp-server/tsconfig.tsbuildinfo +1 -1
- package/packages/native/package.json +1 -1
- package/packages/native/tsconfig.tsbuildinfo +1 -1
- package/packages/pi-agent-core/package.json +1 -1
- package/packages/pi-agent-core/tsconfig.tsbuildinfo +1 -1
- package/packages/pi-ai/package.json +1 -1
- package/packages/pi-ai/tsconfig.tsbuildinfo +1 -1
- package/packages/pi-coding-agent/dist/core/extensions/runner.d.ts +6 -1
- package/packages/pi-coding-agent/dist/core/extensions/runner.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/extensions/runner.js +22 -3
- package/packages/pi-coding-agent/dist/core/extensions/runner.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/resolve-config-value.test.js +11 -0
- package/packages/pi-coding-agent/dist/core/resolve-config-value.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/rpc/raw-stdout.d.ts +47 -0
- package/packages/pi-coding-agent/dist/modes/rpc/raw-stdout.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/modes/rpc/raw-stdout.js +107 -0
- package/packages/pi-coding-agent/dist/modes/rpc/raw-stdout.js.map +1 -0
- package/packages/pi-coding-agent/dist/modes/rpc/raw-stdout.regression.test.d.ts +19 -0
- package/packages/pi-coding-agent/dist/modes/rpc/raw-stdout.regression.test.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/modes/rpc/raw-stdout.regression.test.js +121 -0
- package/packages/pi-coding-agent/dist/modes/rpc/raw-stdout.regression.test.js.map +1 -0
- package/packages/pi-coding-agent/dist/modes/rpc/rpc-mode.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/rpc/rpc-mode.js +17 -1
- package/packages/pi-coding-agent/dist/modes/rpc/rpc-mode.js.map +1 -1
- package/packages/pi-coding-agent/package.json +2 -2
- package/packages/pi-coding-agent/src/core/extensions/runner.ts +22 -3
- package/packages/pi-coding-agent/src/core/resolve-config-value.test.ts +11 -0
- package/packages/pi-coding-agent/src/modes/rpc/raw-stdout.regression.test.ts +129 -0
- package/packages/pi-coding-agent/src/modes/rpc/raw-stdout.ts +117 -0
- package/packages/pi-coding-agent/src/modes/rpc/rpc-mode.ts +18 -1
- package/packages/pi-coding-agent/tsconfig.tsbuildinfo +1 -1
- package/packages/pi-tui/package.json +1 -1
- package/packages/pi-tui/tsconfig.tsbuildinfo +1 -1
- package/packages/rpc-client/package.json +2 -2
- package/packages/rpc-client/tsconfig.tsbuildinfo +1 -1
- package/pkg/package.json +1 -1
- package/scripts/install.js +6 -5
- package/src/resources/extensions/_coworker-paths.test.ts +40 -0
- package/src/resources/extensions/_coworker-paths.ts +10 -0
- package/src/resources/extensions/coworker-artifacts/artifacts-command.test.ts +54 -0
- package/src/resources/extensions/coworker-artifacts/artifacts-command.ts +43 -0
- package/src/resources/extensions/coworker-artifacts/artifacts-singleton.test.ts +25 -0
- package/src/resources/extensions/coworker-artifacts/artifacts-singleton.ts +29 -0
- package/src/resources/extensions/coworker-artifacts/extension-manifest.json +13 -0
- package/src/resources/extensions/coworker-artifacts/index.test.ts +46 -0
- package/src/resources/extensions/coworker-artifacts/index.ts +154 -0
- package/src/resources/extensions/coworker-artifacts/list-tool.test.ts +29 -0
- package/src/resources/extensions/coworker-artifacts/list-tool.ts +53 -0
- package/src/resources/extensions/coworker-artifacts/open-tool.test.ts +30 -0
- package/src/resources/extensions/coworker-artifacts/open-tool.ts +43 -0
- package/src/resources/extensions/coworker-memory/extension-manifest.json +13 -0
- package/src/resources/extensions/coworker-memory/index.test.ts +137 -0
- package/src/resources/extensions/coworker-memory/index.ts +257 -0
- package/src/resources/extensions/coworker-memory/memorize-tool.test.ts +41 -0
- package/src/resources/extensions/coworker-memory/memorize-tool.ts +20 -0
- package/src/resources/extensions/coworker-memory/memory-command.test.ts +134 -0
- package/src/resources/extensions/coworker-memory/memory-command.ts +131 -0
- package/src/resources/extensions/coworker-memory/memory-singleton.test.ts +41 -0
- package/src/resources/extensions/coworker-memory/memory-singleton.ts +89 -0
- package/src/resources/extensions/coworker-memory/recall-tool.test.ts +50 -0
- package/src/resources/extensions/coworker-memory/recall-tool.ts +35 -0
- package/src/resources/extensions/coworker-memory/session-hooks.test.ts +77 -0
- package/src/resources/extensions/coworker-memory/session-hooks.ts +61 -0
- package/src/resources/extensions/coworker-scratchpad/attach-banners.test.ts +124 -0
- package/src/resources/extensions/coworker-scratchpad/attach-banners.ts +67 -0
- package/src/resources/extensions/coworker-scratchpad/extension-manifest.json +13 -0
- package/src/resources/extensions/coworker-scratchpad/format-age.test.ts +30 -0
- package/src/resources/extensions/coworker-scratchpad/format-age.ts +6 -0
- package/src/resources/extensions/coworker-scratchpad/helpers.test.ts +93 -0
- package/src/resources/extensions/coworker-scratchpad/helpers.ts +42 -0
- package/src/resources/extensions/coworker-scratchpad/index.test.ts +514 -0
- package/src/resources/extensions/coworker-scratchpad/index.ts +207 -0
- package/src/resources/extensions/coworker-scratchpad/mime-bundle.test.ts +61 -0
- package/src/resources/extensions/coworker-scratchpad/mime-bundle.ts +23 -0
- package/src/resources/extensions/coworker-scratchpad/scratchpad-tool.test.ts +137 -0
- package/src/resources/extensions/coworker-scratchpad/scratchpad-tool.ts +165 -0
- package/src/resources/extensions/coworker-scratchpad/session-sidecar.test.ts +133 -0
- package/src/resources/extensions/coworker-scratchpad/session-sidecar.ts +68 -0
- package/src/resources/extensions/coworker-scratchpad/sp-command.test.ts +836 -0
- package/src/resources/extensions/coworker-scratchpad/sp-command.ts +602 -0
- package/src/resources/extensions/coworker-scratchpad/workspace-pointer.test.ts +74 -0
- package/src/resources/extensions/coworker-scratchpad/workspace-pointer.ts +55 -0
- package/src/resources/extensions/coworker-scratchpad/workspace-root.test.ts +51 -0
- package/src/resources/extensions/coworker-scratchpad/workspace-root.ts +16 -0
- package/src/resources/extensions/coworker-vault/audit-command.test.ts +109 -0
- package/src/resources/extensions/coworker-vault/audit-command.ts +56 -0
- package/src/resources/extensions/coworker-vault/connect-command.test.ts +103 -0
- package/src/resources/extensions/coworker-vault/connect-command.ts +69 -0
- package/src/resources/extensions/coworker-vault/datasource-command.test.ts +80 -0
- package/src/resources/extensions/coworker-vault/datasource-command.ts +81 -0
- package/src/resources/extensions/coworker-vault/extension-manifest.json +12 -0
- package/src/resources/extensions/coworker-vault/index.test.ts +82 -0
- package/src/resources/extensions/coworker-vault/index.ts +181 -0
- package/src/resources/extensions/coworker-vault/test-helpers.ts +120 -0
- package/src/resources/extensions/coworker-vault/vault-singleton.test.ts +27 -0
- package/src/resources/extensions/coworker-vault/vault-singleton.ts +40 -0
- package/src/resources/extensions/otto/commands/release-notes/_data.ts +85 -0
- package/src/resources/extensions/otto/commands/release-notes/command.ts +16 -3
- package/src/resources/extensions/subagent/index.ts +9 -0
- package/src/resources/extensions/subagent/launch.test.ts +97 -0
- package/src/resources/extensions/subagent/launch.ts +42 -5
- package/src/resources/extensions/subagent/run-store.ts +3 -1
- package/src/resources/extensions/workflow/bootstrap/register-extension.ts +2 -0
- package/src/resources/extensions/workflow/bootstrap/register-hooks.ts +10 -0
- package/src/resources/extensions/workflow/persona-status.ts +109 -0
- package/src/resources/extensions/workflow/tests/auto-recovery.test.ts +34 -0
|
@@ -0,0 +1,131 @@
|
|
|
1
|
+
// src/resources/extensions/coworker-memory/memory-command.ts
|
|
2
|
+
import type { MemoryBundle } from './memory-singleton.js';
|
|
3
|
+
import { runMemorize } from './memorize-tool.js';
|
|
4
|
+
import { runRecall } from './recall-tool.js';
|
|
5
|
+
|
|
6
|
+
export interface MemoryCommandResult {
|
|
7
|
+
message: string;
|
|
8
|
+
}
|
|
9
|
+
|
|
10
|
+
export async function runMemoryCommand(bundle: MemoryBundle, argv: string[]): Promise<MemoryCommandResult> {
|
|
11
|
+
const [sub, ...rest] = argv;
|
|
12
|
+
switch (sub) {
|
|
13
|
+
case 'note': {
|
|
14
|
+
const text = rest.join(' ').trim();
|
|
15
|
+
if (!text) throw new Error('Usage: /memory note <text>');
|
|
16
|
+
await runMemorize(bundle, { text, kind: 'lesson', scope: 'workspace' });
|
|
17
|
+
return { message: `lesson stored in workspace.` };
|
|
18
|
+
}
|
|
19
|
+
case 'status': {
|
|
20
|
+
const status = await bundle.backend.status();
|
|
21
|
+
return {
|
|
22
|
+
message: [
|
|
23
|
+
`scope_mode: ${bundle.scopeMode}`,
|
|
24
|
+
`workspace_wing: ${bundle.workspaceWing}`,
|
|
25
|
+
`drawer_count: ${status.drawer_count}`,
|
|
26
|
+
`layer_b_db_path: ${status.layer_b_db_path}`,
|
|
27
|
+
`schema_version: ${status.schema_version}`,
|
|
28
|
+
].join('\n'),
|
|
29
|
+
};
|
|
30
|
+
}
|
|
31
|
+
case 'clear': {
|
|
32
|
+
let wing: string | undefined;
|
|
33
|
+
let confirm = false;
|
|
34
|
+
for (let i = 0; i < rest.length; i++) {
|
|
35
|
+
if (rest[i] === '--wing' && rest[i+1]) { wing = rest[++i]; }
|
|
36
|
+
else if (rest[i] === '--confirm') { confirm = true; }
|
|
37
|
+
}
|
|
38
|
+
if (!confirm) throw new Error('Usage: /memory clear --wing <wing> --confirm');
|
|
39
|
+
const out = await bundle.backend.clear({ wing, confirm: true });
|
|
40
|
+
return { message: `deleted: ${out.deleted}` };
|
|
41
|
+
}
|
|
42
|
+
case 'wing':
|
|
43
|
+
case 'room': {
|
|
44
|
+
// Session overrides — caller manages state; this is a placeholder return acknowledging.
|
|
45
|
+
const target = rest.join(' ').trim();
|
|
46
|
+
if (!target) throw new Error(`Usage: /memory ${sub} <name>`);
|
|
47
|
+
return { message: `${sub} override: ${target}` };
|
|
48
|
+
}
|
|
49
|
+
case 'seed': {
|
|
50
|
+
// Caller wires this to applyPersonaSeed via session-hooks; here we just acknowledge the request.
|
|
51
|
+
return { message: 're-seed will run on next session_start; flip workspace.json.memory_seed_applied=false and reattach.' };
|
|
52
|
+
}
|
|
53
|
+
case 'show': {
|
|
54
|
+
let target: 'all' | 'profile' | 'rule' | 'lesson' = 'all';
|
|
55
|
+
let scopeOverride: 'workspace' | 'global' | null = null;
|
|
56
|
+
for (let i = 0; i < rest.length; i++) {
|
|
57
|
+
const t = rest[i];
|
|
58
|
+
if (t === '--scope' && rest[i + 1]) {
|
|
59
|
+
const s = rest[++i];
|
|
60
|
+
if (s !== 'workspace' && s !== 'global') throw new Error(`Usage: /memory show [profile|rules|lessons] [--scope workspace|global]`);
|
|
61
|
+
scopeOverride = s;
|
|
62
|
+
} else if (t === 'profile') { target = 'profile'; }
|
|
63
|
+
else if (t === 'rule' || t === 'rules') { target = 'rule'; }
|
|
64
|
+
else if (t === 'lesson' || t === 'lessons') { target = 'lesson'; }
|
|
65
|
+
else if (t === 'all') { target = 'all'; }
|
|
66
|
+
else { throw new Error(`Usage: /memory show [profile|rules|lessons] [--scope workspace|global]`); }
|
|
67
|
+
}
|
|
68
|
+
const kinds: Array<'profile' | 'rule' | 'lesson'> =
|
|
69
|
+
target === 'all' ? ['profile', 'rule', 'lesson'] : [target];
|
|
70
|
+
// Determine scopes: --scope override wins; otherwise derive from bundle.scopeMode.
|
|
71
|
+
const scopes: Array<{ name: 'workspace' | 'global'; store: typeof bundle.workspaceLayerA }> = [];
|
|
72
|
+
if (scopeOverride === 'workspace') {
|
|
73
|
+
scopes.push({ name: 'workspace', store: bundle.workspaceLayerA });
|
|
74
|
+
} else if (scopeOverride === 'global') {
|
|
75
|
+
scopes.push({ name: 'global', store: bundle.globalLayerA });
|
|
76
|
+
} else if (bundle.scopeMode === 'global') {
|
|
77
|
+
scopes.push({ name: 'global', store: bundle.globalLayerA });
|
|
78
|
+
} else if (bundle.scopeMode === 'per-project') {
|
|
79
|
+
scopes.push({ name: 'workspace', store: bundle.workspaceLayerA });
|
|
80
|
+
} else {
|
|
81
|
+
// per-project-tagged: workspace first, then global
|
|
82
|
+
scopes.push({ name: 'workspace', store: bundle.workspaceLayerA });
|
|
83
|
+
scopes.push({ name: 'global', store: bundle.globalLayerA });
|
|
84
|
+
}
|
|
85
|
+
const TITLE_FOR = { profile: 'Profile', rule: 'Rules', lesson: 'Lessons' } as const;
|
|
86
|
+
const sections: string[] = [];
|
|
87
|
+
for (const k of kinds) {
|
|
88
|
+
for (const scope of scopes) {
|
|
89
|
+
const body = await scope.store.read(k);
|
|
90
|
+
sections.push(`## ${TITLE_FOR[k]} (${scope.name})\n${body || '(none)'}`);
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
return { message: sections.join('\n\n') };
|
|
94
|
+
}
|
|
95
|
+
case 'recall': {
|
|
96
|
+
// First flag/positional split: gather positional tokens for the query;
|
|
97
|
+
// recognize --kind, --room, --wing, --limit, --days_back.
|
|
98
|
+
const flags: { kind?: string; room?: string; wing?: string; limit?: number; days_back?: number } = {};
|
|
99
|
+
const queryTokens: string[] = [];
|
|
100
|
+
for (let i = 0; i < rest.length; i++) {
|
|
101
|
+
const t = rest[i];
|
|
102
|
+
if (t === '--kind' && rest[i + 1]) { flags.kind = rest[++i]; }
|
|
103
|
+
else if (t === '--room' && rest[i + 1]) { flags.room = rest[++i]; }
|
|
104
|
+
else if (t === '--wing' && rest[i + 1]) { flags.wing = rest[++i]; }
|
|
105
|
+
else if (t === '--limit' && rest[i + 1]) {
|
|
106
|
+
const n = parseInt(rest[++i]!, 10);
|
|
107
|
+
if (Number.isFinite(n)) flags.limit = n;
|
|
108
|
+
} else if (t === '--days_back' && rest[i + 1]) {
|
|
109
|
+
const n = parseInt(rest[++i]!, 10);
|
|
110
|
+
if (Number.isFinite(n)) flags.days_back = n;
|
|
111
|
+
} else {
|
|
112
|
+
queryTokens.push(t!);
|
|
113
|
+
}
|
|
114
|
+
}
|
|
115
|
+
const query = queryTokens.join(' ').trim();
|
|
116
|
+
if (!query) throw new Error('Usage: /memory recall <query> [--kind <k>] [--room <r>] [--wing <w>] [--limit N] [--days_back N]');
|
|
117
|
+
// Defer to the same runRecall the LLM tool uses for parity.
|
|
118
|
+
const out = await runRecall(bundle, {
|
|
119
|
+
query,
|
|
120
|
+
kind: flags.kind as never, // runRecall accepts string; backend validates
|
|
121
|
+
room: flags.room,
|
|
122
|
+
wing: flags.wing,
|
|
123
|
+
max_results: flags.limit,
|
|
124
|
+
days_back: flags.days_back,
|
|
125
|
+
});
|
|
126
|
+
return { message: out.markdown };
|
|
127
|
+
}
|
|
128
|
+
default:
|
|
129
|
+
throw new Error(`Unknown /memory subcommand: ${sub}. Try: note, status, clear, wing, room, seed, show, recall.`);
|
|
130
|
+
}
|
|
131
|
+
}
|
|
@@ -0,0 +1,41 @@
|
|
|
1
|
+
// src/resources/extensions/coworker-memory/memory-singleton.test.ts
|
|
2
|
+
import { describe, it } from 'node:test';
|
|
3
|
+
import assert from 'node:assert/strict';
|
|
4
|
+
import { mkdtempSync, mkdirSync } from 'node:fs';
|
|
5
|
+
import { tmpdir } from 'node:os';
|
|
6
|
+
import { join } from 'node:path';
|
|
7
|
+
import { createMemoryBundle } from './memory-singleton.js';
|
|
8
|
+
|
|
9
|
+
describe('memory singleton bundle', () => {
|
|
10
|
+
it('constructs scope-aware bundle with all stores', async () => {
|
|
11
|
+
const home = mkdtempSync(join(tmpdir(), 'mem-home-'));
|
|
12
|
+
const ws = mkdtempSync(join(tmpdir(), 'mem-ws-'));
|
|
13
|
+
mkdirSync(ws, { recursive: true });
|
|
14
|
+
const bundle = await createMemoryBundle({
|
|
15
|
+
globalDir: home, workspaceDir: ws,
|
|
16
|
+
scopeMode: 'per-project-tagged',
|
|
17
|
+
currentScratchpadName: () => null,
|
|
18
|
+
});
|
|
19
|
+
assert.ok(bundle.globalLayerA);
|
|
20
|
+
assert.ok(bundle.workspaceLayerA);
|
|
21
|
+
assert.ok(bundle.backend);
|
|
22
|
+
assert.ok(bundle.recorder);
|
|
23
|
+
assert.equal(bundle.scopeMode, 'per-project-tagged');
|
|
24
|
+
assert.match(bundle.workspaceWing, /-[0-9a-f]{6}$/);
|
|
25
|
+
assert.equal(bundle.writeWing, bundle.workspaceWing);
|
|
26
|
+
assert.deepEqual(bundle.readWings, [bundle.workspaceWing, 'global']);
|
|
27
|
+
await bundle.dispose();
|
|
28
|
+
});
|
|
29
|
+
it('global mode bundle uses wing global', async () => {
|
|
30
|
+
const home = mkdtempSync(join(tmpdir(), 'mem-home-g-'));
|
|
31
|
+
const ws = mkdtempSync(join(tmpdir(), 'mem-ws-g-'));
|
|
32
|
+
const bundle = await createMemoryBundle({
|
|
33
|
+
globalDir: home, workspaceDir: ws,
|
|
34
|
+
scopeMode: 'global',
|
|
35
|
+
currentScratchpadName: () => null,
|
|
36
|
+
});
|
|
37
|
+
assert.equal(bundle.writeWing, 'global');
|
|
38
|
+
assert.deepEqual(bundle.readWings, ['global']);
|
|
39
|
+
await bundle.dispose();
|
|
40
|
+
});
|
|
41
|
+
});
|
|
@@ -0,0 +1,89 @@
|
|
|
1
|
+
// src/resources/extensions/coworker-memory/memory-singleton.ts
|
|
2
|
+
//
|
|
3
|
+
// Factory that constructs the layered memory collaborators (LayerAStore x2,
|
|
4
|
+
// LocalSqliteBackend, MemoryRecorder) plus shared audit + secret scanner for
|
|
5
|
+
// given roots. Mirrors the coworker-vault singleton pattern. Pure beyond what
|
|
6
|
+
// the underlying constructors already perform (dir creation, db open, etc.).
|
|
7
|
+
import { join } from 'node:path';
|
|
8
|
+
import { AuditLog, SecretScanner } from '@otto/coworker-utils';
|
|
9
|
+
import {
|
|
10
|
+
LayerAStore,
|
|
11
|
+
LocalSqliteBackend,
|
|
12
|
+
MemoryRecorder,
|
|
13
|
+
resolveScope,
|
|
14
|
+
resolveWorkspaceId,
|
|
15
|
+
type ScopeMode,
|
|
16
|
+
type Wing,
|
|
17
|
+
type CurrentScratchpadProvider,
|
|
18
|
+
type WorkspaceIdRecord,
|
|
19
|
+
} from '@otto/coworker-memory';
|
|
20
|
+
|
|
21
|
+
export interface MemoryBundleOptions {
|
|
22
|
+
globalDir: string;
|
|
23
|
+
workspaceDir: string;
|
|
24
|
+
scopeMode: ScopeMode;
|
|
25
|
+
currentScratchpadName: CurrentScratchpadProvider;
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
export interface MemoryBundle {
|
|
29
|
+
globalLayerA: LayerAStore;
|
|
30
|
+
workspaceLayerA: LayerAStore;
|
|
31
|
+
backend: LocalSqliteBackend;
|
|
32
|
+
recorder: MemoryRecorder;
|
|
33
|
+
audit: AuditLog;
|
|
34
|
+
scanner: SecretScanner;
|
|
35
|
+
workspaceWing: Wing;
|
|
36
|
+
writeWing: Wing;
|
|
37
|
+
readWings: Wing[];
|
|
38
|
+
scopeMode: ScopeMode;
|
|
39
|
+
workspaceRecord: WorkspaceIdRecord;
|
|
40
|
+
workspaceDir: string;
|
|
41
|
+
dispose(): Promise<void>;
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
export async function createMemoryBundle(opts: MemoryBundleOptions): Promise<MemoryBundle> {
|
|
45
|
+
const audit = new AuditLog({ path: join(opts.globalDir, 'audit.jsonl') });
|
|
46
|
+
const scanner = new SecretScanner();
|
|
47
|
+
const wsRecord = await resolveWorkspaceId(opts.workspaceDir);
|
|
48
|
+
const scope = resolveScope({ mode: opts.scopeMode, workspaceId: wsRecord.id });
|
|
49
|
+
const globalLayerA = new LayerAStore({
|
|
50
|
+
scopeDir: join(opts.globalDir, 'memory'),
|
|
51
|
+
scope: 'global',
|
|
52
|
+
audit,
|
|
53
|
+
scanner,
|
|
54
|
+
});
|
|
55
|
+
const workspaceLayerA = new LayerAStore({
|
|
56
|
+
scopeDir: join(opts.workspaceDir, '.otto', 'memory'),
|
|
57
|
+
scope: 'workspace',
|
|
58
|
+
audit,
|
|
59
|
+
scanner,
|
|
60
|
+
});
|
|
61
|
+
const backend = new LocalSqliteBackend({
|
|
62
|
+
dbPath: join(opts.workspaceDir, '.otto', 'memory', 'layer-b.db'),
|
|
63
|
+
});
|
|
64
|
+
await backend.open();
|
|
65
|
+
const recorder = new MemoryRecorder({
|
|
66
|
+
backend,
|
|
67
|
+
scanner,
|
|
68
|
+
audit,
|
|
69
|
+
writeWing: scope.writeWing,
|
|
70
|
+
currentScratchpadName: opts.currentScratchpadName,
|
|
71
|
+
});
|
|
72
|
+
return {
|
|
73
|
+
globalLayerA,
|
|
74
|
+
workspaceLayerA,
|
|
75
|
+
backend,
|
|
76
|
+
recorder,
|
|
77
|
+
audit,
|
|
78
|
+
scanner,
|
|
79
|
+
workspaceWing: wsRecord.id,
|
|
80
|
+
writeWing: scope.writeWing,
|
|
81
|
+
readWings: scope.readWings,
|
|
82
|
+
scopeMode: opts.scopeMode,
|
|
83
|
+
workspaceRecord: wsRecord,
|
|
84
|
+
workspaceDir: opts.workspaceDir,
|
|
85
|
+
async dispose() {
|
|
86
|
+
await backend.close();
|
|
87
|
+
},
|
|
88
|
+
};
|
|
89
|
+
}
|
|
@@ -0,0 +1,50 @@
|
|
|
1
|
+
// src/resources/extensions/coworker-memory/recall-tool.test.ts
|
|
2
|
+
import { describe, it } from 'node:test';
|
|
3
|
+
import assert from 'node:assert/strict';
|
|
4
|
+
import { mkdtempSync, mkdirSync } from 'node:fs';
|
|
5
|
+
import { tmpdir } from 'node:os';
|
|
6
|
+
import { join } from 'node:path';
|
|
7
|
+
import { createMemoryBundle } from './memory-singleton.js';
|
|
8
|
+
import { runRecall } from './recall-tool.js';
|
|
9
|
+
|
|
10
|
+
async function setup() {
|
|
11
|
+
const home = mkdtempSync(join(tmpdir(), 'rt-home-'));
|
|
12
|
+
const ws = mkdtempSync(join(tmpdir(), 'rt-ws-'));
|
|
13
|
+
mkdirSync(ws, { recursive: true });
|
|
14
|
+
return createMemoryBundle({
|
|
15
|
+
globalDir: home, workspaceDir: ws,
|
|
16
|
+
scopeMode: 'per-project-tagged', currentScratchpadName: () => null,
|
|
17
|
+
});
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
describe('recall tool', () => {
|
|
21
|
+
it('returns results with markdown rendering', async () => {
|
|
22
|
+
const b = await setup();
|
|
23
|
+
await b.recorder.recordTurn({ sessionId: 's', userText: 'customer paste about load balancer', turnId: 't1' });
|
|
24
|
+
const r = await runRecall(b, { query: 'load balancer' });
|
|
25
|
+
assert.equal(r.results.length, 1);
|
|
26
|
+
assert.match(r.markdown, /Memory recall \(1 matches\)/);
|
|
27
|
+
assert.match(r.markdown, /drawer:\/\//);
|
|
28
|
+
await b.dispose();
|
|
29
|
+
});
|
|
30
|
+
it('honors max_results clamp 1..64', async () => {
|
|
31
|
+
const b = await setup();
|
|
32
|
+
for (let i = 0; i < 100; i++) {
|
|
33
|
+
await b.recorder.recordTurn({ sessionId: 's', userText: `apple ${i}`, turnId: `t${i}` });
|
|
34
|
+
}
|
|
35
|
+
const big = await runRecall(b, { query: 'apple', max_results: 200 });
|
|
36
|
+
assert.ok(big.results.length <= 64);
|
|
37
|
+
const small = await runRecall(b, { query: 'apple', max_results: 0 });
|
|
38
|
+
assert.equal(small.results.length, 1);
|
|
39
|
+
await b.dispose();
|
|
40
|
+
});
|
|
41
|
+
it('filters by kind', async () => {
|
|
42
|
+
const b = await setup();
|
|
43
|
+
await b.recorder.recordTurn({ sessionId: 's', userText: 'short alpha', turnId: 't1' });
|
|
44
|
+
await b.recorder.recordPaste({ sessionId: 's', content: 'long alpha paste', turnId: 't2' });
|
|
45
|
+
const onlyPaste = await runRecall(b, { query: 'alpha', kind: 'paste' });
|
|
46
|
+
assert.equal(onlyPaste.results.length, 1);
|
|
47
|
+
assert.equal(onlyPaste.results[0]!.drawer.kind, 'paste');
|
|
48
|
+
await b.dispose();
|
|
49
|
+
});
|
|
50
|
+
});
|
|
@@ -0,0 +1,35 @@
|
|
|
1
|
+
// src/resources/extensions/coworker-memory/recall-tool.ts
|
|
2
|
+
import type { MemoryBundle } from './memory-singleton.js';
|
|
3
|
+
import { formatRecall, type RecallQuery, type RecallResult } from '@otto/coworker-memory';
|
|
4
|
+
|
|
5
|
+
export interface RecallToolArgs {
|
|
6
|
+
query: string;
|
|
7
|
+
kind?: RecallQuery['kind'];
|
|
8
|
+
wing?: string;
|
|
9
|
+
room?: string;
|
|
10
|
+
days_back?: number;
|
|
11
|
+
max_results?: number;
|
|
12
|
+
}
|
|
13
|
+
|
|
14
|
+
export interface RecallToolOutput {
|
|
15
|
+
results: RecallResult[];
|
|
16
|
+
markdown: string;
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
export async function runRecall(bundle: MemoryBundle, args: RecallToolArgs): Promise<RecallToolOutput> {
|
|
20
|
+
const wings = args.wing ? [args.wing, ...bundle.readWings.filter(w => w !== args.wing)] : bundle.readWings;
|
|
21
|
+
const limit = args.max_results === undefined ? 8 : Math.min(Math.max(args.max_results, 1), 64);
|
|
22
|
+
const results = await bundle.backend.recall({
|
|
23
|
+
query: args.query, wing: wings, room: args.room, kind: args.kind,
|
|
24
|
+
days_back: args.days_back, max_results: limit,
|
|
25
|
+
});
|
|
26
|
+
bundle.audit.append({
|
|
27
|
+
_schema: 1, ts: new Date().toISOString(), producer: 'memory', action: 'recall',
|
|
28
|
+
detail: {
|
|
29
|
+
wing_filter: wings, room_filter: args.room ?? null,
|
|
30
|
+
kind_filter: args.kind ?? null, days_back: args.days_back ?? null,
|
|
31
|
+
result_count: results.length,
|
|
32
|
+
},
|
|
33
|
+
});
|
|
34
|
+
return { results, markdown: formatRecall(results) };
|
|
35
|
+
}
|
|
@@ -0,0 +1,77 @@
|
|
|
1
|
+
// src/resources/extensions/coworker-memory/session-hooks.test.ts
|
|
2
|
+
import { describe, it } from 'node:test';
|
|
3
|
+
import assert from 'node:assert/strict';
|
|
4
|
+
import { mkdtempSync, mkdirSync, writeFileSync } from 'node:fs';
|
|
5
|
+
import { tmpdir } from 'node:os';
|
|
6
|
+
import { join } from 'node:path';
|
|
7
|
+
import { createMemoryBundle } from './memory-singleton.js';
|
|
8
|
+
import { onSessionStart, onSessionShutdown } from './session-hooks.js';
|
|
9
|
+
|
|
10
|
+
async function setup() {
|
|
11
|
+
const home = mkdtempSync(join(tmpdir(), 'sh-home-'));
|
|
12
|
+
const ws = mkdtempSync(join(tmpdir(), 'sh-ws-'));
|
|
13
|
+
mkdirSync(ws, { recursive: true });
|
|
14
|
+
return {
|
|
15
|
+
home,
|
|
16
|
+
ws,
|
|
17
|
+
bundle: await createMemoryBundle({
|
|
18
|
+
globalDir: home,
|
|
19
|
+
workspaceDir: ws,
|
|
20
|
+
scopeMode: 'per-project-tagged',
|
|
21
|
+
currentScratchpadName: () => null,
|
|
22
|
+
}),
|
|
23
|
+
};
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
describe('session hooks', () => {
|
|
27
|
+
it('onSessionStart returns Layer A context block', async () => {
|
|
28
|
+
const c = await setup();
|
|
29
|
+
await c.bundle.workspaceLayerA.append({
|
|
30
|
+
kind: 'lesson',
|
|
31
|
+
text: 'do not deploy on friday',
|
|
32
|
+
source: 'user',
|
|
33
|
+
ts: '2026-06-02T00:00:00Z',
|
|
34
|
+
});
|
|
35
|
+
const out = await onSessionStart(c.bundle, { tokenLimit: 3000 });
|
|
36
|
+
assert.match(out.contextBlock, /Memory \(Layer A\)/);
|
|
37
|
+
assert.match(out.contextBlock, /do not deploy on friday/);
|
|
38
|
+
await c.bundle.dispose();
|
|
39
|
+
});
|
|
40
|
+
it('onSessionStart applies persona seed when pending', async () => {
|
|
41
|
+
const c = await setup();
|
|
42
|
+
const personaDir = mkdtempSync(join(tmpdir(), 'persona-'));
|
|
43
|
+
const seedDir = join(personaDir, 'memory-seed');
|
|
44
|
+
mkdirSync(seedDir, { recursive: true });
|
|
45
|
+
writeFileSync(join(seedDir, 'rules.md'), 'Persona rule baseline');
|
|
46
|
+
const out = await onSessionStart(c.bundle, {
|
|
47
|
+
tokenLimit: 3000,
|
|
48
|
+
persona: { id: 'noc-ops', personaDir },
|
|
49
|
+
});
|
|
50
|
+
assert.deepEqual(out.seed.copied, ['rules.md']);
|
|
51
|
+
assert.equal(c.bundle.workspaceRecord.memory_seed_applied, true);
|
|
52
|
+
assert.equal(c.bundle.workspaceRecord.memory_seed_persona, 'noc-ops');
|
|
53
|
+
await c.bundle.dispose();
|
|
54
|
+
});
|
|
55
|
+
it('onSessionStart does not re-apply seed once flag is true', async () => {
|
|
56
|
+
const c = await setup();
|
|
57
|
+
const personaDir = mkdtempSync(join(tmpdir(), 'persona2-'));
|
|
58
|
+
const seedDir = join(personaDir, 'memory-seed');
|
|
59
|
+
mkdirSync(seedDir, { recursive: true });
|
|
60
|
+
writeFileSync(join(seedDir, 'rules.md'), 'Persona rule v1');
|
|
61
|
+
await onSessionStart(c.bundle, { tokenLimit: 3000, persona: { id: 'noc-ops', personaDir } });
|
|
62
|
+
// Now change the seed file but the flag is set.
|
|
63
|
+
writeFileSync(join(seedDir, 'rules.md'), 'Persona rule v2');
|
|
64
|
+
const second = await onSessionStart(c.bundle, {
|
|
65
|
+
tokenLimit: 3000,
|
|
66
|
+
persona: { id: 'noc-ops', personaDir },
|
|
67
|
+
});
|
|
68
|
+
assert.deepEqual(second.seed.copied, []);
|
|
69
|
+
await c.bundle.dispose();
|
|
70
|
+
});
|
|
71
|
+
it('onSessionShutdown closes backend without throwing', async () => {
|
|
72
|
+
const c = await setup();
|
|
73
|
+
await onSessionShutdown(c.bundle);
|
|
74
|
+
// Bundle is now disposed; second close should be safe.
|
|
75
|
+
await c.bundle.dispose();
|
|
76
|
+
});
|
|
77
|
+
});
|
|
@@ -0,0 +1,61 @@
|
|
|
1
|
+
// src/resources/extensions/coworker-memory/session-hooks.ts
|
|
2
|
+
//
|
|
3
|
+
// Session lifecycle seams for the coworker-memory extension.
|
|
4
|
+
// - onSessionStart: applies a one-shot persona seed (if pending) and returns
|
|
5
|
+
// the Layer A context block ready for system-prompt injection.
|
|
6
|
+
// - onSessionShutdown: closes the backend (Layer B). Safe to double-call —
|
|
7
|
+
// LocalSqliteBackend.close() nulls its db handle so a second close is a no-op.
|
|
8
|
+
import { buildLayerAContext, applyPersonaSeed, writeWorkspaceId } from '@otto/coworker-memory';
|
|
9
|
+
import type { MemoryBundle } from './memory-singleton.js';
|
|
10
|
+
|
|
11
|
+
export interface SessionStartOptions {
|
|
12
|
+
tokenLimit?: number;
|
|
13
|
+
persona?: { id: string; personaDir: string };
|
|
14
|
+
}
|
|
15
|
+
|
|
16
|
+
export interface SessionStartResult {
|
|
17
|
+
contextBlock: string;
|
|
18
|
+
seed: { copied: string[]; blocked: string[] };
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
export async function onSessionStart(
|
|
22
|
+
bundle: MemoryBundle,
|
|
23
|
+
opts: SessionStartOptions = {},
|
|
24
|
+
): Promise<SessionStartResult> {
|
|
25
|
+
let seed = { copied: [] as string[], blocked: [] as string[] };
|
|
26
|
+
if (opts.persona && !bundle.workspaceRecord.memory_seed_applied) {
|
|
27
|
+
seed = await applyPersonaSeed({
|
|
28
|
+
personaId: opts.persona.id,
|
|
29
|
+
personaDir: opts.persona.personaDir,
|
|
30
|
+
store: bundle.workspaceLayerA,
|
|
31
|
+
});
|
|
32
|
+
if (seed.copied.length > 0 || seed.blocked.length > 0) {
|
|
33
|
+
bundle.workspaceRecord.memory_seed_applied = true;
|
|
34
|
+
bundle.workspaceRecord.memory_seed_persona = opts.persona.id;
|
|
35
|
+
await writeWorkspaceId(bundle.workspaceDir, bundle.workspaceRecord);
|
|
36
|
+
bundle.audit.append({
|
|
37
|
+
_schema: 1,
|
|
38
|
+
ts: new Date().toISOString(),
|
|
39
|
+
producer: 'memory',
|
|
40
|
+
action: 'seed-applied',
|
|
41
|
+
detail: {
|
|
42
|
+
persona_id: opts.persona.id,
|
|
43
|
+
files_copied: seed.copied,
|
|
44
|
+
files_blocked: seed.blocked,
|
|
45
|
+
},
|
|
46
|
+
});
|
|
47
|
+
}
|
|
48
|
+
}
|
|
49
|
+
const contextBlock = await buildLayerAContext({
|
|
50
|
+
mode: bundle.scopeMode,
|
|
51
|
+
globalStore: bundle.globalLayerA,
|
|
52
|
+
workspaceStore: bundle.workspaceLayerA,
|
|
53
|
+
tokenLimit: opts.tokenLimit ?? 3000,
|
|
54
|
+
});
|
|
55
|
+
return { contextBlock, seed };
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
export async function onSessionShutdown(bundle: MemoryBundle): Promise<void> {
|
|
59
|
+
// WAL checkpoint happens on backend close; this is the seam.
|
|
60
|
+
await bundle.dispose();
|
|
61
|
+
}
|
|
@@ -0,0 +1,124 @@
|
|
|
1
|
+
import { describe, it, beforeEach, afterEach } from 'node:test';
|
|
2
|
+
import assert from 'node:assert/strict';
|
|
3
|
+
import { mkdtemp, rm } from 'node:fs/promises';
|
|
4
|
+
import { mkdirSync, writeFileSync } from 'node:fs';
|
|
5
|
+
import { tmpdir } from 'node:os';
|
|
6
|
+
import { join } from 'node:path';
|
|
7
|
+
import { showRecoveryNotesBanner, showDivergenceBanner } from './attach-banners.js';
|
|
8
|
+
|
|
9
|
+
interface FakeUi {
|
|
10
|
+
notifications: Array<[string, string]>;
|
|
11
|
+
notify: (msg: string, level: 'info' | 'warning' | 'error') => void;
|
|
12
|
+
}
|
|
13
|
+
|
|
14
|
+
function makeUi(): FakeUi {
|
|
15
|
+
const notifications: FakeUi['notifications'] = [];
|
|
16
|
+
return { notifications, notify: (m, l) => notifications.push([l, m]) };
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
function writeMeta(root: string, name: string, meta: Record<string, unknown>): void {
|
|
20
|
+
mkdirSync(join(root, name), { recursive: true });
|
|
21
|
+
writeFileSync(join(root, name, 'meta.json'), JSON.stringify(meta));
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
let root: string;
|
|
25
|
+
|
|
26
|
+
describe('attach-banners', () => {
|
|
27
|
+
beforeEach(async () => {
|
|
28
|
+
root = await mkdtemp(join(tmpdir(), 'banners-'));
|
|
29
|
+
});
|
|
30
|
+
afterEach(async () => {
|
|
31
|
+
await rm(root, { recursive: true, force: true });
|
|
32
|
+
});
|
|
33
|
+
|
|
34
|
+
it('showRecoveryNotesBanner emits warning when there are unseen notes; returns markSeen=true', () => {
|
|
35
|
+
writeMeta(root, 'p1', {
|
|
36
|
+
recovery_notes: [
|
|
37
|
+
{ kind: 'snapshot-failed', message: 'boom', at: '2026-05-31T10:00:00.000Z' },
|
|
38
|
+
{ kind: 'cells-since-snapshot', n: 3, at: '2026-05-31T11:00:00.000Z' },
|
|
39
|
+
],
|
|
40
|
+
recovery_notes_seen_at: null,
|
|
41
|
+
});
|
|
42
|
+
const ui = makeUi();
|
|
43
|
+
const { unseenCount, markSeen } = showRecoveryNotesBanner('p1', root, ui);
|
|
44
|
+
assert.equal(unseenCount, 2);
|
|
45
|
+
assert.equal(markSeen, true);
|
|
46
|
+
assert.equal(ui.notifications.length, 1);
|
|
47
|
+
assert.equal(ui.notifications[0][0], 'warning');
|
|
48
|
+
assert.match(ui.notifications[0][1], /2 unread recovery notes/);
|
|
49
|
+
assert.match(ui.notifications[0][1], /snapshot-failed: boom/);
|
|
50
|
+
assert.match(ui.notifications[0][1], /3 cells since last snapshot/);
|
|
51
|
+
});
|
|
52
|
+
|
|
53
|
+
it('showRecoveryNotesBanner does not notify when all notes are seen; returns markSeen=false', () => {
|
|
54
|
+
writeMeta(root, 'p1', {
|
|
55
|
+
recovery_notes: [
|
|
56
|
+
{ kind: 'snapshot-failed', message: 'boom', at: '2026-05-31T10:00:00.000Z' },
|
|
57
|
+
],
|
|
58
|
+
recovery_notes_seen_at: '2026-05-31T11:00:00.000Z',
|
|
59
|
+
});
|
|
60
|
+
const ui = makeUi();
|
|
61
|
+
const { unseenCount, markSeen } = showRecoveryNotesBanner('p1', root, ui);
|
|
62
|
+
assert.equal(unseenCount, 0);
|
|
63
|
+
assert.equal(markSeen, false);
|
|
64
|
+
assert.equal(ui.notifications.length, 0);
|
|
65
|
+
});
|
|
66
|
+
|
|
67
|
+
it('showRecoveryNotesBanner truncates to 5 with "+ N more" footer', () => {
|
|
68
|
+
const notes = Array.from({ length: 8 }, (_, i) => ({
|
|
69
|
+
kind: 'snapshot-failed' as const,
|
|
70
|
+
message: `err-${i}`,
|
|
71
|
+
at: `2026-05-31T1${i}:00:00.000Z`,
|
|
72
|
+
}));
|
|
73
|
+
writeMeta(root, 'p1', { recovery_notes: notes, recovery_notes_seen_at: null });
|
|
74
|
+
const ui = makeUi();
|
|
75
|
+
const { unseenCount } = showRecoveryNotesBanner('p1', root, ui);
|
|
76
|
+
assert.equal(unseenCount, 8);
|
|
77
|
+
assert.match(ui.notifications[0][1], /\+ 3 more \(run \/sp notes\)/);
|
|
78
|
+
// Should include err-0..err-4 (first 5) but not err-5..err-7
|
|
79
|
+
assert.match(ui.notifications[0][1], /err-0/);
|
|
80
|
+
assert.match(ui.notifications[0][1], /err-4/);
|
|
81
|
+
assert.equal(ui.notifications[0][1].includes('err-5'), false);
|
|
82
|
+
});
|
|
83
|
+
|
|
84
|
+
it('showRecoveryNotesBanner tolerates missing or corrupt meta silently', () => {
|
|
85
|
+
const ui = makeUi();
|
|
86
|
+
// Missing scratchpad dir
|
|
87
|
+
const r1 = showRecoveryNotesBanner('absent', root, ui);
|
|
88
|
+
assert.deepEqual(r1, { unseenCount: 0, markSeen: false });
|
|
89
|
+
// Corrupt meta.json
|
|
90
|
+
mkdirSync(join(root, 'p2'), { recursive: true });
|
|
91
|
+
writeFileSync(join(root, 'p2', 'meta.json'), '{not json');
|
|
92
|
+
const r2 = showRecoveryNotesBanner('p2', root, ui);
|
|
93
|
+
assert.deepEqual(r2, { unseenCount: 0, markSeen: false });
|
|
94
|
+
assert.equal(ui.notifications.length, 0);
|
|
95
|
+
});
|
|
96
|
+
|
|
97
|
+
it('showDivergenceBanner emits info when leaf !== kernel; both set', () => {
|
|
98
|
+
writeMeta(root, 'p1', { cell_leaf_id: 5, kernel_at_cell_id: 8 });
|
|
99
|
+
const ui = makeUi();
|
|
100
|
+
const { diverged } = showDivergenceBanner('p1', root, ui);
|
|
101
|
+
assert.equal(diverged, true);
|
|
102
|
+
assert.equal(ui.notifications.length, 1);
|
|
103
|
+
assert.equal(ui.notifications[0][0], 'info');
|
|
104
|
+
assert.match(ui.notifications[0][1], /kernel state is at cell #8/);
|
|
105
|
+
assert.match(ui.notifications[0][1], /view is at cell #5/);
|
|
106
|
+
assert.match(ui.notifications[0][1], /\/sp tree to inspect/);
|
|
107
|
+
});
|
|
108
|
+
|
|
109
|
+
it('showDivergenceBanner does not notify when leaf===kernel, either is null, or meta missing', () => {
|
|
110
|
+
const ui = makeUi();
|
|
111
|
+
// Equal
|
|
112
|
+
writeMeta(root, 'p1', { cell_leaf_id: 5, kernel_at_cell_id: 5 });
|
|
113
|
+
assert.deepEqual(showDivergenceBanner('p1', root, ui), { diverged: false });
|
|
114
|
+
// Leaf null
|
|
115
|
+
writeMeta(root, 'p2', { cell_leaf_id: null, kernel_at_cell_id: 5 });
|
|
116
|
+
assert.deepEqual(showDivergenceBanner('p2', root, ui), { diverged: false });
|
|
117
|
+
// Kernel null
|
|
118
|
+
writeMeta(root, 'p3', { cell_leaf_id: 5, kernel_at_cell_id: null });
|
|
119
|
+
assert.deepEqual(showDivergenceBanner('p3', root, ui), { diverged: false });
|
|
120
|
+
// Meta missing
|
|
121
|
+
assert.deepEqual(showDivergenceBanner('absent', root, ui), { diverged: false });
|
|
122
|
+
assert.equal(ui.notifications.length, 0);
|
|
123
|
+
});
|
|
124
|
+
});
|
|
@@ -0,0 +1,67 @@
|
|
|
1
|
+
import { existsSync, readFileSync } from 'node:fs';
|
|
2
|
+
import { join } from 'node:path';
|
|
3
|
+
import type { RecoveryNote } from '@otto/coworker-scratchpad';
|
|
4
|
+
|
|
5
|
+
type RecoveryNoteEntry = RecoveryNote & { at: string };
|
|
6
|
+
|
|
7
|
+
interface UiNotify {
|
|
8
|
+
notify: (msg: string, level: 'info' | 'warning' | 'error') => void;
|
|
9
|
+
}
|
|
10
|
+
|
|
11
|
+
export function showRecoveryNotesBanner(
|
|
12
|
+
name: string,
|
|
13
|
+
rootDir: string,
|
|
14
|
+
ui: UiNotify,
|
|
15
|
+
): { unseenCount: number; markSeen: boolean } {
|
|
16
|
+
const metaPath = join(rootDir, name, 'meta.json');
|
|
17
|
+
if (!existsSync(metaPath)) return { unseenCount: 0, markSeen: false };
|
|
18
|
+
let meta: Record<string, unknown>;
|
|
19
|
+
try {
|
|
20
|
+
meta = JSON.parse(readFileSync(metaPath, 'utf8')) as Record<string, unknown>;
|
|
21
|
+
} catch {
|
|
22
|
+
return { unseenCount: 0, markSeen: false };
|
|
23
|
+
}
|
|
24
|
+
const notes = Array.isArray(meta.recovery_notes) ? (meta.recovery_notes as RecoveryNoteEntry[]) : [];
|
|
25
|
+
if (notes.length === 0) return { unseenCount: 0, markSeen: false };
|
|
26
|
+
const seenAt = typeof meta.recovery_notes_seen_at === 'string' ? meta.recovery_notes_seen_at : null;
|
|
27
|
+
const unseen = notes.filter((n) => seenAt === null || n.at > seenAt);
|
|
28
|
+
if (unseen.length === 0) return { unseenCount: 0, markSeen: false };
|
|
29
|
+
const head = unseen.slice(0, 5).map(formatNoteLine).join('\n');
|
|
30
|
+
const tail = unseen.length > 5 ? `\n+ ${unseen.length - 5} more (run /sp notes)` : '';
|
|
31
|
+
ui.notify(`⚠ ${unseen.length} unread recovery notes:\n${head}${tail}`, 'warning');
|
|
32
|
+
return { unseenCount: unseen.length, markSeen: true };
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
export function showDivergenceBanner(
|
|
36
|
+
name: string,
|
|
37
|
+
rootDir: string,
|
|
38
|
+
ui: UiNotify,
|
|
39
|
+
): { diverged: boolean } {
|
|
40
|
+
const metaPath = join(rootDir, name, 'meta.json');
|
|
41
|
+
if (!existsSync(metaPath)) return { diverged: false };
|
|
42
|
+
let meta: Record<string, unknown>;
|
|
43
|
+
try {
|
|
44
|
+
meta = JSON.parse(readFileSync(metaPath, 'utf8')) as Record<string, unknown>;
|
|
45
|
+
} catch {
|
|
46
|
+
return { diverged: false };
|
|
47
|
+
}
|
|
48
|
+
const leaf = typeof meta.cell_leaf_id === 'number' ? meta.cell_leaf_id : null;
|
|
49
|
+
const kernel = typeof meta.kernel_at_cell_id === 'number' ? meta.kernel_at_cell_id : null;
|
|
50
|
+
if (leaf === null || kernel === null || leaf === kernel) return { diverged: false };
|
|
51
|
+
ui.notify(
|
|
52
|
+
`ℹ kernel state is at cell #${kernel}; view is at cell #${leaf} (run /sp tree to inspect)`,
|
|
53
|
+
'info',
|
|
54
|
+
);
|
|
55
|
+
return { diverged: true };
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
export function formatNoteLine(n: RecoveryNoteEntry): string {
|
|
59
|
+
const ts = n.at.slice(0, 19);
|
|
60
|
+
switch (n.kind) {
|
|
61
|
+
case 'snapshot-failed': return ` • [${ts}] snapshot-failed: ${n.message}`;
|
|
62
|
+
case 'cells-since-snapshot': return ` • [${ts}] ${n.n} cells since last snapshot`;
|
|
63
|
+
case 'namespace-corrupt': return ` • [${ts}] namespace-corrupt: ${n.message}`;
|
|
64
|
+
case 'namespace-absent': return ` • [${ts}] namespace-absent`;
|
|
65
|
+
default: return ` • [${ts}] ${(n as { kind: string }).kind}`;
|
|
66
|
+
}
|
|
67
|
+
}
|