akm-cli 0.6.0 → 0.7.0-rc1

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 (319) hide show
  1. package/CHANGELOG.md +66 -0
  2. package/dist/{cli.js → src/cli.js} +672 -29
  3. package/dist/{commands → src/commands}/config-cli.js +5 -4
  4. package/dist/src/commands/distill.js +283 -0
  5. package/dist/src/commands/events.js +108 -0
  6. package/dist/src/commands/history.js +120 -0
  7. package/dist/{commands → src/commands}/installed-stashes.js +28 -2
  8. package/dist/src/commands/proposal.js +119 -0
  9. package/dist/src/commands/propose.js +171 -0
  10. package/dist/src/commands/reflect.js +193 -0
  11. package/dist/{commands → src/commands}/registry-search.js +2 -1
  12. package/dist/{commands → src/commands}/remember.js +12 -0
  13. package/dist/{commands → src/commands}/search.js +74 -1
  14. package/dist/{commands → src/commands}/self-update.js +4 -3
  15. package/dist/{commands → src/commands}/show.js +67 -2
  16. package/dist/{core → src/core}/asset-ref.js +5 -5
  17. package/dist/{core → src/core}/asset-spec.js +12 -0
  18. package/dist/{core → src/core}/common.js +1 -1
  19. package/dist/{core → src/core}/config.js +175 -121
  20. package/dist/{core → src/core}/errors.js +4 -0
  21. package/dist/src/core/events.js +239 -0
  22. package/dist/src/core/lesson-lint.js +86 -0
  23. package/dist/src/core/proposals.js +406 -0
  24. package/dist/src/core/warn.js +72 -0
  25. package/dist/{core → src/core}/write-source.js +80 -5
  26. package/dist/{indexer → src/indexer}/db-search.js +119 -27
  27. package/dist/{indexer → src/indexer}/db.js +76 -23
  28. package/dist/{indexer → src/indexer}/file-context.js +0 -3
  29. package/dist/src/indexer/graph-boost.js +179 -0
  30. package/dist/src/indexer/graph-extraction.js +212 -0
  31. package/dist/{indexer → src/indexer}/indexer.js +73 -6
  32. package/dist/src/indexer/memory-inference.js +263 -0
  33. package/dist/{indexer → src/indexer}/metadata.js +114 -11
  34. package/dist/src/integrations/agent/config.js +292 -0
  35. package/dist/src/integrations/agent/detect.js +94 -0
  36. package/dist/src/integrations/agent/index.js +17 -0
  37. package/dist/src/integrations/agent/profiles.js +65 -0
  38. package/dist/src/integrations/agent/prompts.js +167 -0
  39. package/dist/src/integrations/agent/spawn.js +221 -0
  40. package/dist/{integrations → src/integrations}/lockfile.js +0 -26
  41. package/dist/{llm → src/llm}/client.js +33 -2
  42. package/dist/src/llm/feature-gate.js +108 -0
  43. package/dist/src/llm/graph-extract.js +107 -0
  44. package/dist/src/llm/index-passes.js +35 -0
  45. package/dist/src/llm/memory-infer.js +86 -0
  46. package/dist/{output → src/output}/renderers.js +60 -1
  47. package/dist/src/output/shapes.js +516 -0
  48. package/dist/{output → src/output}/text.js +447 -4
  49. package/dist/{registry → src/registry}/build-index.js +14 -4
  50. package/dist/{registry → src/registry}/factory.js +0 -8
  51. package/dist/{registry → src/registry}/providers/static-index.js +3 -2
  52. package/dist/{registry → src/registry}/resolve.js +68 -2
  53. package/dist/{setup → src/setup}/setup.js +43 -5
  54. package/dist/{sources → src/sources}/providers/git.js +7 -15
  55. package/dist/{wiki → src/wiki}/wiki.js +9 -11
  56. package/dist/tests/add-website-source.test.js +119 -0
  57. package/dist/tests/agent/agent-config-loader.test.js +70 -0
  58. package/dist/tests/agent/agent-config.test.js +221 -0
  59. package/dist/tests/agent/agent-detect.test.js +100 -0
  60. package/dist/tests/agent/agent-spawn.test.js +234 -0
  61. package/dist/tests/agent-output.test.js +186 -0
  62. package/dist/tests/architecture/agent-no-llm-sdk-guard.test.js +103 -0
  63. package/dist/tests/architecture/agent-spawn-seam.test.js +193 -0
  64. package/dist/tests/architecture/llm-stateless-seam.test.js +112 -0
  65. package/dist/tests/asset-ref.test.js +192 -0
  66. package/dist/tests/asset-registry.test.js +103 -0
  67. package/dist/tests/asset-spec.test.js +241 -0
  68. package/dist/tests/bench/attribution.test.js +995 -0
  69. package/dist/tests/bench/cleanup-sigint.test.js +83 -0
  70. package/dist/tests/bench/cleanup.js +203 -0
  71. package/dist/tests/bench/cleanup.test.js +166 -0
  72. package/dist/tests/bench/cli.js +683 -0
  73. package/dist/tests/bench/cli.test.js +177 -0
  74. package/dist/tests/bench/compare.test.js +556 -0
  75. package/dist/tests/bench/corpus.js +314 -0
  76. package/dist/tests/bench/corpus.test.js +258 -0
  77. package/dist/tests/bench/driver.js +346 -0
  78. package/dist/tests/bench/driver.test.js +443 -0
  79. package/dist/tests/bench/evolve-metrics.js +179 -0
  80. package/dist/tests/bench/evolve-metrics.test.js +187 -0
  81. package/dist/tests/bench/evolve.js +580 -0
  82. package/dist/tests/bench/evolve.test.js +616 -0
  83. package/dist/tests/bench/failure-modes.test.js +300 -0
  84. package/dist/tests/bench/feedback-integrity.test.js +456 -0
  85. package/dist/tests/bench/leakage.test.js +125 -0
  86. package/dist/tests/bench/learning-curve.test.js +133 -0
  87. package/dist/tests/bench/metrics.js +2319 -0
  88. package/dist/tests/bench/metrics.test.js +1144 -0
  89. package/dist/tests/bench/no-os-tmpdir-invariant.test.js +43 -0
  90. package/dist/tests/bench/report.js +1821 -0
  91. package/dist/tests/bench/report.test.js +989 -0
  92. package/dist/tests/bench/runner.js +536 -0
  93. package/dist/tests/bench/runner.test.js +958 -0
  94. package/dist/tests/bench/search-bridge.test.js +331 -0
  95. package/dist/tests/bench/tmp.js +41 -0
  96. package/dist/tests/bench/trajectory.js +116 -0
  97. package/dist/tests/bench/trajectory.test.js +127 -0
  98. package/dist/tests/bench/verifier.js +109 -0
  99. package/dist/tests/bench/verifier.test.js +118 -0
  100. package/dist/tests/bench/workflow-evaluator.js +557 -0
  101. package/dist/tests/bench/workflow-evaluator.test.js +421 -0
  102. package/dist/tests/bench/workflow-spec.js +358 -0
  103. package/dist/tests/bench/workflow-spec.test.js +363 -0
  104. package/dist/tests/bench/workflow-trace.js +438 -0
  105. package/dist/tests/bench/workflow-trace.test.js +254 -0
  106. package/dist/tests/benchmark-search-quality.js +536 -0
  107. package/dist/tests/benchmark-suite.js +1441 -0
  108. package/dist/tests/capture-cli.test.js +112 -0
  109. package/dist/tests/cli-errors.test.js +203 -0
  110. package/dist/tests/commands/events.test.js +370 -0
  111. package/dist/tests/commands/history.test.js +223 -0
  112. package/dist/tests/commands/import.test.js +103 -0
  113. package/dist/tests/commands/proposal-cli.test.js +209 -0
  114. package/dist/tests/commands/reflect-propose-cli.test.js +333 -0
  115. package/dist/tests/commands/remember.test.js +97 -0
  116. package/dist/tests/commands/scope-flags.test.js +300 -0
  117. package/dist/tests/commands/search.test.js +537 -0
  118. package/dist/tests/commands/show-indexer-parity.test.js +117 -0
  119. package/dist/tests/commands/show.test.js +294 -0
  120. package/dist/tests/common.test.js +266 -0
  121. package/dist/tests/completions.test.js +142 -0
  122. package/dist/tests/config-cli.test.js +193 -0
  123. package/dist/tests/config-llm-features.test.js +139 -0
  124. package/dist/tests/config.test.js +544 -0
  125. package/dist/tests/contracts/migration-baseline.test.js +43 -0
  126. package/dist/tests/contracts/reflect-propose-envelope.test.js +139 -0
  127. package/dist/tests/contracts/spec-helpers.js +46 -0
  128. package/dist/tests/contracts/v1-spec-section-11-proposal-queue.test.js +228 -0
  129. package/dist/tests/contracts/v1-spec-section-12-agent-config.test.js +56 -0
  130. package/dist/tests/contracts/v1-spec-section-13-lesson-type.test.js +34 -0
  131. package/dist/tests/contracts/v1-spec-section-14-llm-features.test.js +94 -0
  132. package/dist/tests/contracts/v1-spec-section-4-1-asset-types.test.js +39 -0
  133. package/dist/tests/contracts/v1-spec-section-4-2-quality-rules.test.js +44 -0
  134. package/dist/tests/contracts/v1-spec-section-5-configuration.test.js +47 -0
  135. package/dist/tests/contracts/v1-spec-section-6-orchestration.test.js +40 -0
  136. package/dist/tests/contracts/v1-spec-section-7-module-layout.test.js +58 -0
  137. package/dist/tests/contracts/v1-spec-section-8-extension-points.test.js +34 -0
  138. package/dist/tests/contracts/v1-spec-section-9-4-cli-surface.test.js +75 -0
  139. package/dist/tests/contracts/v1-spec-section-9-7-llm-agent-boundary.test.js +36 -0
  140. package/dist/tests/core/write-source.test.js +366 -0
  141. package/dist/tests/curate-command.test.js +87 -0
  142. package/dist/tests/db-scoring.test.js +201 -0
  143. package/dist/tests/db.test.js +654 -0
  144. package/dist/tests/distill-cli-flag.test.js +208 -0
  145. package/dist/tests/distill.test.js +515 -0
  146. package/dist/tests/docker-install.test.js +120 -0
  147. package/dist/tests/e2e.test.js +1398 -0
  148. package/dist/tests/embedder.test.js +340 -0
  149. package/dist/tests/embedding-model-config.test.js +379 -0
  150. package/dist/tests/feedback-command.test.js +172 -0
  151. package/dist/tests/file-context.test.js +552 -0
  152. package/dist/tests/fixtures/scripts/git/summarize-diff.js +9 -0
  153. package/dist/tests/fixtures/scripts/lint/eslint-check.js +7 -0
  154. package/dist/tests/fixtures/stashes/load.js +166 -0
  155. package/dist/tests/fixtures/stashes/load.test.js +88 -0
  156. package/dist/tests/fixtures/stashes/ranking-baseline/scripts/mem0-search.js +12 -0
  157. package/dist/tests/frontmatter.test.js +190 -0
  158. package/dist/tests/fts-field-weighting.test.js +254 -0
  159. package/dist/tests/fuzzy-search.test.js +230 -0
  160. package/dist/tests/git-provider-clone.test.js +45 -0
  161. package/dist/tests/github.test.js +161 -0
  162. package/dist/tests/graph-boost-ranking.test.js +305 -0
  163. package/dist/tests/graph-extraction.test.js +282 -0
  164. package/dist/tests/helpers/usage-events.js +8 -0
  165. package/dist/tests/index-pass-llm.test.js +161 -0
  166. package/dist/tests/indexer.test.js +559 -0
  167. package/dist/tests/info-command.test.js +166 -0
  168. package/dist/tests/init.test.js +69 -0
  169. package/dist/tests/install-script.test.js +246 -0
  170. package/dist/tests/integration/agent-real-profile.test.js +94 -0
  171. package/dist/tests/issue-36-repro.test.js +304 -0
  172. package/dist/tests/issues-191-194.test.js +160 -0
  173. package/dist/tests/lesson-lint.test.js +111 -0
  174. package/dist/tests/llm-client.test.js +115 -0
  175. package/dist/tests/llm-feature-gate.test.js +151 -0
  176. package/dist/tests/llm.test.js +139 -0
  177. package/dist/tests/lockfile.test.js +216 -0
  178. package/dist/tests/manifest.test.js +205 -0
  179. package/dist/tests/markdown.test.js +126 -0
  180. package/dist/tests/matchers-unit.test.js +189 -0
  181. package/dist/tests/memory-inference.test.js +299 -0
  182. package/dist/tests/merge-scoring.test.js +136 -0
  183. package/dist/tests/metadata.test.js +313 -0
  184. package/dist/tests/migration-help.test.js +89 -0
  185. package/dist/tests/origin-resolve.test.js +124 -0
  186. package/dist/tests/output-baseline.test.js +217 -0
  187. package/dist/tests/output-shapes-unit.test.js +476 -0
  188. package/dist/tests/parallel-search.test.js +272 -0
  189. package/dist/tests/parameter-metadata.test.js +365 -0
  190. package/dist/tests/paths.test.js +177 -0
  191. package/dist/tests/progressive-disclosure.test.js +280 -0
  192. package/dist/tests/proposals.test.js +279 -0
  193. package/dist/tests/proposed-quality.test.js +271 -0
  194. package/dist/tests/provider-registry.test.js +32 -0
  195. package/dist/tests/ranking-regression.test.js +548 -0
  196. package/dist/tests/reflect-propose.test.js +455 -0
  197. package/dist/tests/registry-build-index.test.js +378 -0
  198. package/dist/tests/registry-cli.test.js +290 -0
  199. package/dist/tests/registry-index-v2.test.js +430 -0
  200. package/dist/tests/registry-install.test.js +728 -0
  201. package/dist/tests/registry-providers/parity.test.js +189 -0
  202. package/dist/tests/registry-providers/skills-sh.test.js +309 -0
  203. package/dist/tests/registry-providers/static-index.test.js +204 -0
  204. package/dist/tests/registry-resolve.test.js +126 -0
  205. package/dist/tests/registry-search.test.js +723 -0
  206. package/dist/tests/remember-frontmatter.test.js +380 -0
  207. package/dist/tests/remember-unit.test.js +123 -0
  208. package/dist/tests/ripgrep-install.test.js +251 -0
  209. package/dist/tests/ripgrep-resolve.test.js +108 -0
  210. package/dist/tests/ripgrep.test.js +163 -0
  211. package/dist/tests/save-command.test.js +94 -0
  212. package/dist/tests/save-trust-qa-fixes.test.js +270 -0
  213. package/dist/tests/scoring-pipeline.test.js +648 -0
  214. package/dist/tests/search-include-proposed-cli.test.js +118 -0
  215. package/dist/tests/self-update.test.js +442 -0
  216. package/dist/tests/semantic-search-e2e.test.js +512 -0
  217. package/dist/tests/semantic-status.test.js +471 -0
  218. package/dist/tests/setup-run.integration.js +877 -0
  219. package/dist/tests/setup-wizard.test.js +198 -0
  220. package/dist/tests/setup.test.js +131 -0
  221. package/dist/tests/source-add.test.js +11 -0
  222. package/dist/tests/source-clone.test.js +254 -0
  223. package/dist/tests/source-manage.test.js +366 -0
  224. package/dist/tests/source-providers/filesystem.test.js +82 -0
  225. package/dist/tests/source-providers/git.test.js +252 -0
  226. package/dist/tests/source-providers/website.test.js +128 -0
  227. package/dist/tests/source-qa-fixes.test.js +268 -0
  228. package/dist/tests/source-registry.test.js +350 -0
  229. package/dist/tests/source-resolve.test.js +100 -0
  230. package/dist/tests/source-source.test.js +221 -0
  231. package/dist/tests/source.test.js +533 -0
  232. package/dist/tests/tar-utils-scan.test.js +73 -0
  233. package/dist/tests/toggle-components.test.js +73 -0
  234. package/dist/tests/usage-telemetry.test.js +265 -0
  235. package/dist/tests/utility-scoring.test.js +558 -0
  236. package/dist/tests/vault-load-error.test.js +78 -0
  237. package/dist/tests/vault-qa-fixes.test.js +194 -0
  238. package/dist/tests/vault.test.js +429 -0
  239. package/dist/tests/vector-search.test.js +608 -0
  240. package/dist/tests/walker.test.js +252 -0
  241. package/dist/tests/wave2-cluster-bc.test.js +228 -0
  242. package/dist/tests/wave2-cluster-d.test.js +180 -0
  243. package/dist/tests/wave2-cluster-e.test.js +179 -0
  244. package/dist/tests/wiki-qa-fixes.test.js +270 -0
  245. package/dist/tests/wiki.test.js +529 -0
  246. package/dist/tests/workflow-cli.test.js +271 -0
  247. package/dist/tests/workflow-markdown.test.js +171 -0
  248. package/dist/tests/workflow-path-escape.test.js +132 -0
  249. package/dist/tests/workflow-qa-fixes.test.js +377 -0
  250. package/dist/tests/workflows/indexer-rejection.test.js +213 -0
  251. package/docs/README.md +8 -0
  252. package/docs/migration/release-notes/0.7.0.md +244 -0
  253. package/package.json +2 -2
  254. package/dist/core/warn.js +0 -27
  255. package/dist/output/shapes.js +0 -212
  256. /package/dist/{commands → src/commands}/completions.js +0 -0
  257. /package/dist/{commands → src/commands}/curate.js +0 -0
  258. /package/dist/{commands → src/commands}/info.js +0 -0
  259. /package/dist/{commands → src/commands}/init.js +0 -0
  260. /package/dist/{commands → src/commands}/install-audit.js +0 -0
  261. /package/dist/{commands → src/commands}/migration-help.js +0 -0
  262. /package/dist/{commands → src/commands}/source-add.js +0 -0
  263. /package/dist/{commands → src/commands}/source-clone.js +0 -0
  264. /package/dist/{commands → src/commands}/source-manage.js +0 -0
  265. /package/dist/{commands → src/commands}/vault.js +0 -0
  266. /package/dist/{core → src/core}/asset-registry.js +0 -0
  267. /package/dist/{core → src/core}/frontmatter.js +0 -0
  268. /package/dist/{core → src/core}/markdown.js +0 -0
  269. /package/dist/{core → src/core}/paths.js +0 -0
  270. /package/dist/{indexer → src/indexer}/manifest.js +0 -0
  271. /package/dist/{indexer → src/indexer}/matchers.js +0 -0
  272. /package/dist/{indexer → src/indexer}/search-fields.js +0 -0
  273. /package/dist/{indexer → src/indexer}/search-source.js +0 -0
  274. /package/dist/{indexer → src/indexer}/semantic-status.js +0 -0
  275. /package/dist/{indexer → src/indexer}/usage-events.js +0 -0
  276. /package/dist/{indexer → src/indexer}/walker.js +0 -0
  277. /package/dist/{integrations → src/integrations}/github.js +0 -0
  278. /package/dist/{llm → src/llm}/embedder.js +0 -0
  279. /package/dist/{llm → src/llm}/embedders/cache.js +0 -0
  280. /package/dist/{llm → src/llm}/embedders/local.js +0 -0
  281. /package/dist/{llm → src/llm}/embedders/remote.js +0 -0
  282. /package/dist/{llm → src/llm}/embedders/types.js +0 -0
  283. /package/dist/{llm → src/llm}/metadata-enhance.js +0 -0
  284. /package/dist/{output → src/output}/cli-hints.js +0 -0
  285. /package/dist/{output → src/output}/context.js +0 -0
  286. /package/dist/{registry → src/registry}/create-provider-registry.js +0 -0
  287. /package/dist/{registry → src/registry}/origin-resolve.js +0 -0
  288. /package/dist/{registry → src/registry}/providers/index.js +0 -0
  289. /package/dist/{registry → src/registry}/providers/skills-sh.js +0 -0
  290. /package/dist/{registry → src/registry}/providers/types.js +0 -0
  291. /package/dist/{registry → src/registry}/types.js +0 -0
  292. /package/dist/{setup → src/setup}/detect.js +0 -0
  293. /package/dist/{setup → src/setup}/ripgrep-install.js +0 -0
  294. /package/dist/{setup → src/setup}/ripgrep-resolve.js +0 -0
  295. /package/dist/{setup → src/setup}/steps.js +0 -0
  296. /package/dist/{sources → src/sources}/include.js +0 -0
  297. /package/dist/{sources → src/sources}/provider-factory.js +0 -0
  298. /package/dist/{sources → src/sources}/provider.js +0 -0
  299. /package/dist/{sources → src/sources}/providers/filesystem.js +0 -0
  300. /package/dist/{sources → src/sources}/providers/index.js +0 -0
  301. /package/dist/{sources → src/sources}/providers/install-types.js +0 -0
  302. /package/dist/{sources → src/sources}/providers/npm.js +0 -0
  303. /package/dist/{sources → src/sources}/providers/provider-utils.js +0 -0
  304. /package/dist/{sources → src/sources}/providers/sync-from-ref.js +0 -0
  305. /package/dist/{sources → src/sources}/providers/tar-utils.js +0 -0
  306. /package/dist/{sources → src/sources}/providers/website.js +0 -0
  307. /package/dist/{sources → src/sources}/resolve.js +0 -0
  308. /package/dist/{sources → src/sources}/types.js +0 -0
  309. /package/dist/{templates → src/templates}/wiki-templates.js +0 -0
  310. /package/dist/{version.js → src/version.js} +0 -0
  311. /package/dist/{workflows → src/workflows}/authoring.js +0 -0
  312. /package/dist/{workflows → src/workflows}/cli.js +0 -0
  313. /package/dist/{workflows → src/workflows}/db.js +0 -0
  314. /package/dist/{workflows → src/workflows}/document-cache.js +0 -0
  315. /package/dist/{workflows → src/workflows}/parser.js +0 -0
  316. /package/dist/{workflows → src/workflows}/renderer.js +0 -0
  317. /package/dist/{workflows → src/workflows}/runs.js +0 -0
  318. /package/dist/{workflows → src/workflows}/schema.js +0 -0
  319. /package/dist/{workflows → src/workflows}/validator.js +0 -0
@@ -0,0 +1,142 @@
1
+ import { afterAll, describe, expect, test } from "bun:test";
2
+ import { spawnSync } from "node:child_process";
3
+ import fs from "node:fs";
4
+ import os from "node:os";
5
+ import path from "node:path";
6
+ // ── Helpers ─────────────────────────────────────────────────────────────────
7
+ const tempDirs = [];
8
+ function makeTempDir() {
9
+ const dir = fs.mkdtempSync(path.join(os.tmpdir(), "akm-completions-"));
10
+ tempDirs.push(dir);
11
+ return dir;
12
+ }
13
+ afterAll(() => {
14
+ for (const dir of tempDirs) {
15
+ fs.rmSync(dir, { recursive: true, force: true });
16
+ }
17
+ });
18
+ const xdgCache = makeTempDir();
19
+ const xdgConfig = makeTempDir();
20
+ const isolatedHome = makeTempDir();
21
+ function runCli(...args) {
22
+ const result = spawnSync("bun", ["./src/cli.ts", ...args], {
23
+ encoding: "utf8",
24
+ timeout: 10_000,
25
+ cwd: path.resolve(import.meta.dir, ".."),
26
+ env: {
27
+ ...process.env,
28
+ AKM_STASH_DIR: undefined,
29
+ HOME: isolatedHome,
30
+ XDG_CACHE_HOME: xdgCache,
31
+ XDG_CONFIG_HOME: xdgConfig,
32
+ XDG_DATA_HOME: undefined,
33
+ },
34
+ });
35
+ return {
36
+ stdout: result.stdout ?? "",
37
+ stderr: result.stderr ?? "",
38
+ status: result.status ?? 1,
39
+ };
40
+ }
41
+ // ── Unit tests (generated script content) ────────────────────────────────────
42
+ describe("completions command", () => {
43
+ const { stdout, status } = runCli("completions");
44
+ const script = stdout;
45
+ test("exits 0 and outputs a bash script", () => {
46
+ expect(status).toBe(0);
47
+ expect(script).toStartWith("#!/bin/bash");
48
+ });
49
+ test("contains complete -F _akm akm", () => {
50
+ expect(script).toContain("complete -F _akm akm");
51
+ });
52
+ test("contains all top-level subcommands", () => {
53
+ const expected = [
54
+ "init",
55
+ "index",
56
+ "add",
57
+ "list",
58
+ "remove",
59
+ "update",
60
+ "upgrade",
61
+ "search",
62
+ "curate",
63
+ "show",
64
+ "workflow",
65
+ "remember",
66
+ "import",
67
+ "clone",
68
+ "save",
69
+ "feedback",
70
+ "registry",
71
+ "config",
72
+ "help",
73
+ "hints",
74
+ "completions",
75
+ ];
76
+ for (const cmd of expected) {
77
+ expect(script).toContain(cmd);
78
+ }
79
+ });
80
+ test("contains nested config subcommands", () => {
81
+ expect(script).toContain('"akm config"');
82
+ for (const sub of ["path", "list", "get", "set", "unset"]) {
83
+ expect(script).toContain(sub);
84
+ }
85
+ });
86
+ test("contains nested registry subcommands", () => {
87
+ expect(script).toContain('"akm registry"');
88
+ for (const sub of ["list", "add", "remove", "search", "build-index"]) {
89
+ expect(script).toContain(sub);
90
+ }
91
+ });
92
+ test("contains flag value completions for --format", () => {
93
+ expect(script).toContain("--format)");
94
+ expect(script).toContain("json text yaml jsonl");
95
+ });
96
+ test("contains flag value completions for --detail", () => {
97
+ expect(script).toContain("--detail)");
98
+ expect(script).toContain("brief normal full summary");
99
+ });
100
+ test("contains flag value completions for --type", () => {
101
+ expect(script).toContain("--type)");
102
+ expect(script).toContain("skill command agent knowledge workflow script memory vault wiki lesson any");
103
+ });
104
+ test("contains flag value completions for --source", () => {
105
+ expect(script).toContain("--source)");
106
+ expect(script).toContain("stash registry both");
107
+ });
108
+ });
109
+ // ── Integration: --install ───────────────────────────────────────────────────
110
+ describe("completions --install", () => {
111
+ test("writes completion file to XDG_DATA_HOME path", () => {
112
+ const xdgData = makeTempDir();
113
+ const result = spawnSync("bun", ["./src/cli.ts", "completions", "--install"], {
114
+ encoding: "utf8",
115
+ timeout: 10_000,
116
+ cwd: path.resolve(import.meta.dir, ".."),
117
+ env: {
118
+ ...process.env,
119
+ AKM_STASH_DIR: undefined,
120
+ HOME: isolatedHome,
121
+ XDG_CACHE_HOME: xdgCache,
122
+ XDG_CONFIG_HOME: xdgConfig,
123
+ XDG_DATA_HOME: xdgData,
124
+ },
125
+ });
126
+ expect(result.status).toBe(0);
127
+ const expectedPath = path.join(xdgData, "bash-completion", "completions", "akm");
128
+ expect(result.stderr).toContain(expectedPath);
129
+ expect(fs.existsSync(expectedPath)).toBe(true);
130
+ const content = fs.readFileSync(expectedPath, "utf8");
131
+ expect(content).toStartWith("#!/bin/bash");
132
+ expect(content).toContain("complete -F _akm akm");
133
+ });
134
+ });
135
+ // ── Unsupported shell ────────────────────────────────────────────────────────
136
+ describe("completions unsupported shell", () => {
137
+ test("rejects unsupported shell type", () => {
138
+ const { stderr, status } = runCli("completions", "--shell", "zsh");
139
+ expect(status).not.toBe(0);
140
+ expect(stderr).toContain("Unsupported shell");
141
+ });
142
+ });
@@ -0,0 +1,193 @@
1
+ import { describe, expect, test } from "bun:test";
2
+ import { getConfigValue, listConfig, parseConfigValue, setConfigValue, unsetConfigValue, } from "../src/commands/config-cli";
3
+ describe("config CLI helpers", () => {
4
+ test("listConfig omits unconfigured embedding and llm", () => {
5
+ const config = listConfig({ semanticSearchMode: "auto" });
6
+ expect(config.embedding).toBeUndefined();
7
+ expect(config.llm).toBeUndefined();
8
+ expect(config.output).toEqual({ format: "json", detail: "brief" });
9
+ });
10
+ test("parseConfigValue supports output config keys", () => {
11
+ expect(parseConfigValue("output.format", "yaml")).toEqual({ output: { format: "yaml" } });
12
+ expect(parseConfigValue("output.detail", "full")).toEqual({ output: { detail: "full" } });
13
+ });
14
+ test("parseConfigValue supports install audit config keys", () => {
15
+ expect(parseConfigValue("security.installAudit.enabled", "false")).toEqual({
16
+ security: { installAudit: { enabled: false } },
17
+ });
18
+ expect(parseConfigValue("security.installAudit.registryAllowlist", '["npm","github.com"]')).toEqual({
19
+ security: { installAudit: { registryAllowlist: ["npm", "github.com"] } },
20
+ });
21
+ expect(parseConfigValue("security.installAudit.allowedFindings", '[{"id":"prompt-reveal-hidden-secrets","ref":"github:owner/repo","path":"skills/review/SKILL.md","reason":"false positive"}]')).toEqual({
22
+ security: {
23
+ installAudit: {
24
+ allowedFindings: [
25
+ {
26
+ id: "prompt-reveal-hidden-secrets",
27
+ ref: "github:owner/repo",
28
+ path: "skills/review/SKILL.md",
29
+ reason: "false positive",
30
+ },
31
+ ],
32
+ },
33
+ },
34
+ });
35
+ });
36
+ test("parseConfigValue still accepts registryWhitelist as a legacy alias", () => {
37
+ expect(parseConfigValue("security.installAudit.registryWhitelist", '["npm"]')).toEqual({
38
+ security: { installAudit: { registryAllowlist: ["npm"] } },
39
+ });
40
+ });
41
+ test("parseConfigValue supports embedding JSON with dimensions", () => {
42
+ expect(parseConfigValue("embedding", '{"endpoint":"https://api.openai.com/v1/embeddings","model":"text-embedding-3-small","dimension":384}')).toEqual({
43
+ embedding: {
44
+ endpoint: "https://api.openai.com/v1/embeddings",
45
+ model: "text-embedding-3-small",
46
+ dimension: 384,
47
+ },
48
+ });
49
+ });
50
+ test("parseConfigValue supports llm JSON with sampling fields", () => {
51
+ expect(parseConfigValue("llm", '{"endpoint":"https://api.openai.com/v1/chat/completions","model":"gpt-4o-mini","temperature":0.6,"maxTokens":300}')).toEqual({
52
+ llm: {
53
+ endpoint: "https://api.openai.com/v1/chat/completions",
54
+ model: "gpt-4o-mini",
55
+ temperature: 0.6,
56
+ maxTokens: 300,
57
+ },
58
+ });
59
+ });
60
+ test("setConfigValue sets embedding via JSON", () => {
61
+ const base = { semanticSearchMode: "auto" };
62
+ const updated = setConfigValue(base, "embedding", '{"endpoint":"http://localhost:11434/v1/embeddings","model":"nomic-embed-text"}');
63
+ expect(updated.embedding).toEqual({
64
+ endpoint: "http://localhost:11434/v1/embeddings",
65
+ model: "nomic-embed-text",
66
+ });
67
+ });
68
+ test("setConfigValue sets llm via JSON", () => {
69
+ const base = { semanticSearchMode: "auto" };
70
+ const updated = setConfigValue(base, "llm", '{"endpoint":"http://localhost:11434/v1/chat/completions","model":"llama3.2","temperature":0.3}');
71
+ expect(updated.llm).toEqual({
72
+ endpoint: "http://localhost:11434/v1/chat/completions",
73
+ model: "llama3.2",
74
+ temperature: 0.3,
75
+ });
76
+ });
77
+ test("getConfigValue returns null for unconfigured embedding/llm", () => {
78
+ const base = { semanticSearchMode: "auto" };
79
+ expect(getConfigValue(base, "embedding")).toBeNull();
80
+ expect(getConfigValue(base, "llm")).toBeNull();
81
+ });
82
+ test("getConfigValue returns configured embedding/llm objects", () => {
83
+ const base = {
84
+ semanticSearchMode: "auto",
85
+ embedding: {
86
+ endpoint: "https://api.openai.com/v1/embeddings",
87
+ model: "text-embedding-3-small",
88
+ },
89
+ llm: {
90
+ endpoint: "http://localhost:11434/v1/chat/completions",
91
+ model: "llama3.2",
92
+ temperature: 0.3,
93
+ },
94
+ };
95
+ expect(getConfigValue(base, "embedding")).toEqual(base.embedding);
96
+ expect(getConfigValue(base, "llm")).toEqual(base.llm);
97
+ });
98
+ test("set/get/unset support install audit config keys", () => {
99
+ const base = { semanticSearchMode: "auto" };
100
+ const configured = setConfigValue(base, "security.installAudit.enabled", "true");
101
+ const withWhitelist = setConfigValue(configured, "security.installAudit.registryAllowlist", '["npm","github.com"]');
102
+ const withAllowedFindings = setConfigValue(withWhitelist, "security.installAudit.allowedFindings", '[{"id":"bundled-package-directory","path":"venv"}]');
103
+ expect(getConfigValue(withAllowedFindings, "security.installAudit.enabled")).toBe(true);
104
+ expect(getConfigValue(withAllowedFindings, "security.installAudit.registryAllowlist")).toEqual([
105
+ "npm",
106
+ "github.com",
107
+ ]);
108
+ expect(getConfigValue(withAllowedFindings, "security.installAudit.allowedFindings")).toEqual([
109
+ { id: "bundled-package-directory", path: "venv" },
110
+ ]);
111
+ expect(unsetConfigValue(withAllowedFindings, "security.installAudit.registryAllowlist").security).toEqual({
112
+ installAudit: { enabled: true, allowedFindings: [{ id: "bundled-package-directory", path: "venv" }] },
113
+ });
114
+ expect(unsetConfigValue(withAllowedFindings, "security.installAudit.allowedFindings").security).toEqual({
115
+ installAudit: { enabled: true, registryAllowlist: ["npm", "github.com"] },
116
+ });
117
+ });
118
+ test("unsetConfigValue clears embedding and llm", () => {
119
+ const base = {
120
+ semanticSearchMode: "auto",
121
+ embedding: {
122
+ endpoint: "https://api.openai.com/v1/embeddings",
123
+ model: "text-embedding-3-small",
124
+ },
125
+ llm: {
126
+ endpoint: "http://localhost:11434/v1/chat/completions",
127
+ model: "llama3.2",
128
+ },
129
+ };
130
+ const noEmbed = unsetConfigValue(base, "embedding");
131
+ expect(noEmbed.embedding).toBeUndefined();
132
+ const noLlm = unsetConfigValue(base, "llm");
133
+ expect(noLlm.llm).toBeUndefined();
134
+ });
135
+ test("setConfigValue merges output format and detail", () => {
136
+ const base = { semanticSearchMode: "auto" };
137
+ const withFormat = setConfigValue(base, "output.format", "text");
138
+ const withDetail = setConfigValue(withFormat, "output.detail", "full");
139
+ expect(withDetail.output).toEqual({ format: "text", detail: "full" });
140
+ });
141
+ test("getConfigValue reads output keys", () => {
142
+ const base = {
143
+ semanticSearchMode: "auto",
144
+ output: { format: "yaml", detail: "normal" },
145
+ };
146
+ expect(getConfigValue(base, "output.format")).toBe("yaml");
147
+ expect(getConfigValue(base, "output.detail")).toBe("normal");
148
+ });
149
+ test("unsetConfigValue clears individual output keys", () => {
150
+ const base = {
151
+ semanticSearchMode: "auto",
152
+ output: { format: "yaml", detail: "normal" },
153
+ };
154
+ expect(unsetConfigValue(base, "output.format").output).toEqual({ detail: "normal" });
155
+ expect(unsetConfigValue(base, "output.detail").output).toEqual({ format: "yaml" });
156
+ });
157
+ test("setConfigValue rejects unknown keys", () => {
158
+ const base = { semanticSearchMode: "auto" };
159
+ expect(() => setConfigValue(base, "embedding.provider", "ollama")).toThrow("Unknown config key");
160
+ expect(() => setConfigValue(base, "llm.temperature", "0.5")).toThrow("Unknown config key");
161
+ });
162
+ test("parseConfigValue rejects non-integer embedding dimension in JSON", () => {
163
+ expect(() => parseConfigValue("embedding", '{"endpoint":"https://api.openai.com/v1/embeddings","model":"text-embedding-3-small","dimension":384.5}')).toThrow("expected a positive integer");
164
+ });
165
+ test("parseConfigValue rejects invalid output values", () => {
166
+ expect(() => parseConfigValue("output.format", "xml")).toThrow("expected one of json|yaml|text");
167
+ expect(() => parseConfigValue("output.detail", "max")).toThrow("expected one of brief|normal|full");
168
+ });
169
+ test("parseConfigValue rejects invalid install audit values", () => {
170
+ expect(() => parseConfigValue("security.installAudit.enabled", "yes")).toThrow("expected true or false");
171
+ expect(() => parseConfigValue("security.installAudit.registryAllowlist", '{"npm":true}')).toThrow("expected a JSON array of strings");
172
+ expect(() => parseConfigValue("security.installAudit.allowedFindings", '{"id":"x"}')).toThrow("expected a JSON array");
173
+ });
174
+ test("parseConfigValue coerces 'true' to 'auto' for semanticSearchMode", () => {
175
+ const result = parseConfigValue("semanticSearchMode", "true");
176
+ expect(result).toEqual({ semanticSearchMode: "auto" });
177
+ });
178
+ test("parseConfigValue coerces 'false' to 'off' for semanticSearchMode", () => {
179
+ const result = parseConfigValue("semanticSearchMode", "false");
180
+ expect(result).toEqual({ semanticSearchMode: "off" });
181
+ });
182
+ test("parseConfigValue accepts 'auto' for semanticSearchMode", () => {
183
+ const result = parseConfigValue("semanticSearchMode", "auto");
184
+ expect(result).toEqual({ semanticSearchMode: "auto" });
185
+ });
186
+ test("parseConfigValue accepts 'off' for semanticSearchMode", () => {
187
+ const result = parseConfigValue("semanticSearchMode", "off");
188
+ expect(result).toEqual({ semanticSearchMode: "off" });
189
+ });
190
+ test("parseConfigValue rejects invalid semanticSearchMode", () => {
191
+ expect(() => parseConfigValue("semanticSearchMode", "yes")).toThrow("Invalid value for semanticSearchMode");
192
+ });
193
+ });
@@ -0,0 +1,139 @@
1
+ /**
2
+ * Config-load contract for `llm.features.*` (v1 spec §14, #227).
3
+ *
4
+ * Locks:
5
+ * - All locked keys with runtime consumers parse through into the runtime
6
+ * `LlmFeatureFlags`.
7
+ * - Defaults are absent (interpreted as `false` at every call site —
8
+ * `isLlmFeatureEnabled` is the seam, see tests/llm-feature-gate.test.ts).
9
+ * - Non-boolean values are warn-and-skipped (no throw, the rest of the
10
+ * features block continues to parse).
11
+ * - Unknown keys are warn-and-skipped (no throw, no schema mutation).
12
+ */
13
+ import { afterEach, beforeEach, describe, expect, spyOn, test } from "bun:test";
14
+ import fs from "node:fs";
15
+ import os from "node:os";
16
+ import path from "node:path";
17
+ import { getConfigPath, loadConfig, resetConfigCache } from "../src/core/config";
18
+ const originalXdgConfigHome = process.env.XDG_CONFIG_HOME;
19
+ const originalHome = process.env.HOME;
20
+ let testConfigHome = "";
21
+ function makeTmpDir() {
22
+ return fs.mkdtempSync(path.join(os.tmpdir(), "akm-cfg-llm-features-"));
23
+ }
24
+ function writeConfig(content) {
25
+ const cfgPath = getConfigPath();
26
+ fs.mkdirSync(path.dirname(cfgPath), { recursive: true });
27
+ fs.writeFileSync(cfgPath, JSON.stringify(content));
28
+ }
29
+ beforeEach(() => {
30
+ testConfigHome = makeTmpDir();
31
+ process.env.XDG_CONFIG_HOME = testConfigHome;
32
+ resetConfigCache();
33
+ });
34
+ afterEach(() => {
35
+ if (originalXdgConfigHome === undefined)
36
+ delete process.env.XDG_CONFIG_HOME;
37
+ else
38
+ process.env.XDG_CONFIG_HOME = originalXdgConfigHome;
39
+ if (originalHome === undefined)
40
+ delete process.env.HOME;
41
+ else
42
+ process.env.HOME = originalHome;
43
+ if (testConfigHome) {
44
+ fs.rmSync(testConfigHome, { recursive: true, force: true });
45
+ testConfigHome = "";
46
+ }
47
+ resetConfigCache();
48
+ });
49
+ describe("loadConfig — llm.features (v1 spec §14)", () => {
50
+ test("parses all locked feature keys with runtime consumers", () => {
51
+ writeConfig({
52
+ llm: {
53
+ endpoint: "http://localhost:11434/v1/chat/completions",
54
+ model: "llama3.2",
55
+ features: {
56
+ curate_rerank: true,
57
+ feedback_distillation: true,
58
+ memory_inference: true,
59
+ graph_extraction: true,
60
+ },
61
+ },
62
+ });
63
+ const cfg = loadConfig();
64
+ expect(cfg.llm?.features).toEqual({
65
+ curate_rerank: true,
66
+ feedback_distillation: true,
67
+ memory_inference: true,
68
+ graph_extraction: true,
69
+ });
70
+ });
71
+ test("absent keys remain absent (default-false at call sites)", () => {
72
+ writeConfig({
73
+ llm: {
74
+ endpoint: "http://localhost:11434/v1/chat/completions",
75
+ model: "llama3.2",
76
+ features: { curate_rerank: true },
77
+ },
78
+ });
79
+ const cfg = loadConfig();
80
+ expect(cfg.llm?.features?.curate_rerank).toBe(true);
81
+ expect(cfg.llm?.features?.feedback_distillation).toBeUndefined();
82
+ expect(cfg.llm?.features?.memory_inference).toBeUndefined();
83
+ expect(cfg.llm?.features?.graph_extraction).toBeUndefined();
84
+ });
85
+ test("non-boolean values warn and are skipped without breaking siblings", () => {
86
+ const warnSpy = spyOn(console, "warn").mockImplementation(() => { });
87
+ try {
88
+ writeConfig({
89
+ llm: {
90
+ endpoint: "http://localhost:11434/v1/chat/completions",
91
+ model: "llama3.2",
92
+ features: {
93
+ curate_rerank: "yes",
94
+ graph_extraction: 1,
95
+ feedback_distillation: true,
96
+ },
97
+ },
98
+ });
99
+ const cfg = loadConfig();
100
+ expect(cfg.llm?.features?.curate_rerank).toBeUndefined();
101
+ expect(cfg.llm?.features?.graph_extraction).toBeUndefined();
102
+ // The valid sibling continues to parse.
103
+ expect(cfg.llm?.features?.feedback_distillation).toBe(true);
104
+ const messages = warnSpy.mock.calls.map((c) => String(c[0]));
105
+ expect(messages.some((m) => m.includes("curate_rerank") && m.includes("expected boolean"))).toBe(true);
106
+ expect(messages.some((m) => m.includes("graph_extraction") && m.includes("expected boolean"))).toBe(true);
107
+ }
108
+ finally {
109
+ warnSpy.mockRestore();
110
+ }
111
+ });
112
+ test("unknown keys warn and are dropped without affecting locked keys", () => {
113
+ const warnSpy = spyOn(console, "warn").mockImplementation(() => { });
114
+ try {
115
+ writeConfig({
116
+ llm: {
117
+ endpoint: "http://localhost:11434/v1/chat/completions",
118
+ model: "llama3.2",
119
+ features: {
120
+ curate_rerank: true,
121
+ // Unknown keys must be warn-and-ignore (v1 spec §14.3).
122
+ future_feature: true,
123
+ another_one: false,
124
+ },
125
+ },
126
+ });
127
+ const cfg = loadConfig();
128
+ expect(cfg.llm?.features?.curate_rerank).toBe(true);
129
+ expect(cfg.llm?.features?.future_feature).toBeUndefined();
130
+ expect(cfg.llm?.features?.another_one).toBeUndefined();
131
+ const messages = warnSpy.mock.calls.map((c) => String(c[0]));
132
+ expect(messages.some((m) => m.includes("future_feature") && m.includes("Ignoring"))).toBe(true);
133
+ expect(messages.some((m) => m.includes("another_one") && m.includes("Ignoring"))).toBe(true);
134
+ }
135
+ finally {
136
+ warnSpy.mockRestore();
137
+ }
138
+ });
139
+ });