akm-cli 0.7.4 → 0.8.0-rc.10

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 (300) hide show
  1. package/CHANGELOG.md +224 -1
  2. package/README.md +22 -6
  3. package/SECURITY.md +93 -0
  4. package/dist/cli/config-migrate.js +144 -0
  5. package/dist/cli/config-validate.js +39 -0
  6. package/dist/cli/confirm.js +73 -0
  7. package/dist/cli/parse-args.js +133 -0
  8. package/dist/cli/shared.js +129 -0
  9. package/dist/cli.js +2631 -1440
  10. package/dist/commands/add-cli.js +279 -0
  11. package/dist/commands/agent-dispatch.js +110 -0
  12. package/dist/commands/agent-support.js +68 -0
  13. package/dist/commands/completions.js +3 -0
  14. package/dist/commands/config-cli.js +130 -534
  15. package/dist/commands/consolidate.js +2122 -0
  16. package/dist/commands/curate.js +45 -3
  17. package/dist/commands/db-cli.js +23 -0
  18. package/dist/commands/distill-promotion-policy.js +660 -0
  19. package/dist/commands/distill.js +1081 -73
  20. package/dist/commands/env.js +213 -0
  21. package/dist/commands/eval-cases.js +43 -0
  22. package/dist/commands/events.js +15 -24
  23. package/dist/commands/extract-cli.js +127 -0
  24. package/dist/commands/extract-prompt.js +204 -0
  25. package/dist/commands/extract.js +477 -0
  26. package/dist/commands/feedback-cli.js +331 -0
  27. package/dist/commands/graph.js +477 -0
  28. package/dist/commands/health.js +1302 -0
  29. package/dist/commands/help/help-accept.md +12 -0
  30. package/dist/commands/help/help-improve.md +69 -0
  31. package/dist/commands/help/help-proposals.md +18 -0
  32. package/dist/commands/help/help-propose.md +17 -0
  33. package/dist/commands/help/help-reject.md +11 -0
  34. package/dist/commands/history.js +54 -46
  35. package/dist/commands/improve-auto-accept.js +97 -0
  36. package/dist/commands/improve-cli.js +217 -0
  37. package/dist/commands/improve-profiles.js +166 -0
  38. package/dist/commands/improve-result-file.js +167 -0
  39. package/dist/commands/improve.js +2373 -0
  40. package/dist/commands/info.js +5 -2
  41. package/dist/commands/init.js +50 -2
  42. package/dist/commands/installed-stashes.js +102 -139
  43. package/dist/commands/knowledge.js +136 -0
  44. package/dist/commands/lint/agent-linter.js +49 -0
  45. package/dist/commands/lint/base-linter.js +479 -0
  46. package/dist/commands/lint/command-linter.js +49 -0
  47. package/dist/commands/lint/default-linter.js +16 -0
  48. package/dist/commands/lint/env-key-rules.js +154 -0
  49. package/dist/commands/lint/index.js +196 -0
  50. package/dist/commands/lint/knowledge-linter.js +16 -0
  51. package/dist/commands/lint/markdown-insertion.js +343 -0
  52. package/dist/commands/lint/memory-linter.js +61 -0
  53. package/dist/commands/lint/registry.js +36 -0
  54. package/dist/commands/lint/skill-linter.js +45 -0
  55. package/dist/commands/lint/task-linter.js +50 -0
  56. package/dist/commands/lint/types.js +4 -0
  57. package/dist/commands/lint/workflow-linter.js +56 -0
  58. package/dist/commands/lint.js +4 -0
  59. package/dist/commands/migration-help.js +3 -0
  60. package/dist/commands/proposal.js +67 -12
  61. package/dist/commands/propose.js +120 -45
  62. package/dist/commands/reflect.js +1104 -60
  63. package/dist/commands/registry-cli.js +150 -0
  64. package/dist/commands/registry-search.js +5 -2
  65. package/dist/commands/remember-cli.js +257 -0
  66. package/dist/commands/remember.js +70 -7
  67. package/dist/commands/schema-repair.js +203 -0
  68. package/dist/commands/search.js +115 -14
  69. package/dist/commands/secret.js +173 -0
  70. package/dist/commands/self-update.js +3 -0
  71. package/dist/commands/show.js +158 -60
  72. package/dist/commands/source-add.js +17 -45
  73. package/dist/commands/source-clone.js +3 -0
  74. package/dist/commands/source-manage.js +14 -19
  75. package/dist/commands/tasks.js +437 -0
  76. package/dist/commands/url-checker.js +42 -0
  77. package/dist/core/action-contributors.js +28 -0
  78. package/dist/core/asset-ref.js +17 -2
  79. package/dist/core/asset-registry.js +12 -17
  80. package/dist/core/asset-serialize.js +88 -0
  81. package/dist/core/asset-spec.js +67 -1
  82. package/dist/core/common.js +182 -0
  83. package/dist/core/concurrent.js +25 -0
  84. package/dist/core/config-io.js +347 -0
  85. package/dist/core/config-migration.js +622 -0
  86. package/dist/core/config-schema.js +534 -0
  87. package/dist/core/config-sources.js +108 -0
  88. package/dist/core/config-types.js +4 -0
  89. package/dist/core/config-walker.js +337 -0
  90. package/dist/core/config.js +364 -968
  91. package/dist/core/errors.js +42 -20
  92. package/dist/core/events.js +105 -135
  93. package/dist/core/file-lock.js +104 -0
  94. package/dist/core/frontmatter.js +75 -8
  95. package/dist/core/lesson-lint.js +3 -0
  96. package/dist/core/markdown.js +20 -0
  97. package/dist/core/memory-belief.js +62 -0
  98. package/dist/core/memory-contradiction-detect.js +274 -0
  99. package/dist/core/memory-improve.js +806 -0
  100. package/dist/core/parse.js +158 -0
  101. package/dist/core/paths.js +280 -14
  102. package/dist/core/proposal-quality-validators.js +380 -0
  103. package/dist/core/proposal-validators.js +69 -0
  104. package/dist/core/proposals.js +512 -42
  105. package/dist/core/state-db.js +1068 -0
  106. package/dist/core/text-truncation.js +107 -0
  107. package/dist/core/time.js +54 -0
  108. package/dist/core/tty.js +59 -0
  109. package/dist/core/warn.js +64 -1
  110. package/dist/core/write-source.js +3 -0
  111. package/dist/indexer/db-backup.js +391 -0
  112. package/dist/indexer/db-search.js +198 -489
  113. package/dist/indexer/db.js +990 -108
  114. package/dist/indexer/ensure-index.js +136 -0
  115. package/dist/indexer/file-context.js +3 -0
  116. package/dist/indexer/graph-boost.js +376 -101
  117. package/dist/indexer/graph-db.js +391 -0
  118. package/dist/indexer/graph-dedup.js +95 -0
  119. package/dist/indexer/graph-extraction.js +550 -114
  120. package/dist/indexer/index-context.js +4 -0
  121. package/dist/indexer/indexer.js +547 -309
  122. package/dist/indexer/llm-cache.js +52 -0
  123. package/dist/indexer/manifest.js +3 -0
  124. package/dist/indexer/matchers.js +167 -160
  125. package/dist/indexer/memory-inference.js +152 -74
  126. package/dist/indexer/metadata-contributors.js +29 -0
  127. package/dist/indexer/metadata.js +275 -196
  128. package/dist/indexer/path-resolver.js +92 -0
  129. package/dist/indexer/project-context.js +192 -0
  130. package/dist/indexer/ranking-contributors.js +331 -0
  131. package/dist/indexer/ranking.js +81 -0
  132. package/dist/indexer/search-fields.js +5 -9
  133. package/dist/indexer/search-hit-enrichers.js +111 -0
  134. package/dist/indexer/search-source.js +44 -10
  135. package/dist/indexer/semantic-status.js +6 -17
  136. package/dist/indexer/staleness-detect.js +447 -0
  137. package/dist/indexer/usage-events.js +12 -9
  138. package/dist/indexer/walker.js +28 -0
  139. package/dist/integrations/agent/builders.js +135 -0
  140. package/dist/integrations/agent/config.js +122 -230
  141. package/dist/integrations/agent/detect.js +3 -0
  142. package/dist/integrations/agent/index.js +7 -13
  143. package/dist/integrations/agent/model-aliases.js +55 -0
  144. package/dist/integrations/agent/profiles.js +70 -5
  145. package/dist/integrations/agent/prompts.js +250 -36
  146. package/dist/integrations/agent/runner.js +151 -0
  147. package/dist/integrations/agent/sdk-runner.js +126 -0
  148. package/dist/integrations/agent/spawn.js +183 -35
  149. package/dist/integrations/github.js +3 -0
  150. package/dist/integrations/lockfile.js +32 -69
  151. package/dist/integrations/session-logs/index.js +69 -0
  152. package/dist/integrations/session-logs/inline-refs.js +35 -0
  153. package/dist/integrations/session-logs/pre-filter.js +152 -0
  154. package/dist/integrations/session-logs/providers/claude-code.js +282 -0
  155. package/dist/integrations/session-logs/providers/opencode.js +258 -0
  156. package/dist/integrations/session-logs/types.js +4 -0
  157. package/dist/llm/call-ai.js +62 -0
  158. package/dist/llm/client.js +79 -88
  159. package/dist/llm/embedder.js +20 -29
  160. package/dist/llm/embedders/cache.js +3 -7
  161. package/dist/llm/embedders/local.js +42 -1
  162. package/dist/llm/embedders/remote.js +20 -8
  163. package/dist/llm/embedders/types.js +3 -7
  164. package/dist/llm/feature-gate.js +95 -48
  165. package/dist/llm/graph-extract.js +676 -72
  166. package/dist/llm/index-passes.js +44 -29
  167. package/dist/llm/memory-infer.js +80 -71
  168. package/dist/llm/metadata-enhance.js +42 -29
  169. package/dist/llm/prompts/extract-session.md +80 -0
  170. package/dist/llm/prompts/graph-extract-user-prompt.md +35 -0
  171. package/dist/output/cli-hints-full.md +292 -0
  172. package/dist/output/cli-hints-short.md +66 -0
  173. package/dist/output/cli-hints.js +7 -311
  174. package/dist/output/context.js +60 -8
  175. package/dist/output/renderers.js +306 -258
  176. package/dist/output/shapes/curate.js +56 -0
  177. package/dist/output/shapes/distill.js +10 -0
  178. package/dist/output/shapes/env-list.js +19 -0
  179. package/dist/output/shapes/events.js +11 -0
  180. package/dist/output/shapes/helpers.js +424 -0
  181. package/dist/output/shapes/history.js +7 -0
  182. package/dist/output/shapes/passthrough.js +102 -0
  183. package/dist/output/shapes/proposal-accept.js +7 -0
  184. package/dist/output/shapes/proposal-diff.js +7 -0
  185. package/dist/output/shapes/proposal-list.js +7 -0
  186. package/dist/output/shapes/proposal-producer.js +11 -0
  187. package/dist/output/shapes/proposal-reject.js +7 -0
  188. package/dist/output/shapes/proposal-show.js +7 -0
  189. package/dist/output/shapes/registry-search.js +6 -0
  190. package/dist/output/shapes/registry.js +30 -0
  191. package/dist/output/shapes/search.js +6 -0
  192. package/dist/output/shapes/secret-list.js +19 -0
  193. package/dist/output/shapes/show.js +6 -0
  194. package/dist/output/shapes/vault-list.js +19 -0
  195. package/dist/output/shapes.js +51 -511
  196. package/dist/output/text/add.js +6 -0
  197. package/dist/output/text/clone.js +6 -0
  198. package/dist/output/text/config.js +6 -0
  199. package/dist/output/text/curate.js +6 -0
  200. package/dist/output/text/distill.js +7 -0
  201. package/dist/output/text/enable-disable.js +7 -0
  202. package/dist/output/text/events.js +10 -0
  203. package/dist/output/text/feedback.js +6 -0
  204. package/dist/output/text/helpers.js +1039 -0
  205. package/dist/output/text/history.js +7 -0
  206. package/dist/output/text/import.js +6 -0
  207. package/dist/output/text/index.js +6 -0
  208. package/dist/output/text/info.js +6 -0
  209. package/dist/output/text/init.js +6 -0
  210. package/dist/output/text/list.js +6 -0
  211. package/dist/output/text/proposal-producer.js +8 -0
  212. package/dist/output/text/proposal.js +11 -0
  213. package/dist/output/text/registry-commands.js +11 -0
  214. package/dist/output/text/registry.js +30 -0
  215. package/dist/output/text/remember.js +6 -0
  216. package/dist/output/text/remove.js +6 -0
  217. package/dist/output/text/save.js +6 -0
  218. package/dist/output/text/search.js +6 -0
  219. package/dist/output/text/show.js +6 -0
  220. package/dist/output/text/update.js +6 -0
  221. package/dist/output/text/upgrade.js +6 -0
  222. package/dist/output/text/vault.js +16 -0
  223. package/dist/output/text/wiki.js +15 -0
  224. package/dist/output/text/workflow.js +14 -0
  225. package/dist/output/text.js +44 -1093
  226. package/dist/registry/build-index.js +3 -0
  227. package/dist/registry/create-provider-registry.js +3 -0
  228. package/dist/registry/factory.js +4 -1
  229. package/dist/registry/origin-resolve.js +3 -0
  230. package/dist/registry/providers/index.js +3 -0
  231. package/dist/registry/providers/skills-sh.js +71 -50
  232. package/dist/registry/providers/static-index.js +53 -48
  233. package/dist/registry/providers/types.js +3 -24
  234. package/dist/registry/resolve.js +11 -16
  235. package/dist/registry/types.js +3 -0
  236. package/dist/scripts/migrate-storage.js +17750 -0
  237. package/dist/scripts/migrations/import-fs-improve-runs-to-db.js +9031 -0
  238. package/dist/scripts/migrations/v16-to-v17.js +141 -0
  239. package/dist/setup/detect.js +3 -0
  240. package/dist/setup/ripgrep-install.js +3 -0
  241. package/dist/setup/ripgrep-resolve.js +3 -0
  242. package/dist/setup/setup.js +775 -37
  243. package/dist/setup/steps.js +3 -15
  244. package/dist/sources/include.js +3 -0
  245. package/dist/sources/provider-factory.js +5 -12
  246. package/dist/sources/provider.js +3 -20
  247. package/dist/sources/providers/filesystem.js +19 -23
  248. package/dist/sources/providers/git.js +179 -20
  249. package/dist/sources/providers/index.js +3 -0
  250. package/dist/sources/providers/install-types.js +3 -13
  251. package/dist/sources/providers/npm.js +3 -4
  252. package/dist/sources/providers/provider-utils.js +3 -0
  253. package/dist/sources/providers/sync-from-ref.js +3 -11
  254. package/dist/sources/providers/tar-utils.js +3 -0
  255. package/dist/sources/providers/website.js +18 -22
  256. package/dist/sources/resolve.js +3 -0
  257. package/dist/sources/types.js +3 -0
  258. package/dist/sources/website-ingest.js +7 -0
  259. package/dist/tasks/backends/cron.js +203 -0
  260. package/dist/tasks/backends/exec-utils.js +28 -0
  261. package/dist/tasks/backends/index.js +24 -0
  262. package/dist/tasks/backends/launchd-template.xml +19 -0
  263. package/dist/tasks/backends/launchd.js +187 -0
  264. package/dist/tasks/backends/schtasks-template.xml +29 -0
  265. package/dist/tasks/backends/schtasks.js +215 -0
  266. package/dist/tasks/parser.js +211 -0
  267. package/dist/tasks/resolveAkmBin.js +87 -0
  268. package/dist/tasks/runner.js +458 -0
  269. package/dist/tasks/schedule.js +227 -0
  270. package/dist/tasks/schema.js +15 -0
  271. package/dist/tasks/validator.js +62 -0
  272. package/dist/version.js +3 -0
  273. package/dist/wiki/index-template.md +12 -0
  274. package/dist/wiki/ingest-workflow-template.md +54 -0
  275. package/dist/wiki/log-template.md +8 -0
  276. package/dist/wiki/schema-template.md +61 -0
  277. package/dist/wiki/wiki-templates.js +15 -0
  278. package/dist/wiki/wiki.js +13 -61
  279. package/dist/workflows/authoring.js +8 -25
  280. package/dist/workflows/cli.js +3 -0
  281. package/dist/workflows/db.js +141 -2
  282. package/dist/workflows/document-cache.js +3 -10
  283. package/dist/workflows/parser.js +3 -0
  284. package/dist/workflows/renderer.js +11 -3
  285. package/dist/workflows/runs.js +91 -89
  286. package/dist/workflows/schema.js +3 -0
  287. package/dist/workflows/scope-key.js +79 -0
  288. package/dist/workflows/validator.js +4 -8
  289. package/dist/workflows/workflow-template.md +24 -0
  290. package/docs/README.md +10 -2
  291. package/docs/data-and-telemetry.md +225 -0
  292. package/docs/migration/release-notes/0.7.0.md +1 -1
  293. package/docs/migration/release-notes/0.7.4.md +1 -1
  294. package/docs/migration/release-notes/0.7.5.md +20 -0
  295. package/docs/migration/release-notes/0.8.0.md +48 -0
  296. package/docs/migration/v0.7-to-v0.8.md +1307 -0
  297. package/package.json +29 -11
  298. package/dist/commands/install-audit.js +0 -381
  299. package/dist/commands/vault.js +0 -333
  300. package/dist/templates/wiki-templates.js +0 -100
@@ -0,0 +1,391 @@
1
+ // This Source Code Form is subject to the terms of the Mozilla Public
2
+ // License, v. 2.0. If a copy of the MPL was not distributed with this
3
+ // file, You can obtain one at https://mozilla.org/MPL/2.0/.
4
+ /**
5
+ * MVP data-directory backup for AKM.
6
+ *
7
+ * The DB upgrade path in `src/indexer/db.ts` `handleVersionUpgrade()` is
8
+ * intentionally destructive: when `DB_VERSION` bumps and a stored DB is at an
9
+ * older version, ~17 tables are dropped and recreated. Until 0.9.0 ships a
10
+ * full migration framework, this MVP captures a recursive copy of the entire
11
+ * data directory just before that drop happens so an operator can manually
12
+ * recover lost rows by stopping akm and moving the backup contents back over
13
+ * the live data dir (see `scripts/migrations/restore-data-dir.sh`).
14
+ *
15
+ * The helper is intentionally narrow:
16
+ * - No `VACUUM INTO`, no selective table backup — just `fs.cpSync` of the
17
+ * data directory into `<dataDir>/backups/<timestamp>-pre-v<targetVersion>/`.
18
+ * - Skips the `backups/` subdirectory inside the data dir so we never
19
+ * recurse into our own backup history.
20
+ * - Opt-out via `AKM_DB_BACKUP=0`. Backup failures NEVER abort the upgrade —
21
+ * they warn and proceed (the alternative would brick a user trying to
22
+ * start a binary that bumped DB_VERSION on a full disk).
23
+ * - Retention is FIFO with default of 5, configurable via
24
+ * `AKM_DB_BACKUP_RETAIN`.
25
+ * - Disk-space guard: refuses to write when free space on the destination
26
+ * filesystem is less than 1.1× the source size.
27
+ */
28
+ import fs from "node:fs";
29
+ import path from "node:path";
30
+ import { warn } from "../core/warn";
31
+ /** Default reason recorded for backups that don't override it. */
32
+ export const DEFAULT_BACKUP_REASON = "version-upgrade";
33
+ /** Reason recorded for backups taken before the embedding-dim drop path. */
34
+ export const EMBEDDING_DIM_CHANGE_REASON = "embedding-dim-change";
35
+ const BACKUPS_DIR_NAME = "backups";
36
+ const BACKUP_METADATA_FILE = "backup.meta.json";
37
+ const DEFAULT_RETAIN = 5;
38
+ const FREE_SPACE_MULTIPLIER = 1.1;
39
+ /**
40
+ * Resolve the configured retention count from the env, with a hard floor of 1.
41
+ *
42
+ * Invalid values (non-integer, negative) fall back to the default and emit a
43
+ * one-line warning so operators notice their env var is wrong.
44
+ */
45
+ export function resolveRetention(env = process.env) {
46
+ const raw = env.AKM_DB_BACKUP_RETAIN?.trim();
47
+ if (!raw)
48
+ return DEFAULT_RETAIN;
49
+ const parsed = Number.parseInt(raw, 10);
50
+ if (Number.isNaN(parsed) || parsed < 1) {
51
+ warn("[akm] AKM_DB_BACKUP_RETAIN=%s is not a positive integer; falling back to %d", raw, DEFAULT_RETAIN);
52
+ return DEFAULT_RETAIN;
53
+ }
54
+ return parsed;
55
+ }
56
+ /**
57
+ * Returns true when the user has explicitly opted out via `AKM_DB_BACKUP=0`
58
+ * (or `false`/`no`/`off`). Any other value — including unset — opts in.
59
+ */
60
+ export function isBackupDisabled(env = process.env) {
61
+ const raw = env.AKM_DB_BACKUP?.trim().toLowerCase();
62
+ if (!raw)
63
+ return false;
64
+ return raw === "0" || raw === "false" || raw === "no" || raw === "off";
65
+ }
66
+ /**
67
+ * Recursively sum the byte size of `dirPath`, skipping the embedded backups
68
+ * directory so the size we report (and check against free space) reflects
69
+ * what we'd actually copy.
70
+ */
71
+ export function measureDataDirSize(dirPath) {
72
+ if (!fs.existsSync(dirPath))
73
+ return 0;
74
+ let total = 0;
75
+ const stack = [dirPath];
76
+ while (stack.length > 0) {
77
+ const current = stack.pop();
78
+ if (current === undefined)
79
+ break;
80
+ let entries;
81
+ try {
82
+ entries = fs.readdirSync(current, { withFileTypes: true });
83
+ }
84
+ catch {
85
+ // Unreadable directory — skip; we don't want measurement to throw.
86
+ continue;
87
+ }
88
+ for (const entry of entries) {
89
+ const full = path.join(current, entry.name);
90
+ // Skip the embedded backups directory at the root so we don't
91
+ // double-count prior backups in size calculations.
92
+ if (current === dirPath && entry.name === BACKUPS_DIR_NAME && entry.isDirectory())
93
+ continue;
94
+ if (entry.isDirectory()) {
95
+ stack.push(full);
96
+ }
97
+ else if (entry.isFile()) {
98
+ try {
99
+ total += fs.statSync(full).size;
100
+ }
101
+ catch {
102
+ // File vanished between readdir and stat — ignore.
103
+ }
104
+ }
105
+ }
106
+ }
107
+ return total;
108
+ }
109
+ /**
110
+ * Best-effort free-space query for the filesystem hosting `dirPath`. Returns
111
+ * `null` when the runtime cannot report statfs (older Node/Bun, exotic FS) —
112
+ * the caller treats `null` as "skip the disk-space check" rather than
113
+ * "abort the backup".
114
+ */
115
+ function getFreeSpace(dirPath) {
116
+ try {
117
+ // `fs.statfsSync` is available in Node 18.15+ and Bun 1.0+.
118
+ const stats = fs.statfsSync;
119
+ if (!stats)
120
+ return null;
121
+ const res = stats(dirPath);
122
+ return Number(res.bavail * res.bsize);
123
+ }
124
+ catch {
125
+ return null;
126
+ }
127
+ }
128
+ /**
129
+ * Format the current time into a filename-safe timestamp.
130
+ *
131
+ * Example: `2026-05-19T04-59-36`.
132
+ */
133
+ function formatTimestamp(d) {
134
+ // ISO 8601 without colons/dots so the path is portable to Windows + tarballs.
135
+ return d
136
+ .toISOString()
137
+ .replace(/[:.]/g, "-")
138
+ .replace(/Z$/, "")
139
+ .replace(/-\d{3}$/, "");
140
+ }
141
+ export function listBackups(dataDir) {
142
+ const backupsRoot = path.join(dataDir, BACKUPS_DIR_NAME);
143
+ if (!fs.existsSync(backupsRoot))
144
+ return [];
145
+ const entries = fs.readdirSync(backupsRoot, { withFileTypes: true });
146
+ const results = [];
147
+ for (const entry of entries) {
148
+ if (!entry.isDirectory())
149
+ continue;
150
+ const full = path.join(backupsRoot, entry.name);
151
+ const metaPath = path.join(full, BACKUP_METADATA_FILE);
152
+ let createdAt;
153
+ let sourceVersion = null;
154
+ let sizeBytes;
155
+ let reason = DEFAULT_BACKUP_REASON;
156
+ if (fs.existsSync(metaPath)) {
157
+ try {
158
+ const raw = fs.readFileSync(metaPath, "utf8");
159
+ const parsed = JSON.parse(raw);
160
+ if (typeof parsed.createdAt === "string")
161
+ createdAt = parsed.createdAt;
162
+ if (typeof parsed.sourceVersion === "number")
163
+ sourceVersion = parsed.sourceVersion;
164
+ else if (parsed.sourceVersion === null)
165
+ sourceVersion = null;
166
+ if (typeof parsed.sizeBytes === "number")
167
+ sizeBytes = parsed.sizeBytes;
168
+ if (typeof parsed.reason === "string" && parsed.reason.length > 0)
169
+ reason = parsed.reason;
170
+ }
171
+ catch {
172
+ // Malformed metadata — fall back to filesystem-derived values.
173
+ }
174
+ }
175
+ if (!createdAt) {
176
+ try {
177
+ createdAt = fs.statSync(full).mtime.toISOString();
178
+ }
179
+ catch {
180
+ createdAt = new Date(0).toISOString();
181
+ }
182
+ }
183
+ if (sizeBytes === undefined) {
184
+ sizeBytes = measureDataDirSize(full);
185
+ }
186
+ results.push({
187
+ path: full,
188
+ name: entry.name,
189
+ createdAt,
190
+ sizeBytes,
191
+ sourceVersion,
192
+ reason,
193
+ });
194
+ }
195
+ // Sort newest first.
196
+ results.sort((a, b) => (a.createdAt < b.createdAt ? 1 : a.createdAt > b.createdAt ? -1 : 0));
197
+ return results;
198
+ }
199
+ /**
200
+ * Drop oldest backups until at most `retain` remain. The newest backup (the
201
+ * one we just created) is always preserved — pruning happens AFTER the new
202
+ * backup is written, so `retain=5` plus a fresh write means we keep the new
203
+ * write and prune down to 5 total entries.
204
+ */
205
+ function pruneOldBackups(dataDir, retain) {
206
+ const existing = listBackups(dataDir);
207
+ if (existing.length <= retain)
208
+ return;
209
+ const toRemove = existing.slice(retain);
210
+ for (const entry of toRemove) {
211
+ try {
212
+ fs.rmSync(entry.path, { recursive: true, force: true });
213
+ }
214
+ catch (err) {
215
+ warn("[akm] failed to prune old backup %s — %s", entry.path, err instanceof Error ? err.message : String(err));
216
+ }
217
+ }
218
+ }
219
+ /**
220
+ * Capture a recursive copy of `dataDir` under `<dataDir>/backups/`, skipping
221
+ * the backups subdirectory itself. Returns the BackupResult on success or
222
+ * `null` when the backup was skipped (opt-out, missing data dir, insufficient
223
+ * disk space, or a copy error — all of which should be non-fatal so the
224
+ * upgrade path can still proceed).
225
+ */
226
+ export function backupDataDir(opts) {
227
+ const env = opts.env ?? process.env;
228
+ if (isBackupDisabled(env))
229
+ return null;
230
+ const { dataDir, sourceVersion, targetVersion } = opts;
231
+ const reason = opts.reason && opts.reason.length > 0 ? opts.reason : DEFAULT_BACKUP_REASON;
232
+ if (!fs.existsSync(dataDir)) {
233
+ // Fresh install — nothing to back up.
234
+ return null;
235
+ }
236
+ const dataDirStat = fs.statSync(dataDir);
237
+ if (!dataDirStat.isDirectory()) {
238
+ warn("[akm] data dir backup skipped — %s is not a directory", dataDir);
239
+ return null;
240
+ }
241
+ // Empty data dir (or only a `backups/` subdir) → nothing meaningful to back up.
242
+ const sourceSize = measureDataDirSize(dataDir);
243
+ if (sourceSize === 0)
244
+ return null;
245
+ const backupsRoot = path.join(dataDir, BACKUPS_DIR_NAME);
246
+ try {
247
+ fs.mkdirSync(backupsRoot, { recursive: true });
248
+ }
249
+ catch (err) {
250
+ warn("[akm] data dir backup skipped — could not create %s: %s", backupsRoot, err instanceof Error ? err.message : String(err));
251
+ return null;
252
+ }
253
+ // Disk-space guard. Skip the check when statfs is unavailable.
254
+ const free = getFreeSpace(backupsRoot);
255
+ if (free !== null && free < sourceSize * FREE_SPACE_MULTIPLIER) {
256
+ warn("[akm] data dir backup skipped — free space %d bytes is less than 1.1× source size %d bytes (need %d)", free, sourceSize, Math.ceil(sourceSize * FREE_SPACE_MULTIPLIER));
257
+ return null;
258
+ }
259
+ const now = (opts.now ?? (() => new Date()))();
260
+ const stamp = formatTimestamp(now);
261
+ // Reason tags drive the directory suffix so operators can tell a
262
+ // version-upgrade snapshot apart from an embedding-dim-change snapshot.
263
+ // `version-upgrade` keeps the historical `pre-v<N>` suffix for backward
264
+ // compatibility with `scripts/migrations/restore-data-dir.sh` and existing
265
+ // tests; any other reason is appended verbatim.
266
+ const dirSuffix = reason === DEFAULT_BACKUP_REASON ? `pre-v${targetVersion}` : reason;
267
+ const dirName = `${stamp}-${dirSuffix}`;
268
+ const destPath = path.join(backupsRoot, dirName);
269
+ // If a previous run on the same second tried to write this name, append a
270
+ // short disambiguator. We don't want to overwrite or merge into an existing
271
+ // backup directory.
272
+ let finalDest = destPath;
273
+ let suffix = 1;
274
+ while (fs.existsSync(finalDest)) {
275
+ finalDest = `${destPath}-${suffix}`;
276
+ suffix += 1;
277
+ }
278
+ try {
279
+ // We can't use fs.cpSync directly because the destination
280
+ // (<dataDir>/backups/<stamp>-pre-v<N>/) is inside the source dataDir, and
281
+ // cpSync refuses to copy into a subdirectory of the source. So we do a
282
+ // manual recursive walk that explicitly skips the backups subtree, plus
283
+ // the lockfile/sentinel that would race with any live process.
284
+ copyDataDirExcludingBackups(dataDir, finalDest);
285
+ }
286
+ catch (err) {
287
+ warn("[akm] data dir backup failed — %s; upgrade will proceed without a snapshot", err instanceof Error ? err.message : String(err));
288
+ // Best-effort cleanup of the partial copy so we don't litter the data dir.
289
+ try {
290
+ fs.rmSync(finalDest, { recursive: true, force: true });
291
+ }
292
+ catch {
293
+ /* ignore */
294
+ }
295
+ return null;
296
+ }
297
+ const createdAt = now.toISOString();
298
+ const metadata = {
299
+ schemaVersion: 1,
300
+ createdAt,
301
+ sourceVersion,
302
+ targetVersion,
303
+ sizeBytes: sourceSize,
304
+ reason,
305
+ hostname: tryHostname(),
306
+ notes: reason === DEFAULT_BACKUP_REASON
307
+ ? "Created by AKM before a destructive DB version upgrade. Restore manually by stopping akm and copying the contents back over the live data dir."
308
+ : `Created by AKM before a destructive ${reason} operation. Restore manually by stopping akm and copying the contents back over the live data dir.`,
309
+ };
310
+ try {
311
+ fs.writeFileSync(path.join(finalDest, BACKUP_METADATA_FILE), JSON.stringify(metadata, null, 2));
312
+ }
313
+ catch (err) {
314
+ // Metadata is non-essential — warn but keep the copy.
315
+ warn("[akm] data dir backup created at %s but metadata write failed — %s", finalDest, err instanceof Error ? err.message : String(err));
316
+ }
317
+ const retain = resolveRetention(env);
318
+ pruneOldBackups(dataDir, retain);
319
+ return {
320
+ path: finalDest,
321
+ name: path.basename(finalDest),
322
+ createdAt,
323
+ sizeBytes: sourceSize,
324
+ sourceVersion,
325
+ targetVersion,
326
+ reason,
327
+ };
328
+ }
329
+ /**
330
+ * Recursively copy `srcRoot` to `destRoot`, skipping:
331
+ * - `<srcRoot>/backups` (so we don't recurse into our own backup history)
332
+ * - `<srcRoot>/akm.lock` and `<srcRoot>/akm.lock.lck` (per-process state
333
+ * that would race with a live process holding the lock)
334
+ *
335
+ * Implemented manually because `fs.cpSync` refuses to copy a directory into a
336
+ * subdirectory of itself, and our destination (`<dataDir>/backups/<stamp>`)
337
+ * is by design inside the source `<dataDir>`.
338
+ */
339
+ function copyDataDirExcludingBackups(srcRoot, destRoot) {
340
+ fs.mkdirSync(destRoot, { recursive: true });
341
+ const stack = [{ src: srcRoot, dest: destRoot }];
342
+ while (stack.length > 0) {
343
+ const frame = stack.pop();
344
+ if (frame === undefined)
345
+ break;
346
+ const { src, dest } = frame;
347
+ const entries = fs.readdirSync(src, { withFileTypes: true });
348
+ for (const entry of entries) {
349
+ // Skip the embedded backups directory and the lockfile/sentinel — only
350
+ // at the root level. (A `backups` directory deep in a wiki source tree,
351
+ // for instance, must still be copied.)
352
+ if (src === srcRoot) {
353
+ if (entry.name === BACKUPS_DIR_NAME && entry.isDirectory())
354
+ continue;
355
+ if (entry.name === "akm.lock" || entry.name === "akm.lock.lck")
356
+ continue;
357
+ }
358
+ const srcPath = path.join(src, entry.name);
359
+ const destPath = path.join(dest, entry.name);
360
+ if (entry.isDirectory()) {
361
+ fs.mkdirSync(destPath, { recursive: true });
362
+ stack.push({ src: srcPath, dest: destPath });
363
+ }
364
+ else if (entry.isFile()) {
365
+ fs.copyFileSync(srcPath, destPath);
366
+ }
367
+ else if (entry.isSymbolicLink()) {
368
+ // Preserve symlinks as-is rather than dereferencing them. A stash
369
+ // dir occasionally carries symlinked source roots; following them
370
+ // could explode the backup size unexpectedly.
371
+ const target = fs.readlinkSync(srcPath);
372
+ try {
373
+ fs.symlinkSync(target, destPath);
374
+ }
375
+ catch {
376
+ /* ignore — symlink creation can fail on Windows without admin */
377
+ }
378
+ }
379
+ // Other entry types (block/character/fifo/socket) are silently skipped.
380
+ }
381
+ }
382
+ }
383
+ function tryHostname() {
384
+ try {
385
+ const os = require("node:os");
386
+ return os.hostname();
387
+ }
388
+ catch {
389
+ return undefined;
390
+ }
391
+ }