@cleocode/core 2026.4.11 → 2026.4.13

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (184) hide show
  1. package/dist/codebase-map/analyzers/architecture.d.ts.map +1 -1
  2. package/dist/codebase-map/analyzers/architecture.js +0 -1
  3. package/dist/codebase-map/analyzers/architecture.js.map +1 -1
  4. package/dist/conduit/local-transport.d.ts +18 -8
  5. package/dist/conduit/local-transport.d.ts.map +1 -1
  6. package/dist/conduit/local-transport.js +23 -13
  7. package/dist/conduit/local-transport.js.map +1 -1
  8. package/dist/config.d.ts.map +1 -1
  9. package/dist/config.js +0 -1
  10. package/dist/config.js.map +1 -1
  11. package/dist/errors.d.ts +19 -0
  12. package/dist/errors.d.ts.map +1 -1
  13. package/dist/errors.js +6 -0
  14. package/dist/errors.js.map +1 -1
  15. package/dist/index.js +175 -68950
  16. package/dist/index.js.map +1 -7
  17. package/dist/init.d.ts +1 -2
  18. package/dist/init.d.ts.map +1 -1
  19. package/dist/init.js +1 -2
  20. package/dist/init.js.map +1 -1
  21. package/dist/internal.d.ts +8 -3
  22. package/dist/internal.d.ts.map +1 -1
  23. package/dist/internal.js +13 -6
  24. package/dist/internal.js.map +1 -1
  25. package/dist/memory/learnings.d.ts +2 -2
  26. package/dist/memory/patterns.d.ts +6 -6
  27. package/dist/output.d.ts +32 -11
  28. package/dist/output.d.ts.map +1 -1
  29. package/dist/output.js +67 -67
  30. package/dist/output.js.map +1 -1
  31. package/dist/paths.js +80 -14
  32. package/dist/paths.js.map +1 -1
  33. package/dist/skills/dynamic-skill-generator.d.ts +0 -2
  34. package/dist/skills/dynamic-skill-generator.d.ts.map +1 -1
  35. package/dist/skills/dynamic-skill-generator.js.map +1 -1
  36. package/dist/store/agent-registry-accessor.d.ts +203 -12
  37. package/dist/store/agent-registry-accessor.d.ts.map +1 -1
  38. package/dist/store/agent-registry-accessor.js +618 -100
  39. package/dist/store/agent-registry-accessor.js.map +1 -1
  40. package/dist/store/api-key-kdf.d.ts +73 -0
  41. package/dist/store/api-key-kdf.d.ts.map +1 -0
  42. package/dist/store/api-key-kdf.js +84 -0
  43. package/dist/store/api-key-kdf.js.map +1 -0
  44. package/dist/store/cleanup-legacy.js +171 -0
  45. package/dist/store/cleanup-legacy.js.map +1 -0
  46. package/dist/store/conduit-sqlite.d.ts +184 -0
  47. package/dist/store/conduit-sqlite.d.ts.map +1 -0
  48. package/dist/store/conduit-sqlite.js +570 -0
  49. package/dist/store/conduit-sqlite.js.map +1 -0
  50. package/dist/store/global-salt.d.ts +78 -0
  51. package/dist/store/global-salt.d.ts.map +1 -0
  52. package/dist/store/global-salt.js +147 -0
  53. package/dist/store/global-salt.js.map +1 -0
  54. package/dist/store/migrate-signaldock-to-conduit.d.ts +81 -0
  55. package/dist/store/migrate-signaldock-to-conduit.d.ts.map +1 -0
  56. package/dist/store/migrate-signaldock-to-conduit.js +555 -0
  57. package/dist/store/migrate-signaldock-to-conduit.js.map +1 -0
  58. package/dist/store/nexus-sqlite.js +28 -3
  59. package/dist/store/nexus-sqlite.js.map +1 -1
  60. package/dist/store/signaldock-sqlite.d.ts +122 -19
  61. package/dist/store/signaldock-sqlite.d.ts.map +1 -1
  62. package/dist/store/signaldock-sqlite.js +401 -251
  63. package/dist/store/signaldock-sqlite.js.map +1 -1
  64. package/dist/store/sqlite-backup.js +122 -4
  65. package/dist/store/sqlite-backup.js.map +1 -1
  66. package/dist/system/backup.d.ts +0 -26
  67. package/dist/system/backup.d.ts.map +1 -1
  68. package/dist/system/runtime.d.ts +0 -2
  69. package/dist/system/runtime.d.ts.map +1 -1
  70. package/dist/system/runtime.js +3 -3
  71. package/dist/system/runtime.js.map +1 -1
  72. package/dist/tasks/add.d.ts +1 -1
  73. package/dist/tasks/add.d.ts.map +1 -1
  74. package/dist/tasks/add.js +98 -23
  75. package/dist/tasks/add.js.map +1 -1
  76. package/dist/tasks/complete.d.ts.map +1 -1
  77. package/dist/tasks/complete.js +4 -1
  78. package/dist/tasks/complete.js.map +1 -1
  79. package/dist/tasks/find.d.ts.map +1 -1
  80. package/dist/tasks/find.js +4 -1
  81. package/dist/tasks/find.js.map +1 -1
  82. package/dist/tasks/labels.d.ts.map +1 -1
  83. package/dist/tasks/labels.js +4 -1
  84. package/dist/tasks/labels.js.map +1 -1
  85. package/dist/tasks/relates.d.ts.map +1 -1
  86. package/dist/tasks/relates.js +16 -4
  87. package/dist/tasks/relates.js.map +1 -1
  88. package/dist/tasks/show.d.ts.map +1 -1
  89. package/dist/tasks/show.js +4 -1
  90. package/dist/tasks/show.js.map +1 -1
  91. package/dist/tasks/update.d.ts.map +1 -1
  92. package/dist/tasks/update.js +32 -6
  93. package/dist/tasks/update.js.map +1 -1
  94. package/dist/validation/engine.d.ts.map +1 -1
  95. package/dist/validation/engine.js +16 -4
  96. package/dist/validation/engine.js.map +1 -1
  97. package/dist/validation/param-utils.d.ts +5 -3
  98. package/dist/validation/param-utils.d.ts.map +1 -1
  99. package/dist/validation/param-utils.js +8 -6
  100. package/dist/validation/param-utils.js.map +1 -1
  101. package/dist/validation/protocols/_shared.d.ts.map +1 -1
  102. package/dist/validation/protocols/_shared.js +13 -6
  103. package/dist/validation/protocols/_shared.js.map +1 -1
  104. package/package.json +9 -7
  105. package/src/adapters/__tests__/manager.test.ts +0 -1
  106. package/src/codebase-map/analyzers/architecture.ts +0 -1
  107. package/src/conduit/__tests__/local-credential-flow.test.ts +20 -18
  108. package/src/conduit/__tests__/local-transport.test.ts +14 -12
  109. package/src/conduit/local-transport.ts +23 -13
  110. package/src/config.ts +0 -1
  111. package/src/errors.ts +24 -0
  112. package/src/hooks/handlers/__tests__/hook-automation-e2e.test.ts +2 -5
  113. package/src/init.ts +1 -2
  114. package/src/internal.ts +96 -2
  115. package/src/lifecycle/cant/lifecycle-rcasd.cant +133 -0
  116. package/src/memory/__tests__/engine-compat.test.ts +2 -2
  117. package/src/memory/__tests__/pipeline-manifest-sqlite.test.ts +4 -4
  118. package/src/observability/__tests__/index.test.ts +4 -4
  119. package/src/observability/__tests__/log-filter.test.ts +4 -4
  120. package/src/output.ts +73 -75
  121. package/src/sessions/__tests__/session-grade.integration.test.ts +1 -1
  122. package/src/sessions/__tests__/session-grade.test.ts +2 -2
  123. package/src/skills/__tests__/dynamic-skill-generator.test.ts +0 -2
  124. package/src/skills/dynamic-skill-generator.ts +0 -2
  125. package/src/store/__tests__/agent-registry-accessor.test.ts +807 -0
  126. package/src/store/__tests__/api-key-kdf.test.ts +113 -0
  127. package/src/store/__tests__/backup-crypto.test.ts +101 -0
  128. package/src/store/__tests__/backup-pack.test.ts +491 -0
  129. package/src/store/__tests__/backup-unpack.test.ts +298 -0
  130. package/src/store/__tests__/conduit-sqlite.test.ts +413 -0
  131. package/src/store/__tests__/global-salt.test.ts +195 -0
  132. package/src/store/__tests__/migrate-signaldock-to-conduit.test.ts +715 -0
  133. package/src/store/__tests__/regenerators.test.ts +234 -0
  134. package/src/store/__tests__/restore-conflict-report.test.ts +274 -0
  135. package/src/store/__tests__/restore-json-merge.test.ts +521 -0
  136. package/src/store/__tests__/signaldock-sqlite.test.ts +652 -0
  137. package/src/store/__tests__/sqlite-backup-global.test.ts +307 -3
  138. package/src/store/__tests__/sqlite-backup.test.ts +5 -1
  139. package/src/store/__tests__/t310-integration.test.ts +1150 -0
  140. package/src/store/__tests__/t310-readiness.test.ts +111 -0
  141. package/src/store/__tests__/t311-integration.test.ts +661 -0
  142. package/src/store/agent-registry-accessor.ts +847 -140
  143. package/src/store/api-key-kdf.ts +104 -0
  144. package/src/store/backup-crypto.ts +209 -0
  145. package/src/store/backup-pack.ts +739 -0
  146. package/src/store/backup-unpack.ts +583 -0
  147. package/src/store/conduit-sqlite.ts +655 -0
  148. package/src/store/global-salt.ts +175 -0
  149. package/src/store/migrate-signaldock-to-conduit.ts +669 -0
  150. package/src/store/regenerators.ts +243 -0
  151. package/src/store/restore-conflict-report.ts +317 -0
  152. package/src/store/restore-json-merge.ts +653 -0
  153. package/src/store/signaldock-sqlite.ts +431 -254
  154. package/src/store/sqlite-backup.ts +185 -10
  155. package/src/store/t310-readiness.ts +119 -0
  156. package/src/system/backup.ts +2 -62
  157. package/src/system/runtime.ts +4 -6
  158. package/src/tasks/__tests__/error-hints.test.ts +256 -0
  159. package/src/tasks/add.ts +99 -9
  160. package/src/tasks/complete.ts +4 -1
  161. package/src/tasks/find.ts +4 -1
  162. package/src/tasks/labels.ts +4 -1
  163. package/src/tasks/relates.ts +16 -4
  164. package/src/tasks/show.ts +4 -1
  165. package/src/tasks/update.ts +32 -3
  166. package/src/validation/__tests__/error-hints.test.ts +97 -0
  167. package/src/validation/engine.ts +16 -1
  168. package/src/validation/param-utils.ts +10 -7
  169. package/src/validation/protocols/_shared.ts +14 -6
  170. package/src/validation/protocols/cant/architecture-decision.cant +80 -0
  171. package/src/validation/protocols/cant/artifact-publish.cant +95 -0
  172. package/src/validation/protocols/cant/consensus.cant +74 -0
  173. package/src/validation/protocols/cant/contribution.cant +82 -0
  174. package/src/validation/protocols/cant/decomposition.cant +92 -0
  175. package/src/validation/protocols/cant/implementation.cant +67 -0
  176. package/src/validation/protocols/cant/provenance.cant +88 -0
  177. package/src/validation/protocols/cant/release.cant +96 -0
  178. package/src/validation/protocols/cant/research.cant +66 -0
  179. package/src/validation/protocols/cant/specification.cant +67 -0
  180. package/src/validation/protocols/cant/testing.cant +88 -0
  181. package/src/validation/protocols/cant/validation.cant +65 -0
  182. package/src/validation/protocols/protocols-markdown/decomposition.md +0 -4
  183. package/templates/config.template.json +0 -1
  184. package/templates/global-config.template.json +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
+ }