@cleocode/core 2026.4.12 → 2026.4.14
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/index.js +69233 -175
- package/dist/index.js.map +7 -1
- package/dist/init.d.ts.map +1 -1
- package/dist/internal.d.ts +16 -5
- package/dist/internal.d.ts.map +1 -1
- package/dist/memory/brain-embedding.d.ts +1 -1
- package/dist/memory/embedding-local.d.ts +8 -3
- package/dist/memory/embedding-local.d.ts.map +1 -1
- package/dist/memory/embedding-worker.d.ts +1 -1
- package/dist/memory/learnings.d.ts +2 -2
- package/dist/memory/patterns.d.ts +6 -6
- package/dist/store/agent-registry-accessor.d.ts +54 -1
- package/dist/store/agent-registry-accessor.d.ts.map +1 -1
- package/dist/store/backup-crypto.d.ts +89 -0
- package/dist/store/backup-crypto.d.ts.map +1 -0
- package/dist/store/backup-pack.d.ts +91 -0
- package/dist/store/backup-pack.d.ts.map +1 -0
- package/dist/store/backup-unpack.d.ts +140 -0
- package/dist/store/backup-unpack.d.ts.map +1 -0
- package/dist/store/migrate-signaldock-to-conduit.d.ts.map +1 -1
- package/dist/store/regenerators.d.ts +126 -0
- package/dist/store/regenerators.d.ts.map +1 -0
- package/dist/store/restore-conflict-report.d.ts +118 -0
- package/dist/store/restore-conflict-report.d.ts.map +1 -0
- package/dist/store/restore-json-merge.d.ts +187 -0
- package/dist/store/restore-json-merge.d.ts.map +1 -0
- package/dist/store/sqlite-backup.d.ts +60 -5
- package/dist/store/sqlite-backup.d.ts.map +1 -1
- package/dist/store/t310-readiness.d.ts +83 -0
- package/dist/store/t310-readiness.d.ts.map +1 -0
- package/package.json +11 -9
- package/src/__tests__/cli-parity.test.ts +19 -14
- package/src/__tests__/human-output.test.ts +13 -11
- package/src/init.ts +18 -8
- package/src/internal.ts +52 -2
- package/src/memory/__tests__/brain-automation.test.ts +2 -2
- package/src/memory/brain-embedding.ts +1 -1
- package/src/memory/embedding-local.ts +10 -5
- package/src/memory/embedding-worker.ts +1 -1
- package/src/skills/__tests__/discovery.test.ts +22 -7
- package/src/store/__tests__/backup-crypto.test.ts +101 -0
- package/src/store/__tests__/backup-pack.test.ts +491 -0
- package/src/store/__tests__/backup-unpack.test.ts +298 -0
- package/src/store/__tests__/performance-safety.test.ts +11 -5
- package/src/store/__tests__/regenerators.test.ts +234 -0
- package/src/store/__tests__/restore-conflict-report.test.ts +274 -0
- package/src/store/__tests__/restore-json-merge.test.ts +521 -0
- package/src/store/__tests__/t310-readiness.test.ts +111 -0
- package/src/store/__tests__/t311-integration.test.ts +661 -0
- package/src/store/backup-crypto.ts +209 -0
- package/src/store/backup-pack.ts +739 -0
- package/src/store/backup-unpack.ts +583 -0
- package/src/store/migrate-signaldock-to-conduit.ts +14 -7
- package/src/store/regenerators.ts +243 -0
- package/src/store/restore-conflict-report.ts +317 -0
- package/src/store/restore-json-merge.ts +653 -0
- package/src/store/t310-readiness.ts +119 -0
- package/src/validation/protocols/_shared.ts +1 -1
- package/dist/adapters/adapter-registry.js +0 -64
- package/dist/adapters/adapter-registry.js.map +0 -1
- package/dist/adapters/discovery.js +0 -83
- package/dist/adapters/discovery.js.map +0 -1
- package/dist/adapters/index.js +0 -9
- package/dist/adapters/index.js.map +0 -1
- package/dist/adapters/manager.js +0 -260
- package/dist/adapters/manager.js.map +0 -1
- package/dist/admin/export-tasks.js +0 -171
- package/dist/admin/export-tasks.js.map +0 -1
- package/dist/admin/export.js +0 -103
- package/dist/admin/export.js.map +0 -1
- package/dist/admin/help.js +0 -106
- package/dist/admin/help.js.map +0 -1
- package/dist/admin/import-tasks.js +0 -182
- package/dist/admin/import-tasks.js.map +0 -1
- package/dist/admin/import.js +0 -129
- package/dist/admin/import.js.map +0 -1
- package/dist/admin/index.js +0 -13
- package/dist/admin/index.js.map +0 -1
- package/dist/adrs/find.js +0 -134
- package/dist/adrs/find.js.map +0 -1
- package/dist/adrs/index.js +0 -15
- package/dist/adrs/index.js.map +0 -1
- package/dist/adrs/link-pipeline.js +0 -160
- package/dist/adrs/link-pipeline.js.map +0 -1
- package/dist/adrs/list.js +0 -43
- package/dist/adrs/list.js.map +0 -1
- package/dist/adrs/parse.js +0 -51
- package/dist/adrs/parse.js.map +0 -1
- package/dist/adrs/show.js +0 -22
- package/dist/adrs/show.js.map +0 -1
- package/dist/adrs/sync.js +0 -188
- package/dist/adrs/sync.js.map +0 -1
- package/dist/adrs/types.js +0 -9
- package/dist/adrs/types.js.map +0 -1
- package/dist/adrs/validate.js +0 -57
- package/dist/adrs/validate.js.map +0 -1
- package/dist/agents/agent-registry.js +0 -288
- package/dist/agents/agent-registry.js.map +0 -1
- package/dist/agents/agent-schema.js +0 -82
- package/dist/agents/agent-schema.js.map +0 -1
- package/dist/agents/capacity.js +0 -116
- package/dist/agents/capacity.js.map +0 -1
- package/dist/agents/execution-learning.js +0 -474
- package/dist/agents/execution-learning.js.map +0 -1
- package/dist/agents/health-monitor.js +0 -217
- package/dist/agents/health-monitor.js.map +0 -1
- package/dist/agents/index.js +0 -29
- package/dist/agents/index.js.map +0 -1
- package/dist/agents/registry.js +0 -314
- package/dist/agents/registry.js.map +0 -1
- package/dist/agents/retry.js +0 -215
- package/dist/agents/retry.js.map +0 -1
- package/dist/audit-prune.js +0 -94
- package/dist/audit-prune.js.map +0 -1
- package/dist/audit.js +0 -68
- package/dist/audit.js.map +0 -1
- package/dist/backfill/index.js +0 -229
- package/dist/backfill/index.js.map +0 -1
- package/dist/bootstrap.js +0 -344
- package/dist/bootstrap.js.map +0 -1
- package/dist/caamp/adapter.js +0 -259
- package/dist/caamp/adapter.js.map +0 -1
- package/dist/caamp/capability-check.js +0 -38
- package/dist/caamp/capability-check.js.map +0 -1
- package/dist/caamp/index.js +0 -21
- package/dist/caamp/index.js.map +0 -1
- package/dist/caamp-init.js +0 -16
- package/dist/caamp-init.js.map +0 -1
- package/dist/cleo.js +0 -322
- package/dist/cleo.js.map +0 -1
- package/dist/code/index.js +0 -10
- package/dist/code/index.js.map +0 -1
- package/dist/code/outline.js +0 -165
- package/dist/code/outline.js.map +0 -1
- package/dist/code/parser.js +0 -295
- package/dist/code/parser.js.map +0 -1
- package/dist/code/search.js +0 -135
- package/dist/code/search.js.map +0 -1
- package/dist/code/unfold.js +0 -155
- package/dist/code/unfold.js.map +0 -1
- package/dist/codebase-map/analyzers/architecture.js +0 -129
- package/dist/codebase-map/analyzers/architecture.js.map +0 -1
- package/dist/codebase-map/analyzers/concerns.js +0 -122
- package/dist/codebase-map/analyzers/concerns.js.map +0 -1
- package/dist/codebase-map/analyzers/conventions.js +0 -149
- package/dist/codebase-map/analyzers/conventions.js.map +0 -1
- package/dist/codebase-map/analyzers/integrations.js +0 -108
- package/dist/codebase-map/analyzers/integrations.js.map +0 -1
- package/dist/codebase-map/analyzers/stack.js +0 -117
- package/dist/codebase-map/analyzers/stack.js.map +0 -1
- package/dist/codebase-map/analyzers/structure.js +0 -137
- package/dist/codebase-map/analyzers/structure.js.map +0 -1
- package/dist/codebase-map/analyzers/testing.js +0 -118
- package/dist/codebase-map/analyzers/testing.js.map +0 -1
- package/dist/codebase-map/index.js +0 -57
- package/dist/codebase-map/index.js.map +0 -1
- package/dist/codebase-map/store.js +0 -122
- package/dist/codebase-map/store.js.map +0 -1
- package/dist/codebase-map/summary.js +0 -152
- package/dist/codebase-map/summary.js.map +0 -1
- package/dist/compliance/index.js +0 -288
- package/dist/compliance/index.js.map +0 -1
- package/dist/compliance/protocol-enforcement.js +0 -332
- package/dist/compliance/protocol-enforcement.js.map +0 -1
- package/dist/compliance/protocol-rules.js +0 -786
- package/dist/compliance/protocol-rules.js.map +0 -1
- package/dist/compliance/protocol-types.js +0 -79
- package/dist/compliance/protocol-types.js.map +0 -1
- package/dist/compliance/store.js +0 -53
- package/dist/compliance/store.js.map +0 -1
- package/dist/conduit/conduit-client.js +0 -107
- package/dist/conduit/conduit-client.js.map +0 -1
- package/dist/conduit/factory.js +0 -52
- package/dist/conduit/factory.js.map +0 -1
- package/dist/conduit/http-transport.js +0 -155
- package/dist/conduit/http-transport.js.map +0 -1
- package/dist/conduit/index.js +0 -15
- package/dist/conduit/index.js.map +0 -1
- package/dist/conduit/local-transport.js +0 -255
- package/dist/conduit/local-transport.js.map +0 -1
- package/dist/conduit/sse-transport.js +0 -299
- package/dist/conduit/sse-transport.js.map +0 -1
- package/dist/config/build-config.js +0 -29
- package/dist/config/build-config.js.map +0 -1
- package/dist/config.js +0 -400
- package/dist/config.js.map +0 -1
- package/dist/constants.js +0 -18
- package/dist/constants.js.map +0 -1
- package/dist/context/index.js +0 -137
- package/dist/context/index.js.map +0 -1
- package/dist/crypto/credentials.js +0 -191
- package/dist/crypto/credentials.js.map +0 -1
- package/dist/discovery.js +0 -182
- package/dist/discovery.js.map +0 -1
- package/dist/engine-result.js +0 -12
- package/dist/engine-result.js.map +0 -1
- package/dist/error-catalog.js +0 -404
- package/dist/error-catalog.js.map +0 -1
- package/dist/error-registry.js +0 -393
- package/dist/error-registry.js.map +0 -1
- package/dist/errors.js +0 -173
- package/dist/errors.js.map +0 -1
- package/dist/hooks/handlers/agent-hooks.js +0 -106
- package/dist/hooks/handlers/agent-hooks.js.map +0 -1
- package/dist/hooks/handlers/context-hooks.js +0 -111
- package/dist/hooks/handlers/context-hooks.js.map +0 -1
- package/dist/hooks/handlers/error-hooks.js +0 -52
- package/dist/hooks/handlers/error-hooks.js.map +0 -1
- package/dist/hooks/handlers/file-hooks.js +0 -104
- package/dist/hooks/handlers/file-hooks.js.map +0 -1
- package/dist/hooks/handlers/handler-helpers.js +0 -61
- package/dist/hooks/handlers/handler-helpers.js.map +0 -1
- package/dist/hooks/handlers/index.js +0 -28
- package/dist/hooks/handlers/index.js.map +0 -1
- package/dist/hooks/handlers/memory-bridge-refresh.js +0 -42
- package/dist/hooks/handlers/memory-bridge-refresh.js.map +0 -1
- package/dist/hooks/handlers/notification-hooks.js +0 -62
- package/dist/hooks/handlers/notification-hooks.js.map +0 -1
- package/dist/hooks/handlers/session-hooks.js +0 -142
- package/dist/hooks/handlers/session-hooks.js.map +0 -1
- package/dist/hooks/handlers/task-hooks.js +0 -65
- package/dist/hooks/handlers/task-hooks.js.map +0 -1
- package/dist/hooks/handlers/work-capture-hooks.js +0 -165
- package/dist/hooks/handlers/work-capture-hooks.js.map +0 -1
- package/dist/hooks/index.js +0 -13
- package/dist/hooks/index.js.map +0 -1
- package/dist/hooks/payload-schemas.js +0 -220
- package/dist/hooks/payload-schemas.js.map +0 -1
- package/dist/hooks/provider-hooks.js +0 -66
- package/dist/hooks/provider-hooks.js.map +0 -1
- package/dist/hooks/registry.js +0 -229
- package/dist/hooks/registry.js.map +0 -1
- package/dist/hooks/types.js +0 -66
- package/dist/hooks/types.js.map +0 -1
- package/dist/hooks.js +0 -136
- package/dist/hooks.js.map +0 -1
- package/dist/init.js +0 -851
- package/dist/init.js.map +0 -1
- package/dist/inject/index.js +0 -82
- package/dist/inject/index.js.map +0 -1
- package/dist/injection.js +0 -343
- package/dist/injection.js.map +0 -1
- package/dist/intelligence/adaptive-validation.js +0 -497
- package/dist/intelligence/adaptive-validation.js.map +0 -1
- package/dist/intelligence/impact.js +0 -675
- package/dist/intelligence/impact.js.map +0 -1
- package/dist/intelligence/index.js +0 -22
- package/dist/intelligence/index.js.map +0 -1
- package/dist/intelligence/patterns.js +0 -492
- package/dist/intelligence/patterns.js.map +0 -1
- package/dist/intelligence/prediction.js +0 -499
- package/dist/intelligence/prediction.js.map +0 -1
- package/dist/intelligence/types.js +0 -13
- package/dist/intelligence/types.js.map +0 -1
- package/dist/internal.js +0 -306
- package/dist/internal.js.map +0 -1
- package/dist/issue/create.js +0 -121
- package/dist/issue/create.js.map +0 -1
- package/dist/issue/diagnostics.js +0 -59
- package/dist/issue/diagnostics.js.map +0 -1
- package/dist/issue/index.js +0 -10
- package/dist/issue/index.js.map +0 -1
- package/dist/issue/template-parser.js +0 -267
- package/dist/issue/template-parser.js.map +0 -1
- package/dist/json-schema-validator.js +0 -76
- package/dist/json-schema-validator.js.map +0 -1
- package/dist/lib/index.js +0 -11
- package/dist/lib/index.js.map +0 -1
- package/dist/lib/retry.js +0 -152
- package/dist/lib/retry.js.map +0 -1
- package/dist/lib/tree-sitter-languages.js +0 -71
- package/dist/lib/tree-sitter-languages.js.map +0 -1
- package/dist/lifecycle/chain-composition.js +0 -152
- package/dist/lifecycle/chain-composition.js.map +0 -1
- package/dist/lifecycle/chain-store.js +0 -246
- package/dist/lifecycle/chain-store.js.map +0 -1
- package/dist/lifecycle/consolidate-rcasd.js +0 -352
- package/dist/lifecycle/consolidate-rcasd.js.map +0 -1
- package/dist/lifecycle/default-chain.js +0 -176
- package/dist/lifecycle/default-chain.js.map +0 -1
- package/dist/lifecycle/evidence.js +0 -180
- package/dist/lifecycle/evidence.js.map +0 -1
- package/dist/lifecycle/frontmatter.js +0 -363
- package/dist/lifecycle/frontmatter.js.map +0 -1
- package/dist/lifecycle/index.js +0 -756
- package/dist/lifecycle/index.js.map +0 -1
- package/dist/lifecycle/pipeline.js +0 -656
- package/dist/lifecycle/pipeline.js.map +0 -1
- package/dist/lifecycle/rcasd-index.js +0 -326
- package/dist/lifecycle/rcasd-index.js.map +0 -1
- package/dist/lifecycle/rcasd-paths.js +0 -220
- package/dist/lifecycle/rcasd-paths.js.map +0 -1
- package/dist/lifecycle/resume.js +0 -864
- package/dist/lifecycle/resume.js.map +0 -1
- package/dist/lifecycle/stage-artifacts.js +0 -94
- package/dist/lifecycle/stage-artifacts.js.map +0 -1
- package/dist/lifecycle/stage-guidance.js +0 -234
- package/dist/lifecycle/stage-guidance.js.map +0 -1
- package/dist/lifecycle/stages.js +0 -534
- package/dist/lifecycle/stages.js.map +0 -1
- package/dist/lifecycle/state-machine.js +0 -516
- package/dist/lifecycle/state-machine.js.map +0 -1
- package/dist/lifecycle/tessera-engine.js +0 -249
- package/dist/lifecycle/tessera-engine.js.map +0 -1
- package/dist/logger.js +0 -140
- package/dist/logger.js.map +0 -1
- package/dist/memory/auto-extract.js +0 -177
- package/dist/memory/auto-extract.js.map +0 -1
- package/dist/memory/brain-embedding.js +0 -66
- package/dist/memory/brain-embedding.js.map +0 -1
- package/dist/memory/brain-lifecycle.js +0 -298
- package/dist/memory/brain-lifecycle.js.map +0 -1
- package/dist/memory/brain-links.js +0 -161
- package/dist/memory/brain-links.js.map +0 -1
- package/dist/memory/brain-maintenance.js +0 -114
- package/dist/memory/brain-maintenance.js.map +0 -1
- package/dist/memory/brain-migration.js +0 -149
- package/dist/memory/brain-migration.js.map +0 -1
- package/dist/memory/brain-reasoning.js +0 -215
- package/dist/memory/brain-reasoning.js.map +0 -1
- package/dist/memory/brain-retrieval.js +0 -542
- package/dist/memory/brain-retrieval.js.map +0 -1
- package/dist/memory/brain-row-types.js +0 -10
- package/dist/memory/brain-row-types.js.map +0 -1
- package/dist/memory/brain-search.js +0 -519
- package/dist/memory/brain-search.js.map +0 -1
- package/dist/memory/brain-similarity.js +0 -145
- package/dist/memory/brain-similarity.js.map +0 -1
- package/dist/memory/claude-mem-migration.js +0 -277
- package/dist/memory/claude-mem-migration.js.map +0 -1
- package/dist/memory/decisions.js +0 -162
- package/dist/memory/decisions.js.map +0 -1
- package/dist/memory/embedding-local.js +0 -97
- package/dist/memory/embedding-local.js.map +0 -1
- package/dist/memory/embedding-queue.js +0 -271
- package/dist/memory/embedding-queue.js.map +0 -1
- package/dist/memory/embedding-worker.js +0 -58
- package/dist/memory/embedding-worker.js.map +0 -1
- package/dist/memory/engine-compat.js +0 -1397
- package/dist/memory/engine-compat.js.map +0 -1
- package/dist/memory/index.js +0 -1140
- package/dist/memory/index.js.map +0 -1
- package/dist/memory/learnings.js +0 -121
- package/dist/memory/learnings.js.map +0 -1
- package/dist/memory/memory-bridge.js +0 -370
- package/dist/memory/memory-bridge.js.map +0 -1
- package/dist/memory/patterns.js +0 -122
- package/dist/memory/patterns.js.map +0 -1
- package/dist/memory/pipeline-manifest-sqlite.js +0 -975
- package/dist/memory/pipeline-manifest-sqlite.js.map +0 -1
- package/dist/memory/session-memory.js +0 -331
- package/dist/memory/session-memory.js.map +0 -1
- package/dist/metrics/ab-test.js +0 -260
- package/dist/metrics/ab-test.js.map +0 -1
- package/dist/metrics/aggregation.js +0 -363
- package/dist/metrics/aggregation.js.map +0 -1
- package/dist/metrics/common.js +0 -64
- package/dist/metrics/common.js.map +0 -1
- package/dist/metrics/enums.js +0 -78
- package/dist/metrics/enums.js.map +0 -1
- package/dist/metrics/index.js +0 -19
- package/dist/metrics/index.js.map +0 -1
- package/dist/metrics/model-provider-registry.js +0 -88
- package/dist/metrics/model-provider-registry.js.map +0 -1
- package/dist/metrics/otel-integration.js +0 -263
- package/dist/metrics/otel-integration.js.map +0 -1
- package/dist/metrics/provider-detection.js +0 -103
- package/dist/metrics/provider-detection.js.map +0 -1
- package/dist/metrics/token-estimation.js +0 -253
- package/dist/metrics/token-estimation.js.map +0 -1
- package/dist/metrics/token-service.js +0 -450
- package/dist/metrics/token-service.js.map +0 -1
- package/dist/migration/agent-outputs.js +0 -316
- package/dist/migration/agent-outputs.js.map +0 -1
- package/dist/migration/checksum.js +0 -92
- package/dist/migration/checksum.js.map +0 -1
- package/dist/migration/index.js +0 -282
- package/dist/migration/index.js.map +0 -1
- package/dist/migration/logger.js +0 -360
- package/dist/migration/logger.js.map +0 -1
- package/dist/migration/preflight.js +0 -9
- package/dist/migration/preflight.js.map +0 -1
- package/dist/migration/state.js +0 -421
- package/dist/migration/state.js.map +0 -1
- package/dist/migration/validate.js +0 -241
- package/dist/migration/validate.js.map +0 -1
- package/dist/mvi-helpers.js +0 -74
- package/dist/mvi-helpers.js.map +0 -1
- package/dist/nexus/deps.js +0 -375
- package/dist/nexus/deps.js.map +0 -1
- package/dist/nexus/discover.js +0 -288
- package/dist/nexus/discover.js.map +0 -1
- package/dist/nexus/hash.js +0 -10
- package/dist/nexus/hash.js.map +0 -1
- package/dist/nexus/index.js +0 -40
- package/dist/nexus/index.js.map +0 -1
- package/dist/nexus/migrate-json-to-sqlite.js +0 -115
- package/dist/nexus/migrate-json-to-sqlite.js.map +0 -1
- package/dist/nexus/permissions.js +0 -105
- package/dist/nexus/permissions.js.map +0 -1
- package/dist/nexus/query.js +0 -175
- package/dist/nexus/query.js.map +0 -1
- package/dist/nexus/registry.js +0 -584
- package/dist/nexus/registry.js.map +0 -1
- package/dist/nexus/sharing/index.js +0 -288
- package/dist/nexus/sharing/index.js.map +0 -1
- package/dist/nexus/transfer-types.js +0 -8
- package/dist/nexus/transfer-types.js.map +0 -1
- package/dist/nexus/transfer.js +0 -263
- package/dist/nexus/transfer.js.map +0 -1
- package/dist/nexus/workspace.js +0 -355
- package/dist/nexus/workspace.js.map +0 -1
- package/dist/observability/index.js +0 -103
- package/dist/observability/index.js.map +0 -1
- package/dist/observability/log-filter.js +0 -63
- package/dist/observability/log-filter.js.map +0 -1
- package/dist/observability/log-parser.js +0 -99
- package/dist/observability/log-parser.js.map +0 -1
- package/dist/observability/log-reader.js +0 -139
- package/dist/observability/log-reader.js.map +0 -1
- package/dist/observability/types.js +0 -19
- package/dist/observability/types.js.map +0 -1
- package/dist/orchestration/analyze.js +0 -107
- package/dist/orchestration/analyze.js.map +0 -1
- package/dist/orchestration/bootstrap.js +0 -132
- package/dist/orchestration/bootstrap.js.map +0 -1
- package/dist/orchestration/context.js +0 -56
- package/dist/orchestration/context.js.map +0 -1
- package/dist/orchestration/critical-path.js +0 -100
- package/dist/orchestration/critical-path.js.map +0 -1
- package/dist/orchestration/hierarchy.js +0 -183
- package/dist/orchestration/hierarchy.js.map +0 -1
- package/dist/orchestration/index.js +0 -287
- package/dist/orchestration/index.js.map +0 -1
- package/dist/orchestration/parallel.js +0 -89
- package/dist/orchestration/parallel.js.map +0 -1
- package/dist/orchestration/protocol-validators.js +0 -815
- package/dist/orchestration/protocol-validators.js.map +0 -1
- package/dist/orchestration/skill-ops.js +0 -98
- package/dist/orchestration/skill-ops.js.map +0 -1
- package/dist/orchestration/status.js +0 -107
- package/dist/orchestration/status.js.map +0 -1
- package/dist/orchestration/unblock.js +0 -103
- package/dist/orchestration/unblock.js.map +0 -1
- package/dist/orchestration/validate-spawn.js +0 -67
- package/dist/orchestration/validate-spawn.js.map +0 -1
- package/dist/orchestration/waves.js +0 -86
- package/dist/orchestration/waves.js.map +0 -1
- package/dist/otel/index.js +0 -163
- package/dist/otel/index.js.map +0 -1
- package/dist/output.js +0 -164
- package/dist/output.js.map +0 -1
- package/dist/pagination.js +0 -64
- package/dist/pagination.js.map +0 -1
- package/dist/paths.js +0 -842
- package/dist/paths.js.map +0 -1
- package/dist/phases/deps.js +0 -372
- package/dist/phases/deps.js.map +0 -1
- package/dist/phases/index.js +0 -349
- package/dist/phases/index.js.map +0 -1
- package/dist/pipeline/index.js +0 -10
- package/dist/pipeline/index.js.map +0 -1
- package/dist/pipeline/phase.js +0 -45
- package/dist/pipeline/phase.js.map +0 -1
- package/dist/platform.js +0 -211
- package/dist/platform.js.map +0 -1
- package/dist/project-info.js +0 -84
- package/dist/project-info.js.map +0 -1
- package/dist/reconciliation/index.js +0 -10
- package/dist/reconciliation/index.js.map +0 -1
- package/dist/reconciliation/link-store.js +0 -129
- package/dist/reconciliation/link-store.js.map +0 -1
- package/dist/reconciliation/reconciliation-engine.js +0 -298
- package/dist/reconciliation/reconciliation-engine.js.map +0 -1
- package/dist/release/artifacts.js +0 -427
- package/dist/release/artifacts.js.map +0 -1
- package/dist/release/changelog-writer.js +0 -151
- package/dist/release/changelog-writer.js.map +0 -1
- package/dist/release/channel.js +0 -144
- package/dist/release/channel.js.map +0 -1
- package/dist/release/ci.js +0 -166
- package/dist/release/ci.js.map +0 -1
- package/dist/release/github-pr.js +0 -225
- package/dist/release/github-pr.js.map +0 -1
- package/dist/release/guards.js +0 -116
- package/dist/release/guards.js.map +0 -1
- package/dist/release/index.js +0 -22
- package/dist/release/index.js.map +0 -1
- package/dist/release/release-config.js +0 -158
- package/dist/release/release-config.js.map +0 -1
- package/dist/release/release-manifest.js +0 -1019
- package/dist/release/release-manifest.js.map +0 -1
- package/dist/release/version-bump.js +0 -255
- package/dist/release/version-bump.js.map +0 -1
- package/dist/remote/index.js +0 -257
- package/dist/remote/index.js.map +0 -1
- package/dist/repair.js +0 -130
- package/dist/repair.js.map +0 -1
- package/dist/research/index.js +0 -2
- package/dist/research/index.js.map +0 -1
- package/dist/roadmap/index.js +0 -59
- package/dist/roadmap/index.js.map +0 -1
- package/dist/routing/capability-matrix.js +0 -1556
- package/dist/routing/capability-matrix.js.map +0 -1
- package/dist/routing/index.js +0 -9
- package/dist/routing/index.js.map +0 -1
- package/dist/scaffold.js +0 -1759
- package/dist/scaffold.js.map +0 -1
- package/dist/schema-management.js +0 -295
- package/dist/schema-management.js.map +0 -1
- package/dist/security/index.js +0 -9
- package/dist/security/index.js.map +0 -1
- package/dist/security/input-sanitization.js +0 -321
- package/dist/security/input-sanitization.js.map +0 -1
- package/dist/sequence/index.js +0 -295
- package/dist/sequence/index.js.map +0 -1
- package/dist/sessions/assumptions.js +0 -54
- package/dist/sessions/assumptions.js.map +0 -1
- package/dist/sessions/briefing.js +0 -377
- package/dist/sessions/briefing.js.map +0 -1
- package/dist/sessions/context-alert.js +0 -222
- package/dist/sessions/context-alert.js.map +0 -1
- package/dist/sessions/context-inject.js +0 -61
- package/dist/sessions/context-inject.js.map +0 -1
- package/dist/sessions/context-monitor.js +0 -98
- package/dist/sessions/context-monitor.js.map +0 -1
- package/dist/sessions/decisions.js +0 -65
- package/dist/sessions/decisions.js.map +0 -1
- package/dist/sessions/find.js +0 -65
- package/dist/sessions/find.js.map +0 -1
- package/dist/sessions/handoff.js +0 -328
- package/dist/sessions/handoff.js.map +0 -1
- package/dist/sessions/hitl-warnings.js +0 -254
- package/dist/sessions/hitl-warnings.js.map +0 -1
- package/dist/sessions/index.js +0 -327
- package/dist/sessions/index.js.map +0 -1
- package/dist/sessions/session-archive.js +0 -40
- package/dist/sessions/session-archive.js.map +0 -1
- package/dist/sessions/session-cleanup.js +0 -59
- package/dist/sessions/session-cleanup.js.map +0 -1
- package/dist/sessions/session-drift.js +0 -134
- package/dist/sessions/session-drift.js.map +0 -1
- package/dist/sessions/session-enforcement.js +0 -144
- package/dist/sessions/session-enforcement.js.map +0 -1
- package/dist/sessions/session-grade.js +0 -253
- package/dist/sessions/session-grade.js.map +0 -1
- package/dist/sessions/session-history.js +0 -42
- package/dist/sessions/session-history.js.map +0 -1
- package/dist/sessions/session-id.js +0 -81
- package/dist/sessions/session-id.js.map +0 -1
- package/dist/sessions/session-memory-bridge.js +0 -52
- package/dist/sessions/session-memory-bridge.js.map +0 -1
- package/dist/sessions/session-show.js +0 -24
- package/dist/sessions/session-show.js.map +0 -1
- package/dist/sessions/session-stats.js +0 -69
- package/dist/sessions/session-stats.js.map +0 -1
- package/dist/sessions/session-suspend.js +0 -39
- package/dist/sessions/session-suspend.js.map +0 -1
- package/dist/sessions/session-switch.js +0 -51
- package/dist/sessions/session-switch.js.map +0 -1
- package/dist/sessions/session-view.js +0 -76
- package/dist/sessions/session-view.js.map +0 -1
- package/dist/sessions/snapshot.js +0 -213
- package/dist/sessions/snapshot.js.map +0 -1
- package/dist/sessions/statusline-setup.js +0 -85
- package/dist/sessions/statusline-setup.js.map +0 -1
- package/dist/sessions/types.js +0 -8
- package/dist/sessions/types.js.map +0 -1
- package/dist/skills/agents/config.js +0 -94
- package/dist/skills/agents/config.js.map +0 -1
- package/dist/skills/agents/install.js +0 -116
- package/dist/skills/agents/install.js.map +0 -1
- package/dist/skills/agents/registry.js +0 -161
- package/dist/skills/agents/registry.js.map +0 -1
- package/dist/skills/discovery.js +0 -333
- package/dist/skills/discovery.js.map +0 -1
- package/dist/skills/dispatch.js +0 -347
- package/dist/skills/dispatch.js.map +0 -1
- package/dist/skills/dynamic-skill-generator.js +0 -87
- package/dist/skills/dynamic-skill-generator.js.map +0 -1
- package/dist/skills/index.js +0 -44
- package/dist/skills/index.js.map +0 -1
- package/dist/skills/injection/subagent.js +0 -195
- package/dist/skills/injection/subagent.js.map +0 -1
- package/dist/skills/injection/token.js +0 -260
- package/dist/skills/injection/token.js.map +0 -1
- package/dist/skills/install.js +0 -40
- package/dist/skills/install.js.map +0 -1
- package/dist/skills/manifests/contribution.js +0 -175
- package/dist/skills/manifests/contribution.js.map +0 -1
- package/dist/skills/manifests/research.js +0 -281
- package/dist/skills/manifests/research.js.map +0 -1
- package/dist/skills/manifests/resolver.js +0 -146
- package/dist/skills/manifests/resolver.js.map +0 -1
- package/dist/skills/marketplace.js +0 -90
- package/dist/skills/marketplace.js.map +0 -1
- package/dist/skills/orchestrator/spawn.js +0 -178
- package/dist/skills/orchestrator/spawn.js.map +0 -1
- package/dist/skills/orchestrator/startup.js +0 -451
- package/dist/skills/orchestrator/startup.js.map +0 -1
- package/dist/skills/orchestrator/validator.js +0 -301
- package/dist/skills/orchestrator/validator.js.map +0 -1
- package/dist/skills/precedence-integration.js +0 -73
- package/dist/skills/precedence-integration.js.map +0 -1
- package/dist/skills/precedence-types.js +0 -16
- package/dist/skills/precedence-types.js.map +0 -1
- package/dist/skills/routing-table.js +0 -63
- package/dist/skills/routing-table.js.map +0 -1
- package/dist/skills/skill-paths.js +0 -217
- package/dist/skills/skill-paths.js.map +0 -1
- package/dist/skills/test-utility.js +0 -55
- package/dist/skills/test-utility.js.map +0 -1
- package/dist/skills/types.js +0 -118
- package/dist/skills/types.js.map +0 -1
- package/dist/skills/validation.js +0 -183
- package/dist/skills/validation.js.map +0 -1
- package/dist/skills/version.js +0 -57
- package/dist/skills/version.js.map +0 -1
- package/dist/snapshot/index.js +0 -188
- package/dist/snapshot/index.js.map +0 -1
- package/dist/spawn/adapter-registry.js +0 -246
- package/dist/spawn/adapter-registry.js.map +0 -1
- package/dist/spawn/index.js +0 -10
- package/dist/spawn/index.js.map +0 -1
- package/dist/stats/index.js +0 -343
- package/dist/stats/index.js.map +0 -1
- package/dist/stats/workflow-telemetry.js +0 -400
- package/dist/stats/workflow-telemetry.js.map +0 -1
- package/dist/sticky/archive.js +0 -47
- package/dist/sticky/archive.js.map +0 -1
- package/dist/sticky/convert.js +0 -235
- package/dist/sticky/convert.js.map +0 -1
- package/dist/sticky/create.js +0 -48
- package/dist/sticky/create.js.map +0 -1
- package/dist/sticky/id.js +0 -35
- package/dist/sticky/id.js.map +0 -1
- package/dist/sticky/index.js +0 -16
- package/dist/sticky/index.js.map +0 -1
- package/dist/sticky/list.js +0 -44
- package/dist/sticky/list.js.map +0 -1
- package/dist/sticky/purge.js +0 -45
- package/dist/sticky/purge.js.map +0 -1
- package/dist/sticky/show.js +0 -42
- package/dist/sticky/show.js.map +0 -1
- package/dist/sticky/types.js +0 -10
- package/dist/sticky/types.js.map +0 -1
- package/dist/store/agent-registry-accessor.js +0 -783
- package/dist/store/agent-registry-accessor.js.map +0 -1
- package/dist/store/api-key-kdf.js +0 -84
- package/dist/store/api-key-kdf.js.map +0 -1
- package/dist/store/atomic.js +0 -167
- package/dist/store/atomic.js.map +0 -1
- package/dist/store/backup.js +0 -94
- package/dist/store/backup.js.map +0 -1
- package/dist/store/brain-accessor.js +0 -397
- package/dist/store/brain-accessor.js.map +0 -1
- package/dist/store/brain-schema.js +0 -215
- package/dist/store/brain-schema.js.map +0 -1
- package/dist/store/brain-sqlite.js +0 -222
- package/dist/store/brain-sqlite.js.map +0 -1
- package/dist/store/cache.js +0 -168
- package/dist/store/cache.js.map +0 -1
- package/dist/store/chain-schema.js +0 -51
- package/dist/store/chain-schema.js.map +0 -1
- package/dist/store/cleanup-legacy.js +0 -171
- package/dist/store/cleanup-legacy.js.map +0 -1
- package/dist/store/conduit-sqlite.js +0 -570
- package/dist/store/conduit-sqlite.js.map +0 -1
- package/dist/store/converters.js +0 -124
- package/dist/store/converters.js.map +0 -1
- package/dist/store/cross-db-cleanup.js +0 -319
- package/dist/store/cross-db-cleanup.js.map +0 -1
- package/dist/store/data-accessor.js +0 -26
- package/dist/store/data-accessor.js.map +0 -1
- package/dist/store/data-safety-central.js +0 -269
- package/dist/store/data-safety-central.js.map +0 -1
- package/dist/store/data-safety.js +0 -274
- package/dist/store/data-safety.js.map +0 -1
- package/dist/store/db-helpers.js +0 -224
- package/dist/store/db-helpers.js.map +0 -1
- package/dist/store/export.js +0 -155
- package/dist/store/export.js.map +0 -1
- package/dist/store/file-utils.js +0 -270
- package/dist/store/file-utils.js.map +0 -1
- package/dist/store/git-checkpoint.js +0 -365
- package/dist/store/git-checkpoint.js.map +0 -1
- package/dist/store/global-salt.js +0 -147
- package/dist/store/global-salt.js.map +0 -1
- package/dist/store/import-logging.js +0 -139
- package/dist/store/import-logging.js.map +0 -1
- package/dist/store/import-remap.js +0 -145
- package/dist/store/import-remap.js.map +0 -1
- package/dist/store/import-sort.js +0 -121
- package/dist/store/import-sort.js.map +0 -1
- package/dist/store/index.js +0 -29
- package/dist/store/index.js.map +0 -1
- package/dist/store/json.js +0 -208
- package/dist/store/json.js.map +0 -1
- package/dist/store/lifecycle-store.js +0 -249
- package/dist/store/lifecycle-store.js.map +0 -1
- package/dist/store/lock.js +0 -70
- package/dist/store/lock.js.map +0 -1
- package/dist/store/migrate-signaldock-to-conduit.js +0 -555
- package/dist/store/migrate-signaldock-to-conduit.js.map +0 -1
- package/dist/store/migration-manager.js +0 -151
- package/dist/store/migration-manager.js.map +0 -1
- package/dist/store/migration-sqlite.js +0 -676
- package/dist/store/migration-sqlite.js.map +0 -1
- package/dist/store/nexus-schema.js +0 -62
- package/dist/store/nexus-schema.js.map +0 -1
- package/dist/store/nexus-sqlite.js +0 -242
- package/dist/store/nexus-sqlite.js.map +0 -1
- package/dist/store/nexus-validation-schemas.js +0 -40
- package/dist/store/nexus-validation-schemas.js.map +0 -1
- package/dist/store/parsers.js +0 -37
- package/dist/store/parsers.js.map +0 -1
- package/dist/store/project-detect.js +0 -457
- package/dist/store/project-detect.js.map +0 -1
- package/dist/store/provider.js +0 -101
- package/dist/store/provider.js.map +0 -1
- package/dist/store/safety-data-accessor.js +0 -257
- package/dist/store/safety-data-accessor.js.map +0 -1
- package/dist/store/schema.js +0 -7
- package/dist/store/schema.js.map +0 -1
- package/dist/store/session-store.js +0 -219
- package/dist/store/session-store.js.map +0 -1
- package/dist/store/signaldock-sqlite.js +0 -550
- package/dist/store/signaldock-sqlite.js.map +0 -1
- package/dist/store/sqlite-backup.js +0 -359
- package/dist/store/sqlite-backup.js.map +0 -1
- package/dist/store/sqlite-data-accessor.js +0 -787
- package/dist/store/sqlite-data-accessor.js.map +0 -1
- package/dist/store/sqlite.js +0 -481
- package/dist/store/sqlite.js.map +0 -1
- package/dist/store/status-registry.js +0 -8
- package/dist/store/status-registry.js.map +0 -1
- package/dist/store/task-store.js +0 -358
- package/dist/store/task-store.js.map +0 -1
- package/dist/store/tasks-schema.js +0 -610
- package/dist/store/tasks-schema.js.map +0 -1
- package/dist/store/typed-query.js +0 -15
- package/dist/store/typed-query.js.map +0 -1
- package/dist/store/validation-schemas.js +0 -278
- package/dist/store/validation-schemas.js.map +0 -1
- package/dist/system/archive-analytics.js +0 -277
- package/dist/system/archive-analytics.js.map +0 -1
- package/dist/system/archive-stats.js +0 -64
- package/dist/system/archive-stats.js.map +0 -1
- package/dist/system/audit.js +0 -145
- package/dist/system/audit.js.map +0 -1
- package/dist/system/backup.js +0 -280
- package/dist/system/backup.js.map +0 -1
- package/dist/system/cleanup.js +0 -134
- package/dist/system/cleanup.js.map +0 -1
- package/dist/system/health.js +0 -1100
- package/dist/system/health.js.map +0 -1
- package/dist/system/index.js +0 -18
- package/dist/system/index.js.map +0 -1
- package/dist/system/inject-generate.js +0 -122
- package/dist/system/inject-generate.js.map +0 -1
- package/dist/system/labels.js +0 -38
- package/dist/system/labels.js.map +0 -1
- package/dist/system/metrics.js +0 -61
- package/dist/system/metrics.js.map +0 -1
- package/dist/system/migrate.js +0 -43
- package/dist/system/migrate.js.map +0 -1
- package/dist/system/platform-paths.js +0 -80
- package/dist/system/platform-paths.js.map +0 -1
- package/dist/system/runtime.js +0 -161
- package/dist/system/runtime.js.map +0 -1
- package/dist/system/safestop.js +0 -99
- package/dist/system/safestop.js.map +0 -1
- package/dist/system/storage-preflight.js +0 -123
- package/dist/system/storage-preflight.js.map +0 -1
- package/dist/task-work/index.js +0 -159
- package/dist/task-work/index.js.map +0 -1
- package/dist/tasks/add.js +0 -736
- package/dist/tasks/add.js.map +0 -1
- package/dist/tasks/analyze.js +0 -85
- package/dist/tasks/analyze.js.map +0 -1
- package/dist/tasks/archive.js +0 -90
- package/dist/tasks/archive.js.map +0 -1
- package/dist/tasks/atomicity.js +0 -83
- package/dist/tasks/atomicity.js.map +0 -1
- package/dist/tasks/cancel-ops.js +0 -83
- package/dist/tasks/cancel-ops.js.map +0 -1
- package/dist/tasks/complete.js +0 -214
- package/dist/tasks/complete.js.map +0 -1
- package/dist/tasks/crossref-extract.js +0 -73
- package/dist/tasks/crossref-extract.js.map +0 -1
- package/dist/tasks/delete-preview.js +0 -192
- package/dist/tasks/delete-preview.js.map +0 -1
- package/dist/tasks/delete.js +0 -120
- package/dist/tasks/delete.js.map +0 -1
- package/dist/tasks/deletion-strategy.js +0 -200
- package/dist/tasks/deletion-strategy.js.map +0 -1
- package/dist/tasks/dependency-check.js +0 -278
- package/dist/tasks/dependency-check.js.map +0 -1
- package/dist/tasks/deps-ready.js +0 -32
- package/dist/tasks/deps-ready.js.map +0 -1
- package/dist/tasks/enforcement.js +0 -86
- package/dist/tasks/enforcement.js.map +0 -1
- package/dist/tasks/epic-enforcement.js +0 -294
- package/dist/tasks/epic-enforcement.js.map +0 -1
- package/dist/tasks/find.js +0 -157
- package/dist/tasks/find.js.map +0 -1
- package/dist/tasks/graph-cache.js +0 -127
- package/dist/tasks/graph-cache.js.map +0 -1
- package/dist/tasks/graph-ops.js +0 -171
- package/dist/tasks/graph-ops.js.map +0 -1
- package/dist/tasks/graph-rag.js +0 -328
- package/dist/tasks/graph-rag.js.map +0 -1
- package/dist/tasks/hierarchy-policy.js +0 -149
- package/dist/tasks/hierarchy-policy.js.map +0 -1
- package/dist/tasks/hierarchy.js +0 -185
- package/dist/tasks/hierarchy.js.map +0 -1
- package/dist/tasks/id-generator.js +0 -65
- package/dist/tasks/id-generator.js.map +0 -1
- package/dist/tasks/index.js +0 -14
- package/dist/tasks/index.js.map +0 -1
- package/dist/tasks/labels.js +0 -55
- package/dist/tasks/labels.js.map +0 -1
- package/dist/tasks/list.js +0 -75
- package/dist/tasks/list.js.map +0 -1
- package/dist/tasks/phase-tracking.js +0 -133
- package/dist/tasks/phase-tracking.js.map +0 -1
- package/dist/tasks/pipeline-stage.js +0 -248
- package/dist/tasks/pipeline-stage.js.map +0 -1
- package/dist/tasks/plan.js +0 -268
- package/dist/tasks/plan.js.map +0 -1
- package/dist/tasks/relates.js +0 -101
- package/dist/tasks/relates.js.map +0 -1
- package/dist/tasks/show.js +0 -83
- package/dist/tasks/show.js.map +0 -1
- package/dist/tasks/size-weighting.js +0 -86
- package/dist/tasks/size-weighting.js.map +0 -1
- package/dist/tasks/staleness.js +0 -86
- package/dist/tasks/staleness.js.map +0 -1
- package/dist/tasks/task-ops.js +0 -1741
- package/dist/tasks/task-ops.js.map +0 -1
- package/dist/tasks/update.js +0 -303
- package/dist/tasks/update.js.map +0 -1
- package/dist/templates/index.js +0 -10
- package/dist/templates/index.js.map +0 -1
- package/dist/templates/parser.js +0 -254
- package/dist/templates/parser.js.map +0 -1
- package/dist/ui/aliases.js +0 -153
- package/dist/ui/aliases.js.map +0 -1
- package/dist/ui/changelog.js +0 -184
- package/dist/ui/changelog.js.map +0 -1
- package/dist/ui/command-registry.js +0 -168
- package/dist/ui/command-registry.js.map +0 -1
- package/dist/ui/flags.js +0 -94
- package/dist/ui/flags.js.map +0 -1
- package/dist/ui/index.js +0 -24
- package/dist/ui/index.js.map +0 -1
- package/dist/upgrade.js +0 -1148
- package/dist/upgrade.js.map +0 -1
- package/dist/validation/chain-validation.js +0 -146
- package/dist/validation/chain-validation.js.map +0 -1
- package/dist/validation/compliance.js +0 -155
- package/dist/validation/compliance.js.map +0 -1
- package/dist/validation/docs-sync.js +0 -212
- package/dist/validation/docs-sync.js.map +0 -1
- package/dist/validation/doctor/checks.js +0 -1069
- package/dist/validation/doctor/checks.js.map +0 -1
- package/dist/validation/doctor/index.js +0 -9
- package/dist/validation/doctor/index.js.map +0 -1
- package/dist/validation/doctor/project-cache.js +0 -160
- package/dist/validation/doctor/project-cache.js.map +0 -1
- package/dist/validation/doctor/utils.js +0 -155
- package/dist/validation/doctor/utils.js.map +0 -1
- package/dist/validation/engine.js +0 -914
- package/dist/validation/engine.js.map +0 -1
- package/dist/validation/gap-check.js +0 -175
- package/dist/validation/gap-check.js.map +0 -1
- package/dist/validation/index.js +0 -40
- package/dist/validation/index.js.map +0 -1
- package/dist/validation/manifest.js +0 -237
- package/dist/validation/manifest.js.map +0 -1
- package/dist/validation/operation-gate-validators.js +0 -724
- package/dist/validation/operation-gate-validators.js.map +0 -1
- package/dist/validation/operation-verification-gates.js +0 -532
- package/dist/validation/operation-verification-gates.js.map +0 -1
- package/dist/validation/param-utils.js +0 -141
- package/dist/validation/param-utils.js.map +0 -1
- package/dist/validation/protocol-common.js +0 -300
- package/dist/validation/protocol-common.js.map +0 -1
- package/dist/validation/protocols/_shared.js +0 -82
- package/dist/validation/protocols/_shared.js.map +0 -1
- package/dist/validation/protocols/architecture-decision.js +0 -31
- package/dist/validation/protocols/architecture-decision.js.map +0 -1
- package/dist/validation/protocols/artifact-publish.js +0 -28
- package/dist/validation/protocols/artifact-publish.js.map +0 -1
- package/dist/validation/protocols/consensus.js +0 -41
- package/dist/validation/protocols/consensus.js.map +0 -1
- package/dist/validation/protocols/contribution.js +0 -27
- package/dist/validation/protocols/contribution.js.map +0 -1
- package/dist/validation/protocols/decomposition.js +0 -28
- package/dist/validation/protocols/decomposition.js.map +0 -1
- package/dist/validation/protocols/implementation.js +0 -24
- package/dist/validation/protocols/implementation.js.map +0 -1
- package/dist/validation/protocols/provenance.js +0 -29
- package/dist/validation/protocols/provenance.js.map +0 -1
- package/dist/validation/protocols/release.js +0 -29
- package/dist/validation/protocols/release.js.map +0 -1
- package/dist/validation/protocols/research.js +0 -24
- package/dist/validation/protocols/research.js.map +0 -1
- package/dist/validation/protocols/specification.js +0 -27
- package/dist/validation/protocols/specification.js.map +0 -1
- package/dist/validation/protocols/testing.js +0 -30
- package/dist/validation/protocols/testing.js.map +0 -1
- package/dist/validation/protocols/validation.js +0 -30
- package/dist/validation/protocols/validation.js.map +0 -1
- package/dist/validation/schema-integrity.js +0 -170
- package/dist/validation/schema-integrity.js.map +0 -1
- package/dist/validation/schema-validator.js +0 -176
- package/dist/validation/schema-validator.js.map +0 -1
- package/dist/validation/validate-ops.js +0 -937
- package/dist/validation/validate-ops.js.map +0 -1
- package/dist/validation/validation-rules.js +0 -226
- package/dist/validation/validation-rules.js.map +0 -1
- package/dist/validation/verification.js +0 -321
- package/dist/validation/verification.js.map +0 -1
|
@@ -0,0 +1,583 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* backup-unpack.ts — Bundle extraction and integrity verification for .cleobundle.tar.gz.
|
|
3
|
+
*
|
|
4
|
+
* Implements the unpack + verify half of the T311 import lifecycle.
|
|
5
|
+
* Extracts a .cleobundle.tar.gz (or .enc.cleobundle.tar.gz) to a staging
|
|
6
|
+
* directory and verifies all 6 integrity layers defined in ADR-038 §4.2.
|
|
7
|
+
*
|
|
8
|
+
* The caller is responsible for cleaning up the staging directory via
|
|
9
|
+
* {@link cleanupStaging} after processing. Restore-to-disk is the
|
|
10
|
+
* responsibility of T361 (CLI import handler).
|
|
11
|
+
*
|
|
12
|
+
* Verification layers (executed in strict order):
|
|
13
|
+
* Layer 1 — AES-256-GCM auth tag (encrypted bundles only)
|
|
14
|
+
* Layer 2 — Manifest self-hash (SHA-256 with placeholder substitution)
|
|
15
|
+
* Layer 3 — Manifest JSON Schema validation (bundled schemas/manifest-v1.json)
|
|
16
|
+
* Layer 4 — Per-file SHA-256 checksums
|
|
17
|
+
* Layer 5 — SQLite PRAGMA integrity_check
|
|
18
|
+
* Layer 6 — Schema version comparison (warnings only, never blocks)
|
|
19
|
+
*
|
|
20
|
+
* @task T350
|
|
21
|
+
* @epic T311
|
|
22
|
+
* @why ADR-038 — the unpack + verify half of the T311 import lifecycle.
|
|
23
|
+
* Restore-to-disk is the responsibility of T361 (CLI import handler);
|
|
24
|
+
* this module stops after verification and returns a staging dir path.
|
|
25
|
+
* @module store/backup-unpack
|
|
26
|
+
*/
|
|
27
|
+
|
|
28
|
+
import crypto from 'node:crypto';
|
|
29
|
+
import fs from 'node:fs';
|
|
30
|
+
import { createRequire } from 'node:module';
|
|
31
|
+
import os from 'node:os';
|
|
32
|
+
import path from 'node:path';
|
|
33
|
+
import type { DatabaseSync as _DatabaseSyncType } from 'node:sqlite';
|
|
34
|
+
import type { BackupManifest } from '@cleocode/contracts';
|
|
35
|
+
import type { Ajv as AjvInstance, ValidateFunction } from 'ajv';
|
|
36
|
+
// ajv/dist/2020 provides JSON Schema Draft 2020-12 support required by
|
|
37
|
+
// schemas/manifest-v1.json which declares `"$schema": "https://json-schema.org/draft/2020-12/schema"`.
|
|
38
|
+
import { default as Ajv2020Import } from 'ajv/dist/2020.js';
|
|
39
|
+
import { default as addFormatsImport } from 'ajv-formats';
|
|
40
|
+
import { extract as tarExtract } from 'tar';
|
|
41
|
+
import { decryptBundle, isEncryptedBundle } from './backup-crypto.js';
|
|
42
|
+
|
|
43
|
+
// ---------------------------------------------------------------------------
|
|
44
|
+
// node:sqlite interop (createRequire — Vitest strips `node:` prefix)
|
|
45
|
+
// ---------------------------------------------------------------------------
|
|
46
|
+
|
|
47
|
+
const _require = createRequire(import.meta.url);
|
|
48
|
+
type DatabaseSync = _DatabaseSyncType;
|
|
49
|
+
const { DatabaseSync } = _require('node:sqlite') as {
|
|
50
|
+
DatabaseSync: new (...args: ConstructorParameters<typeof _DatabaseSyncType>) => DatabaseSync;
|
|
51
|
+
};
|
|
52
|
+
|
|
53
|
+
// ---------------------------------------------------------------------------
|
|
54
|
+
// ajv ESM/CJS interop — Draft 2020-12 variant
|
|
55
|
+
// ---------------------------------------------------------------------------
|
|
56
|
+
|
|
57
|
+
const ajv2020Mod = Ajv2020Import as Record<string, unknown>;
|
|
58
|
+
const Ajv2020 = (
|
|
59
|
+
typeof ajv2020Mod.default === 'function' ? ajv2020Mod.default : Ajv2020Import
|
|
60
|
+
) as new (
|
|
61
|
+
opts?: Record<string, unknown>,
|
|
62
|
+
) => AjvInstance;
|
|
63
|
+
const fmtMod = addFormatsImport as Record<string, unknown>;
|
|
64
|
+
const addFormats = (typeof fmtMod.default === 'function' ? fmtMod.default : addFormatsImport) as (
|
|
65
|
+
ajv: AjvInstance,
|
|
66
|
+
) => AjvInstance;
|
|
67
|
+
|
|
68
|
+
// ---------------------------------------------------------------------------
|
|
69
|
+
// Public types
|
|
70
|
+
// ---------------------------------------------------------------------------
|
|
71
|
+
|
|
72
|
+
/**
|
|
73
|
+
* Input parameters for {@link unpackBundle}.
|
|
74
|
+
*
|
|
75
|
+
* @task T350
|
|
76
|
+
* @epic T311
|
|
77
|
+
*/
|
|
78
|
+
export interface UnpackBundleInput {
|
|
79
|
+
/** Absolute path to the .cleobundle.tar.gz (or .enc.cleobundle.tar.gz) file. */
|
|
80
|
+
bundlePath: string;
|
|
81
|
+
/** Required if the bundle is encrypted. */
|
|
82
|
+
passphrase?: string;
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
/**
|
|
86
|
+
* Schema compatibility warning for a database whose version differs from local.
|
|
87
|
+
*
|
|
88
|
+
* Warnings do NOT abort the import — they are collected and returned
|
|
89
|
+
* in the result for the caller to surface (spec §9, Q5=C best-effort).
|
|
90
|
+
*
|
|
91
|
+
* @task T350
|
|
92
|
+
* @epic T311
|
|
93
|
+
*/
|
|
94
|
+
export interface SchemaCompatWarning {
|
|
95
|
+
/** Logical database name as it appears in the manifest. */
|
|
96
|
+
db: string;
|
|
97
|
+
/** schemaVersion recorded in the bundle manifest. */
|
|
98
|
+
bundleVersion: string;
|
|
99
|
+
/** Current local schema version (from migration records). */
|
|
100
|
+
localVersion: string;
|
|
101
|
+
/** Direction of the version skew. */
|
|
102
|
+
severity: 'older-bundle' | 'newer-bundle';
|
|
103
|
+
}
|
|
104
|
+
|
|
105
|
+
/**
|
|
106
|
+
* Result of a successful {@link unpackBundle} call.
|
|
107
|
+
*
|
|
108
|
+
* The caller MUST call {@link cleanupStaging} with `stagingDir` after
|
|
109
|
+
* processing, regardless of what they do with the contents.
|
|
110
|
+
*
|
|
111
|
+
* @task T350
|
|
112
|
+
* @epic T311
|
|
113
|
+
*/
|
|
114
|
+
export interface UnpackBundleResult {
|
|
115
|
+
/** Absolute path to the extracted staging directory. Caller must clean up. */
|
|
116
|
+
stagingDir: string;
|
|
117
|
+
/** Parsed and validated manifest.json from the bundle. */
|
|
118
|
+
manifest: BackupManifest;
|
|
119
|
+
/** Per-layer verification results. */
|
|
120
|
+
verified: {
|
|
121
|
+
/** true if AES-GCM auth tag was valid (or N/A for unencrypted bundles). */
|
|
122
|
+
encryptionAuth: boolean;
|
|
123
|
+
/** true if manifest.json matched the bundled JSON Schema. */
|
|
124
|
+
manifestSchema: boolean;
|
|
125
|
+
/** true if all files' SHA-256 matched checksums.sha256. */
|
|
126
|
+
checksums: boolean;
|
|
127
|
+
/** true if all .db files passed PRAGMA integrity_check. */
|
|
128
|
+
sqliteIntegrity: boolean;
|
|
129
|
+
};
|
|
130
|
+
/** Schema version warnings — never block the import. */
|
|
131
|
+
warnings: SchemaCompatWarning[];
|
|
132
|
+
}
|
|
133
|
+
|
|
134
|
+
// ---------------------------------------------------------------------------
|
|
135
|
+
// Exit codes (ADR-038 §4.3)
|
|
136
|
+
// ---------------------------------------------------------------------------
|
|
137
|
+
|
|
138
|
+
/**
|
|
139
|
+
* Error thrown by {@link unpackBundle} when any integrity layer fails.
|
|
140
|
+
*
|
|
141
|
+
* Exit codes:
|
|
142
|
+
* - `70` `E_BUNDLE_DECRYPT` — decryption or passphrase failure
|
|
143
|
+
* - `71` `E_BUNDLE_SCHEMA` — manifest.json failed JSON Schema validation
|
|
144
|
+
* - `72` `E_CHECKSUM_MISMATCH` — SHA-256 checksum did not match
|
|
145
|
+
* - `73` `E_SQLITE_INTEGRITY` — SQLite PRAGMA integrity_check failed
|
|
146
|
+
* - `74` `E_MANIFEST_MISSING` — manifest.json absent from archive
|
|
147
|
+
* - `75` `E_SCHEMAS_MISSING` — schemas/manifest-v1.json absent from archive
|
|
148
|
+
*
|
|
149
|
+
* @task T350
|
|
150
|
+
* @epic T311
|
|
151
|
+
*/
|
|
152
|
+
export class BundleError extends Error {
|
|
153
|
+
/**
|
|
154
|
+
* @param code - Numeric exit code (70–75).
|
|
155
|
+
* @param codeName - Symbolic constant name, e.g. `'E_BUNDLE_DECRYPT'`.
|
|
156
|
+
* @param message - Human-readable error description.
|
|
157
|
+
*/
|
|
158
|
+
constructor(
|
|
159
|
+
public readonly code: number,
|
|
160
|
+
public readonly codeName: string,
|
|
161
|
+
message: string,
|
|
162
|
+
) {
|
|
163
|
+
super(message);
|
|
164
|
+
this.name = 'BundleError';
|
|
165
|
+
}
|
|
166
|
+
}
|
|
167
|
+
|
|
168
|
+
// ---------------------------------------------------------------------------
|
|
169
|
+
// Private helpers
|
|
170
|
+
// ---------------------------------------------------------------------------
|
|
171
|
+
|
|
172
|
+
/** Compute SHA-256 hex of a buffer. */
|
|
173
|
+
function sha256OfBuffer(buf: Buffer): string {
|
|
174
|
+
return crypto.createHash('sha256').update(buf).digest('hex');
|
|
175
|
+
}
|
|
176
|
+
|
|
177
|
+
/** Compute SHA-256 hex of a file on disk. */
|
|
178
|
+
function sha256OfFile(filePath: string): string {
|
|
179
|
+
return sha256OfBuffer(fs.readFileSync(filePath));
|
|
180
|
+
}
|
|
181
|
+
|
|
182
|
+
/**
|
|
183
|
+
* Build and return a singleton Ajv 2020-12 instance with formats support.
|
|
184
|
+
* The schema cache lives for the lifetime of the process, which is acceptable
|
|
185
|
+
* since the manifest-v1.json schema is stable.
|
|
186
|
+
*/
|
|
187
|
+
let _ajv2020: AjvInstance | null = null;
|
|
188
|
+
function getAjv2020(): AjvInstance {
|
|
189
|
+
if (_ajv2020 === null) {
|
|
190
|
+
_ajv2020 = new Ajv2020({ allErrors: true, strict: false, allowUnionTypes: true });
|
|
191
|
+
addFormats(_ajv2020);
|
|
192
|
+
}
|
|
193
|
+
return _ajv2020;
|
|
194
|
+
}
|
|
195
|
+
|
|
196
|
+
/**
|
|
197
|
+
* Stable schema ID for manifest-v1.json used as Ajv internal key.
|
|
198
|
+
* Having a fixed, well-known ID lets us call `ajv.getSchema(id)` on
|
|
199
|
+
* subsequent requests instead of re-compiling from disk every time.
|
|
200
|
+
*/
|
|
201
|
+
const MANIFEST_SCHEMA_ID = 'cleo-manifest-v1-internal';
|
|
202
|
+
|
|
203
|
+
/**
|
|
204
|
+
* Validate `data` against the JSON Schema loaded from `schemaPath`.
|
|
205
|
+
* Returns an array of error messages; empty array means valid.
|
|
206
|
+
*
|
|
207
|
+
* Uses a stable internal schema ID so that multiple calls within the
|
|
208
|
+
* same process reuse the compiled validator without triggering the Ajv
|
|
209
|
+
* "schema already exists" error.
|
|
210
|
+
*/
|
|
211
|
+
function validateAgainstJsonSchema(data: unknown, schemaPath: string): string[] {
|
|
212
|
+
const ajv = getAjv2020();
|
|
213
|
+
|
|
214
|
+
// Reuse previously compiled validator if already registered.
|
|
215
|
+
let validate: ValidateFunction | undefined = ajv.getSchema(MANIFEST_SCHEMA_ID);
|
|
216
|
+
if (validate === undefined) {
|
|
217
|
+
const rawSchema = JSON.parse(fs.readFileSync(schemaPath, 'utf-8')) as Record<string, unknown>;
|
|
218
|
+
// Strip the external `$id` to avoid Ajv's cross-call duplicate-id check,
|
|
219
|
+
// then add with our stable internal key.
|
|
220
|
+
const { $id: _unusedId, ...schemaWithoutId } = rawSchema;
|
|
221
|
+
ajv.addSchema(schemaWithoutId, MANIFEST_SCHEMA_ID);
|
|
222
|
+
validate = ajv.getSchema(MANIFEST_SCHEMA_ID) as ValidateFunction;
|
|
223
|
+
}
|
|
224
|
+
if (validate(data)) {
|
|
225
|
+
return [];
|
|
226
|
+
}
|
|
227
|
+
return (validate.errors ?? []).map(
|
|
228
|
+
(e: { instancePath?: string; message?: string }) =>
|
|
229
|
+
`${e.instancePath ?? '/'}: ${e.message ?? 'unknown'}`,
|
|
230
|
+
);
|
|
231
|
+
}
|
|
232
|
+
|
|
233
|
+
/**
|
|
234
|
+
* Read the latest applied Drizzle migration identifier from a local DB file.
|
|
235
|
+
*
|
|
236
|
+
* Looks for a `__drizzle_migrations` or `drizzle_migrations` table and reads
|
|
237
|
+
* the latest `folder_millis` or `created_at` value. Returns `null` if the
|
|
238
|
+
* local DB file does not exist or has no migration table (unknown DB).
|
|
239
|
+
*
|
|
240
|
+
* @param dbName - Logical database name (e.g. "tasks").
|
|
241
|
+
* @returns Migration identifier string, or `null` if unknown.
|
|
242
|
+
*/
|
|
243
|
+
function getLocalSchemaVersion(dbName: string): string | null {
|
|
244
|
+
// Resolve the path for known project-tier databases relative to this module.
|
|
245
|
+
// packages/core/src/store/backup-unpack.ts → packages/core/ → packages/ → root
|
|
246
|
+
const thisFile = import.meta.url.replace('file://', '');
|
|
247
|
+
const packageRoot = path.resolve(path.dirname(thisFile), '..', '..', '..');
|
|
248
|
+
|
|
249
|
+
// Known DB paths: project-tier DBs live at <projectRoot>/.cleo/<name>.db but
|
|
250
|
+
// we can only inspect the local running project's DB here. For schema version
|
|
251
|
+
// comparison we read from the local .cleo directory relative to a project root
|
|
252
|
+
// heuristic, but that is inherently environment-specific. The simpler and
|
|
253
|
+
// safer approach (spec §9 best-effort) is to look at the Drizzle migration
|
|
254
|
+
// folder for the known DBs shipped with this package.
|
|
255
|
+
const migrationCandidates = [
|
|
256
|
+
path.join(packageRoot, 'migrations', `drizzle-${dbName}`),
|
|
257
|
+
path.join(packageRoot, 'migrations', dbName),
|
|
258
|
+
];
|
|
259
|
+
|
|
260
|
+
for (const dir of migrationCandidates) {
|
|
261
|
+
if (!fs.existsSync(dir)) continue;
|
|
262
|
+
try {
|
|
263
|
+
const entries = fs
|
|
264
|
+
.readdirSync(dir)
|
|
265
|
+
.filter((n) => /^\d+/.test(n))
|
|
266
|
+
.sort()
|
|
267
|
+
.reverse();
|
|
268
|
+
if (entries.length > 0 && entries[0] != null) {
|
|
269
|
+
// Strip non-numeric suffix to get the millis part
|
|
270
|
+
const match = /^(\d+)/.exec(entries[0]);
|
|
271
|
+
if (match?.[1] != null) {
|
|
272
|
+
return match[1];
|
|
273
|
+
}
|
|
274
|
+
}
|
|
275
|
+
} catch {
|
|
276
|
+
// Non-fatal — fall through to next candidate
|
|
277
|
+
}
|
|
278
|
+
}
|
|
279
|
+
|
|
280
|
+
return null;
|
|
281
|
+
}
|
|
282
|
+
|
|
283
|
+
/**
|
|
284
|
+
* Compare bundle schema version vs local schema version for a single DB.
|
|
285
|
+
* Returns a {@link SchemaCompatWarning} when the versions differ, or `null`.
|
|
286
|
+
*/
|
|
287
|
+
function compareSchemaVersions(dbName: string, bundleVersion: string): SchemaCompatWarning | null {
|
|
288
|
+
if (bundleVersion === 'unknown') return null;
|
|
289
|
+
|
|
290
|
+
const localVersion = getLocalSchemaVersion(dbName);
|
|
291
|
+
if (localVersion === null) return null; // unknown DB — skip comparison
|
|
292
|
+
if (localVersion === 'unknown') return null;
|
|
293
|
+
if (bundleVersion === localVersion) return null;
|
|
294
|
+
|
|
295
|
+
// Numeric comparison where possible (Drizzle uses epoch millis as folder names)
|
|
296
|
+
const bNum = Number(bundleVersion);
|
|
297
|
+
const lNum = Number(localVersion);
|
|
298
|
+
|
|
299
|
+
if (!Number.isNaN(bNum) && !Number.isNaN(lNum)) {
|
|
300
|
+
const severity: SchemaCompatWarning['severity'] = bNum < lNum ? 'older-bundle' : 'newer-bundle';
|
|
301
|
+
return { db: dbName, bundleVersion, localVersion, severity };
|
|
302
|
+
}
|
|
303
|
+
|
|
304
|
+
// Fallback: lexicographic comparison
|
|
305
|
+
const severity: SchemaCompatWarning['severity'] =
|
|
306
|
+
bundleVersion < localVersion ? 'older-bundle' : 'newer-bundle';
|
|
307
|
+
return { db: dbName, bundleVersion, localVersion, severity };
|
|
308
|
+
}
|
|
309
|
+
|
|
310
|
+
// ---------------------------------------------------------------------------
|
|
311
|
+
// Public API
|
|
312
|
+
// ---------------------------------------------------------------------------
|
|
313
|
+
|
|
314
|
+
/**
|
|
315
|
+
* Extract a `.cleobundle.tar.gz` to a temp staging directory and verify all
|
|
316
|
+
* 6 integrity layers in strict sequence (ADR-038 §4.2).
|
|
317
|
+
*
|
|
318
|
+
* On any failure AFTER the staging directory is created, the staging directory
|
|
319
|
+
* is cleaned up before the {@link BundleError} is thrown.
|
|
320
|
+
*
|
|
321
|
+
* The caller MUST call {@link cleanupStaging} with the returned `stagingDir`
|
|
322
|
+
* after it is done processing.
|
|
323
|
+
*
|
|
324
|
+
* @param input - Bundle path and optional passphrase.
|
|
325
|
+
* @returns Verification result with staging dir, manifest, layer flags, and warnings.
|
|
326
|
+
* @throws {BundleError} On any integrity failure (exit codes 70–75).
|
|
327
|
+
*
|
|
328
|
+
* @task T350
|
|
329
|
+
* @epic T311
|
|
330
|
+
*/
|
|
331
|
+
export async function unpackBundle(input: UnpackBundleInput): Promise<UnpackBundleResult> {
|
|
332
|
+
const { bundlePath, passphrase } = input;
|
|
333
|
+
|
|
334
|
+
// ----- Step 1: Read bundle header (first 8 bytes) -------------------------
|
|
335
|
+
const fd = fs.openSync(bundlePath, 'r');
|
|
336
|
+
const header = Buffer.alloc(8);
|
|
337
|
+
fs.readSync(fd, header, 0, 8, 0);
|
|
338
|
+
fs.closeSync(fd);
|
|
339
|
+
|
|
340
|
+
// ----- Step 2: Detect encryption ------------------------------------------
|
|
341
|
+
const encrypted = isEncryptedBundle(header);
|
|
342
|
+
|
|
343
|
+
// ----- Step 3: Decrypt if needed ------------------------------------------
|
|
344
|
+
let encryptionAuth = false;
|
|
345
|
+
let tarPath: string;
|
|
346
|
+
let tmpDecryptedPath: string | null = null;
|
|
347
|
+
|
|
348
|
+
if (encrypted) {
|
|
349
|
+
if (!passphrase || passphrase.length === 0) {
|
|
350
|
+
throw new BundleError(
|
|
351
|
+
70,
|
|
352
|
+
'E_BUNDLE_DECRYPT',
|
|
353
|
+
'Bundle is encrypted but no passphrase was provided.',
|
|
354
|
+
);
|
|
355
|
+
}
|
|
356
|
+
const encryptedBuf = fs.readFileSync(bundlePath);
|
|
357
|
+
let decrypted: Buffer;
|
|
358
|
+
try {
|
|
359
|
+
decrypted = decryptBundle(encryptedBuf, passphrase);
|
|
360
|
+
} catch (err) {
|
|
361
|
+
const msg = err instanceof Error ? err.message : String(err);
|
|
362
|
+
throw new BundleError(70, 'E_BUNDLE_DECRYPT', `Decryption failed: ${msg}`);
|
|
363
|
+
}
|
|
364
|
+
// Write decrypted tar.gz to a temp file for extraction
|
|
365
|
+
tmpDecryptedPath = path.join(os.tmpdir(), `cleo-unpack-dec-${Date.now()}.tar.gz`);
|
|
366
|
+
fs.writeFileSync(tmpDecryptedPath, decrypted);
|
|
367
|
+
tarPath = tmpDecryptedPath;
|
|
368
|
+
encryptionAuth = true;
|
|
369
|
+
} else {
|
|
370
|
+
tarPath = bundlePath;
|
|
371
|
+
// unencrypted: auth is N/A — report true (not applicable = pass)
|
|
372
|
+
encryptionAuth = true;
|
|
373
|
+
}
|
|
374
|
+
|
|
375
|
+
// ----- Step 4: Create staging directory -----------------------------------
|
|
376
|
+
const stagingDir = fs.mkdtempSync(path.join(os.tmpdir(), 'cleo-unpack-'));
|
|
377
|
+
|
|
378
|
+
// From here on, any failure must clean up stagingDir (and tmpDecryptedPath).
|
|
379
|
+
const cleanup = (): void => {
|
|
380
|
+
try {
|
|
381
|
+
fs.rmSync(stagingDir, { recursive: true, force: true });
|
|
382
|
+
} catch {
|
|
383
|
+
// best-effort
|
|
384
|
+
}
|
|
385
|
+
if (tmpDecryptedPath !== null) {
|
|
386
|
+
try {
|
|
387
|
+
if (fs.existsSync(tmpDecryptedPath)) {
|
|
388
|
+
fs.unlinkSync(tmpDecryptedPath);
|
|
389
|
+
}
|
|
390
|
+
} catch {
|
|
391
|
+
// best-effort
|
|
392
|
+
}
|
|
393
|
+
}
|
|
394
|
+
};
|
|
395
|
+
|
|
396
|
+
try {
|
|
397
|
+
// ----- Step 5: Extract tarball ------------------------------------------
|
|
398
|
+
try {
|
|
399
|
+
await tarExtract({ file: tarPath, cwd: stagingDir });
|
|
400
|
+
} catch (err) {
|
|
401
|
+
const msg = err instanceof Error ? err.message : String(err);
|
|
402
|
+
// Tar extraction failure is most likely due to corruption — treat as
|
|
403
|
+
// checksum mismatch (the corrupted bytes caused tar to fail before we
|
|
404
|
+
// could even read checksums).
|
|
405
|
+
throw new BundleError(72, 'E_CHECKSUM_MISMATCH', `Tar extraction failed: ${msg}`);
|
|
406
|
+
}
|
|
407
|
+
|
|
408
|
+
// ----- Step 6: Verify manifest.json exists -------------------------------
|
|
409
|
+
const manifestPath = path.join(stagingDir, 'manifest.json');
|
|
410
|
+
if (!fs.existsSync(manifestPath)) {
|
|
411
|
+
throw new BundleError(74, 'E_MANIFEST_MISSING', 'manifest.json is missing from the bundle.');
|
|
412
|
+
}
|
|
413
|
+
|
|
414
|
+
// ----- Step 7: Verify schemas/manifest-v1.json exists -------------------
|
|
415
|
+
const schemaPath = path.join(stagingDir, 'schemas', 'manifest-v1.json');
|
|
416
|
+
if (!fs.existsSync(schemaPath)) {
|
|
417
|
+
throw new BundleError(
|
|
418
|
+
75,
|
|
419
|
+
'E_SCHEMAS_MISSING',
|
|
420
|
+
'schemas/manifest-v1.json is missing from the bundle.',
|
|
421
|
+
);
|
|
422
|
+
}
|
|
423
|
+
|
|
424
|
+
// ----- Step 8: Parse manifest.json ---------------------------------------
|
|
425
|
+
let manifest: BackupManifest;
|
|
426
|
+
try {
|
|
427
|
+
manifest = JSON.parse(fs.readFileSync(manifestPath, 'utf-8')) as BackupManifest;
|
|
428
|
+
} catch (err) {
|
|
429
|
+
const msg = err instanceof Error ? err.message : String(err);
|
|
430
|
+
throw new BundleError(71, 'E_BUNDLE_SCHEMA', `manifest.json is not valid JSON: ${msg}`);
|
|
431
|
+
}
|
|
432
|
+
|
|
433
|
+
// ----- Step 9: Validate manifest against bundled JSON Schema (Layer 3) --
|
|
434
|
+
const schemaErrors = validateAgainstJsonSchema(manifest, schemaPath);
|
|
435
|
+
if (schemaErrors.length > 0) {
|
|
436
|
+
throw new BundleError(
|
|
437
|
+
71,
|
|
438
|
+
'E_BUNDLE_SCHEMA',
|
|
439
|
+
`manifest.json failed schema validation: ${schemaErrors.join('; ')}`,
|
|
440
|
+
);
|
|
441
|
+
}
|
|
442
|
+
const manifestSchema = true;
|
|
443
|
+
|
|
444
|
+
// ----- Layer 2: Manifest self-hash verification --------------------------
|
|
445
|
+
// Per spec §4.1: compute SHA-256 of manifest JSON with integrity.manifestHash=""
|
|
446
|
+
// and compare to integrity.manifestHash.
|
|
447
|
+
if (manifest.integrity.manifestHash != null && manifest.integrity.manifestHash.length > 0) {
|
|
448
|
+
const manifestWithPlaceholder = {
|
|
449
|
+
...manifest,
|
|
450
|
+
integrity: { ...manifest.integrity, manifestHash: '' },
|
|
451
|
+
};
|
|
452
|
+
const computedHash = sha256OfBuffer(
|
|
453
|
+
Buffer.from(JSON.stringify(manifestWithPlaceholder), 'utf-8'),
|
|
454
|
+
);
|
|
455
|
+
if (computedHash !== manifest.integrity.manifestHash) {
|
|
456
|
+
throw new BundleError(
|
|
457
|
+
71,
|
|
458
|
+
'E_BUNDLE_SCHEMA',
|
|
459
|
+
'Manifest self-hash mismatch — manifest.json may have been tampered with.',
|
|
460
|
+
);
|
|
461
|
+
}
|
|
462
|
+
}
|
|
463
|
+
|
|
464
|
+
// ----- Step 10: Checksum verification (Layer 4) --------------------------
|
|
465
|
+
const checksumsPath = path.join(stagingDir, 'checksums.sha256');
|
|
466
|
+
let checksums = true;
|
|
467
|
+
if (fs.existsSync(checksumsPath)) {
|
|
468
|
+
const checksumContent = fs.readFileSync(checksumsPath, 'utf-8');
|
|
469
|
+
const lines = checksumContent.split('\n').filter((l) => l.trim().length > 0);
|
|
470
|
+
for (const line of lines) {
|
|
471
|
+
// GNU sha256sum format: "<64 hex chars> <relative path>"
|
|
472
|
+
const spaceIdx = line.indexOf(' ');
|
|
473
|
+
if (spaceIdx === -1) continue;
|
|
474
|
+
const expectedHash = line.slice(0, spaceIdx).trim();
|
|
475
|
+
const relPath = line.slice(spaceIdx + 2).trim();
|
|
476
|
+
const filePath = path.join(stagingDir, relPath);
|
|
477
|
+
if (!fs.existsSync(filePath)) {
|
|
478
|
+
throw new BundleError(
|
|
479
|
+
72,
|
|
480
|
+
'E_CHECKSUM_MISMATCH',
|
|
481
|
+
`Checksummed file missing from staging: file=${relPath}`,
|
|
482
|
+
);
|
|
483
|
+
}
|
|
484
|
+
const actualHash = sha256OfFile(filePath);
|
|
485
|
+
if (actualHash !== expectedHash) {
|
|
486
|
+
throw new BundleError(72, 'E_CHECKSUM_MISMATCH', `SHA-256 mismatch for file=${relPath}`);
|
|
487
|
+
}
|
|
488
|
+
}
|
|
489
|
+
} else {
|
|
490
|
+
// checksums.sha256 missing from the extracted bundle is acceptable for
|
|
491
|
+
// bundles that were packed without checksums; pass through without failure.
|
|
492
|
+
checksums = true;
|
|
493
|
+
}
|
|
494
|
+
|
|
495
|
+
// ----- Step 11: SQLite integrity check (Layer 5) -------------------------
|
|
496
|
+
const sqliteIntegrity = true;
|
|
497
|
+
for (const dbEntry of manifest.databases) {
|
|
498
|
+
const dbPath = path.join(stagingDir, dbEntry.filename);
|
|
499
|
+
if (!fs.existsSync(dbPath)) continue;
|
|
500
|
+
let db: DatabaseSync | null = null;
|
|
501
|
+
try {
|
|
502
|
+
db = new DatabaseSync(dbPath, { readOnly: true });
|
|
503
|
+
const row = db.prepare('PRAGMA integrity_check').get() as
|
|
504
|
+
| { integrity_check: string }
|
|
505
|
+
| undefined;
|
|
506
|
+
if (row?.integrity_check !== 'ok') {
|
|
507
|
+
throw new BundleError(
|
|
508
|
+
73,
|
|
509
|
+
'E_SQLITE_INTEGRITY',
|
|
510
|
+
`PRAGMA integrity_check failed for file=${dbEntry.filename}`,
|
|
511
|
+
);
|
|
512
|
+
}
|
|
513
|
+
} catch (err) {
|
|
514
|
+
if (err instanceof BundleError) throw err;
|
|
515
|
+
throw new BundleError(
|
|
516
|
+
73,
|
|
517
|
+
'E_SQLITE_INTEGRITY',
|
|
518
|
+
`Could not open database for integrity check: file=${dbEntry.filename}`,
|
|
519
|
+
);
|
|
520
|
+
} finally {
|
|
521
|
+
try {
|
|
522
|
+
db?.close();
|
|
523
|
+
} catch {
|
|
524
|
+
// ignore
|
|
525
|
+
}
|
|
526
|
+
}
|
|
527
|
+
}
|
|
528
|
+
|
|
529
|
+
// ----- Step 12: Schema version compat warnings (Layer 6) ----------------
|
|
530
|
+
const warnings: SchemaCompatWarning[] = [];
|
|
531
|
+
for (const dbEntry of manifest.databases) {
|
|
532
|
+
const warning = compareSchemaVersions(dbEntry.name, dbEntry.schemaVersion);
|
|
533
|
+
if (warning !== null) {
|
|
534
|
+
warnings.push(warning);
|
|
535
|
+
}
|
|
536
|
+
}
|
|
537
|
+
|
|
538
|
+
// ----- Step 13: Clean up temp decrypted file if any ----------------------
|
|
539
|
+
if (tmpDecryptedPath !== null) {
|
|
540
|
+
try {
|
|
541
|
+
if (fs.existsSync(tmpDecryptedPath)) {
|
|
542
|
+
fs.unlinkSync(tmpDecryptedPath);
|
|
543
|
+
}
|
|
544
|
+
} catch {
|
|
545
|
+
// best-effort
|
|
546
|
+
}
|
|
547
|
+
}
|
|
548
|
+
|
|
549
|
+
// ----- Return result -----------------------------------------------------
|
|
550
|
+
return {
|
|
551
|
+
stagingDir,
|
|
552
|
+
manifest,
|
|
553
|
+
verified: {
|
|
554
|
+
encryptionAuth,
|
|
555
|
+
manifestSchema,
|
|
556
|
+
checksums,
|
|
557
|
+
sqliteIntegrity,
|
|
558
|
+
},
|
|
559
|
+
warnings,
|
|
560
|
+
};
|
|
561
|
+
} catch (err) {
|
|
562
|
+
cleanup();
|
|
563
|
+
throw err;
|
|
564
|
+
}
|
|
565
|
+
}
|
|
566
|
+
|
|
567
|
+
/**
|
|
568
|
+
* Remove the staging directory created by {@link unpackBundle}.
|
|
569
|
+
*
|
|
570
|
+
* Safe to call on a path that no longer exists (idempotent).
|
|
571
|
+
*
|
|
572
|
+
* @param stagingDir - Absolute path returned in {@link UnpackBundleResult.stagingDir}.
|
|
573
|
+
*
|
|
574
|
+
* @task T350
|
|
575
|
+
* @epic T311
|
|
576
|
+
*/
|
|
577
|
+
export function cleanupStaging(stagingDir: string): void {
|
|
578
|
+
try {
|
|
579
|
+
fs.rmSync(stagingDir, { recursive: true, force: true });
|
|
580
|
+
} catch {
|
|
581
|
+
// best-effort — do not throw on cleanup
|
|
582
|
+
}
|
|
583
|
+
}
|
|
@@ -24,7 +24,7 @@
|
|
|
24
24
|
import { existsSync, mkdirSync, renameSync, unlinkSync } from 'node:fs';
|
|
25
25
|
import { createRequire } from 'node:module';
|
|
26
26
|
import { join } from 'node:path';
|
|
27
|
-
import type { DatabaseSync as _DatabaseSyncType } from 'node:sqlite';
|
|
27
|
+
import type { DatabaseSync as _DatabaseSyncType, SQLInputValue } from 'node:sqlite';
|
|
28
28
|
import { getLogger } from '../logger.js';
|
|
29
29
|
import { getCleoHome } from '../paths.js';
|
|
30
30
|
import { ensureConduitDb } from './conduit-sqlite.js';
|
|
@@ -195,7 +195,10 @@ function copyTableRows(
|
|
|
195
195
|
|
|
196
196
|
const stmt = destDb.prepare(insertSql);
|
|
197
197
|
for (const row of rows) {
|
|
198
|
-
|
|
198
|
+
// Values originate from another SQLite row via prepare().all(), so at runtime
|
|
199
|
+
// they are already SQLInputValue-compatible (string | number | bigint | Uint8Array | null).
|
|
200
|
+
// The surrounding `Record<string, unknown>` type erases this, hence the narrow cast.
|
|
201
|
+
const values = cols.map((c) => (row[c] ?? null) as SQLInputValue);
|
|
199
202
|
stmt.run(...values);
|
|
200
203
|
}
|
|
201
204
|
return rows.length;
|
|
@@ -278,7 +281,7 @@ export function migrateSignaldockToConduit(projectRoot: string): MigrationResult
|
|
|
278
281
|
// -----------------------------------------------------------------------
|
|
279
282
|
let legacy: DatabaseSync | null = null;
|
|
280
283
|
try {
|
|
281
|
-
legacy = new DatabaseSync(legacyPath, {
|
|
284
|
+
legacy = new DatabaseSync(legacyPath, { readOnly: true });
|
|
282
285
|
} catch (err) {
|
|
283
286
|
const message = err instanceof Error ? err.message : String(err);
|
|
284
287
|
log.error({ legacyPath, error: message }, 'T310 migration: cannot open legacy signaldock.db');
|
|
@@ -479,8 +482,10 @@ export function migrateSignaldockToConduit(projectRoot: string): MigrationResult
|
|
|
479
482
|
log.error({ error: message }, 'T310 migration: conduit.db integrity_check threw');
|
|
480
483
|
result.errors.push({ step: 'step-10-conduit-integrity', error: message });
|
|
481
484
|
result.status = 'failed';
|
|
482
|
-
conduit
|
|
483
|
-
|
|
485
|
+
if (conduit) {
|
|
486
|
+
conduit.close();
|
|
487
|
+
conduit = null;
|
|
488
|
+
}
|
|
484
489
|
legacy.close();
|
|
485
490
|
return result;
|
|
486
491
|
}
|
|
@@ -610,8 +615,10 @@ export function migrateSignaldockToConduit(projectRoot: string): MigrationResult
|
|
|
610
615
|
log.error({ error: message }, 'T310 migration: global signaldock.db integrity_check threw');
|
|
611
616
|
result.errors.push({ step: 'step-14-global-integrity', error: message });
|
|
612
617
|
result.status = 'failed';
|
|
613
|
-
globalDb
|
|
614
|
-
|
|
618
|
+
if (globalDb) {
|
|
619
|
+
globalDb.close();
|
|
620
|
+
globalDb = null;
|
|
621
|
+
}
|
|
615
622
|
legacy.close();
|
|
616
623
|
return result;
|
|
617
624
|
}
|