akm-cli 0.6.1 → 0.7.0

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 (333) hide show
  1. package/CHANGELOG.md +66 -0
  2. package/dist/{cli.js → src/cli.js} +712 -34
  3. package/dist/{commands → src/commands}/config-cli.js +47 -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 +191 -0
  7. package/dist/{commands → src/commands}/installed-stashes.js +1 -1
  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 +71 -7
  12. package/dist/{commands → src/commands}/remember.js +12 -0
  13. package/dist/{commands → src/commands}/search.js +104 -4
  14. package/dist/{commands → src/commands}/self-update.js +4 -3
  15. package/dist/{commands → src/commands}/show.js +73 -0
  16. package/dist/{commands → src/commands}/source-add.js +5 -1
  17. package/dist/{commands → src/commands}/source-manage.js +7 -1
  18. package/dist/{core → src/core}/asset-ref.js +5 -5
  19. package/dist/{core → src/core}/asset-spec.js +12 -0
  20. package/dist/{core → src/core}/common.js +1 -1
  21. package/dist/{core → src/core}/config.js +203 -121
  22. package/dist/{core → src/core}/errors.js +4 -0
  23. package/dist/src/core/events.js +239 -0
  24. package/dist/src/core/lesson-lint.js +86 -0
  25. package/dist/src/core/proposals.js +406 -0
  26. package/dist/src/core/warn.js +72 -0
  27. package/dist/{core → src/core}/write-source.js +80 -5
  28. package/dist/{indexer → src/indexer}/db-search.js +114 -24
  29. package/dist/{indexer → src/indexer}/db.js +76 -23
  30. package/dist/{indexer → src/indexer}/file-context.js +0 -3
  31. package/dist/src/indexer/graph-boost.js +179 -0
  32. package/dist/src/indexer/graph-extraction.js +212 -0
  33. package/dist/{indexer → src/indexer}/indexer.js +88 -7
  34. package/dist/{indexer → src/indexer}/matchers.js +1 -1
  35. package/dist/src/indexer/memory-inference.js +263 -0
  36. package/dist/{indexer → src/indexer}/metadata.js +111 -3
  37. package/dist/{indexer → src/indexer}/search-source.js +4 -2
  38. package/dist/src/integrations/agent/config.js +292 -0
  39. package/dist/src/integrations/agent/detect.js +94 -0
  40. package/dist/src/integrations/agent/index.js +17 -0
  41. package/dist/src/integrations/agent/profiles.js +65 -0
  42. package/dist/src/integrations/agent/prompts.js +167 -0
  43. package/dist/src/integrations/agent/spawn.js +272 -0
  44. package/dist/{integrations → src/integrations}/github.js +9 -3
  45. package/dist/{integrations → src/integrations}/lockfile.js +0 -26
  46. package/dist/{llm → src/llm}/client.js +33 -2
  47. package/dist/{llm → src/llm}/embedders/remote.js +37 -3
  48. package/dist/src/llm/feature-gate.js +108 -0
  49. package/dist/src/llm/graph-extract.js +107 -0
  50. package/dist/src/llm/index-passes.js +35 -0
  51. package/dist/src/llm/memory-infer.js +86 -0
  52. package/dist/{output → src/output}/cli-hints.js +15 -2
  53. package/dist/{output → src/output}/renderers.js +63 -2
  54. package/dist/src/output/shapes.js +523 -0
  55. package/dist/src/output/text.js +1116 -0
  56. package/dist/{registry → src/registry}/build-index.js +19 -8
  57. package/dist/{registry → src/registry}/factory.js +0 -8
  58. package/dist/{registry → src/registry}/providers/static-index.js +6 -3
  59. package/dist/{registry → src/registry}/resolve.js +68 -2
  60. package/dist/{setup → src/setup}/setup.js +52 -5
  61. package/dist/{sources → src/sources}/providers/git.js +7 -15
  62. package/dist/{wiki → src/wiki}/wiki.js +54 -6
  63. package/dist/{workflows → src/workflows}/runs.js +37 -3
  64. package/dist/tests/add-website-source.test.js +119 -0
  65. package/dist/tests/agent/agent-config-loader.test.js +70 -0
  66. package/dist/tests/agent/agent-config.test.js +221 -0
  67. package/dist/tests/agent/agent-detect.test.js +100 -0
  68. package/dist/tests/agent/agent-spawn.test.js +234 -0
  69. package/dist/tests/agent-output.test.js +186 -0
  70. package/dist/tests/architecture/agent-no-llm-sdk-guard.test.js +103 -0
  71. package/dist/tests/architecture/agent-spawn-seam.test.js +193 -0
  72. package/dist/tests/architecture/llm-stateless-seam.test.js +112 -0
  73. package/dist/tests/asset-ref.test.js +192 -0
  74. package/dist/tests/asset-registry.test.js +103 -0
  75. package/dist/tests/asset-spec.test.js +241 -0
  76. package/dist/tests/bench/attribution.test.js +996 -0
  77. package/dist/tests/bench/cleanup-sigint.test.js +83 -0
  78. package/dist/tests/bench/cleanup.js +234 -0
  79. package/dist/tests/bench/cleanup.test.js +166 -0
  80. package/dist/tests/bench/cli.js +1018 -0
  81. package/dist/tests/bench/cli.test.js +445 -0
  82. package/dist/tests/bench/compare.test.js +556 -0
  83. package/dist/tests/bench/corpus.js +317 -0
  84. package/dist/tests/bench/corpus.test.js +258 -0
  85. package/dist/tests/bench/doctor.js +525 -0
  86. package/dist/tests/bench/driver.js +401 -0
  87. package/dist/tests/bench/driver.test.js +584 -0
  88. package/dist/tests/bench/environment.js +233 -0
  89. package/dist/tests/bench/environment.test.js +199 -0
  90. package/dist/tests/bench/evolve-metrics.js +179 -0
  91. package/dist/tests/bench/evolve-metrics.test.js +187 -0
  92. package/dist/tests/bench/evolve.js +647 -0
  93. package/dist/tests/bench/evolve.test.js +624 -0
  94. package/dist/tests/bench/failure-modes.test.js +349 -0
  95. package/dist/tests/bench/feedback-integrity.test.js +457 -0
  96. package/dist/tests/bench/leakage.test.js +228 -0
  97. package/dist/tests/bench/learning-curve.test.js +134 -0
  98. package/dist/tests/bench/metrics.js +2395 -0
  99. package/dist/tests/bench/metrics.test.js +1150 -0
  100. package/dist/tests/bench/no-os-tmpdir-invariant.test.js +43 -0
  101. package/dist/tests/bench/opencode-config.js +194 -0
  102. package/dist/tests/bench/opencode-config.test.js +370 -0
  103. package/dist/tests/bench/report.js +1885 -0
  104. package/dist/tests/bench/report.test.js +1038 -0
  105. package/dist/tests/bench/run-config.js +355 -0
  106. package/dist/tests/bench/run-config.test.js +298 -0
  107. package/dist/tests/bench/run-curate-test.js +32 -0
  108. package/dist/tests/bench/run-failing-tasks.js +56 -0
  109. package/dist/tests/bench/run-full-bench.js +51 -0
  110. package/dist/tests/bench/run-items36-targeted.js +69 -0
  111. package/dist/tests/bench/run-nano-quick.js +42 -0
  112. package/dist/tests/bench/run-waveg-targeted.js +62 -0
  113. package/dist/tests/bench/runner.js +699 -0
  114. package/dist/tests/bench/runner.test.js +958 -0
  115. package/dist/tests/bench/search-bridge.test.js +331 -0
  116. package/dist/tests/bench/tmp.js +131 -0
  117. package/dist/tests/bench/trajectory.js +116 -0
  118. package/dist/tests/bench/trajectory.test.js +127 -0
  119. package/dist/tests/bench/verifier.js +114 -0
  120. package/dist/tests/bench/verifier.test.js +118 -0
  121. package/dist/tests/bench/workflow-evaluator.js +557 -0
  122. package/dist/tests/bench/workflow-evaluator.test.js +421 -0
  123. package/dist/tests/bench/workflow-spec.js +345 -0
  124. package/dist/tests/bench/workflow-spec.test.js +363 -0
  125. package/dist/tests/bench/workflow-trace.js +472 -0
  126. package/dist/tests/bench/workflow-trace.test.js +254 -0
  127. package/dist/tests/benchmark-search-quality.js +536 -0
  128. package/dist/tests/benchmark-suite.js +1441 -0
  129. package/dist/tests/capture-cli.test.js +112 -0
  130. package/dist/tests/cli-errors.test.js +204 -0
  131. package/dist/tests/commands/events.test.js +370 -0
  132. package/dist/tests/commands/history.test.js +418 -0
  133. package/dist/tests/commands/import.test.js +103 -0
  134. package/dist/tests/commands/proposal-cli.test.js +209 -0
  135. package/dist/tests/commands/reflect-propose-cli.test.js +333 -0
  136. package/dist/tests/commands/remember.test.js +97 -0
  137. package/dist/tests/commands/scope-flags.test.js +300 -0
  138. package/dist/tests/commands/search.test.js +537 -0
  139. package/dist/tests/commands/show-indexer-parity.test.js +117 -0
  140. package/dist/tests/commands/show.test.js +294 -0
  141. package/dist/tests/common.test.js +266 -0
  142. package/dist/tests/completions.test.js +142 -0
  143. package/dist/tests/config-cli.test.js +193 -0
  144. package/dist/tests/config-llm-features.test.js +139 -0
  145. package/dist/tests/config.test.js +569 -0
  146. package/dist/tests/contracts/migration-baseline.test.js +43 -0
  147. package/dist/tests/contracts/reflect-propose-envelope.test.js +139 -0
  148. package/dist/tests/contracts/spec-helpers.js +46 -0
  149. package/dist/tests/contracts/v1-spec-section-11-proposal-queue.test.js +228 -0
  150. package/dist/tests/contracts/v1-spec-section-12-agent-config.test.js +56 -0
  151. package/dist/tests/contracts/v1-spec-section-13-lesson-type.test.js +34 -0
  152. package/dist/tests/contracts/v1-spec-section-14-llm-features.test.js +94 -0
  153. package/dist/tests/contracts/v1-spec-section-4-1-asset-types.test.js +39 -0
  154. package/dist/tests/contracts/v1-spec-section-4-2-quality-rules.test.js +44 -0
  155. package/dist/tests/contracts/v1-spec-section-5-configuration.test.js +47 -0
  156. package/dist/tests/contracts/v1-spec-section-6-orchestration.test.js +40 -0
  157. package/dist/tests/contracts/v1-spec-section-7-module-layout.test.js +58 -0
  158. package/dist/tests/contracts/v1-spec-section-8-extension-points.test.js +34 -0
  159. package/dist/tests/contracts/v1-spec-section-9-4-cli-surface.test.js +75 -0
  160. package/dist/tests/contracts/v1-spec-section-9-7-llm-agent-boundary.test.js +36 -0
  161. package/dist/tests/core/write-source.test.js +366 -0
  162. package/dist/tests/curate-command.test.js +87 -0
  163. package/dist/tests/db-scoring.test.js +201 -0
  164. package/dist/tests/db.test.js +654 -0
  165. package/dist/tests/distill-cli-flag.test.js +208 -0
  166. package/dist/tests/distill.test.js +515 -0
  167. package/dist/tests/docker-install.test.js +120 -0
  168. package/dist/tests/e2e.test.js +1419 -0
  169. package/dist/tests/embedder.test.js +340 -0
  170. package/dist/tests/embedding-model-config.test.js +379 -0
  171. package/dist/tests/feedback-command.test.js +172 -0
  172. package/dist/tests/file-context.test.js +552 -0
  173. package/dist/tests/fixtures/scripts/git/summarize-diff.js +9 -0
  174. package/dist/tests/fixtures/scripts/lint/eslint-check.js +7 -0
  175. package/dist/tests/fixtures/stashes/load.js +166 -0
  176. package/dist/tests/fixtures/stashes/load.test.js +97 -0
  177. package/dist/tests/fixtures/stashes/ranking-baseline/scripts/mem0-search.js +12 -0
  178. package/dist/tests/frontmatter.test.js +190 -0
  179. package/dist/tests/fts-field-weighting.test.js +254 -0
  180. package/dist/tests/fuzzy-search.test.js +230 -0
  181. package/dist/tests/git-provider-clone.test.js +45 -0
  182. package/dist/tests/github.test.js +161 -0
  183. package/dist/tests/graph-boost-ranking.test.js +305 -0
  184. package/dist/tests/graph-extraction.test.js +282 -0
  185. package/dist/tests/helpers/usage-events.js +8 -0
  186. package/dist/tests/index-pass-llm.test.js +161 -0
  187. package/dist/tests/indexer.test.js +570 -0
  188. package/dist/tests/info-command.test.js +166 -0
  189. package/dist/tests/init.test.js +69 -0
  190. package/dist/tests/install-script.test.js +246 -0
  191. package/dist/tests/integration/agent-real-profile.test.js +94 -0
  192. package/dist/tests/issue-36-repro.test.js +304 -0
  193. package/dist/tests/issues-191-194.test.js +160 -0
  194. package/dist/tests/lesson-lint.test.js +111 -0
  195. package/dist/tests/llm-client.test.js +115 -0
  196. package/dist/tests/llm-feature-gate.test.js +151 -0
  197. package/dist/tests/llm.test.js +139 -0
  198. package/dist/tests/lockfile.test.js +216 -0
  199. package/dist/tests/manifest.test.js +205 -0
  200. package/dist/tests/markdown.test.js +126 -0
  201. package/dist/tests/matchers-unit.test.js +189 -0
  202. package/dist/tests/memory-inference.test.js +299 -0
  203. package/dist/tests/merge-scoring.test.js +136 -0
  204. package/dist/tests/metadata.test.js +313 -0
  205. package/dist/tests/migration-help.test.js +89 -0
  206. package/dist/tests/origin-resolve.test.js +124 -0
  207. package/dist/tests/output-baseline.test.js +218 -0
  208. package/dist/tests/output-shapes-unit.test.js +478 -0
  209. package/dist/tests/parallel-search.test.js +272 -0
  210. package/dist/tests/parameter-metadata.test.js +365 -0
  211. package/dist/tests/paths.test.js +177 -0
  212. package/dist/tests/progressive-disclosure.test.js +280 -0
  213. package/dist/tests/proposals.test.js +279 -0
  214. package/dist/tests/proposed-quality.test.js +271 -0
  215. package/dist/tests/provider-registry.test.js +32 -0
  216. package/dist/tests/ranking-regression.test.js +548 -0
  217. package/dist/tests/reflect-propose.test.js +455 -0
  218. package/dist/tests/registry-build-index.test.js +394 -0
  219. package/dist/tests/registry-cli.test.js +290 -0
  220. package/dist/tests/registry-index-v2.test.js +430 -0
  221. package/dist/tests/registry-install.test.js +728 -0
  222. package/dist/tests/registry-providers/parity.test.js +189 -0
  223. package/dist/tests/registry-providers/skills-sh.test.js +309 -0
  224. package/dist/tests/registry-providers/static-index.test.js +238 -0
  225. package/dist/tests/registry-resolve.test.js +126 -0
  226. package/dist/tests/registry-search.test.js +923 -0
  227. package/dist/tests/remember-frontmatter.test.js +378 -0
  228. package/dist/tests/remember-unit.test.js +123 -0
  229. package/dist/tests/ripgrep-install.test.js +251 -0
  230. package/dist/tests/ripgrep-resolve.test.js +108 -0
  231. package/dist/tests/ripgrep.test.js +163 -0
  232. package/dist/tests/save-command.test.js +94 -0
  233. package/dist/tests/save-trust-qa-fixes.test.js +270 -0
  234. package/dist/tests/scoring-pipeline.test.js +648 -0
  235. package/dist/tests/search-include-proposed-cli.test.js +118 -0
  236. package/dist/tests/self-update.test.js +442 -0
  237. package/dist/tests/semantic-search-e2e.test.js +512 -0
  238. package/dist/tests/semantic-status.test.js +471 -0
  239. package/dist/tests/setup-run.integration.js +877 -0
  240. package/dist/tests/setup-wizard.test.js +198 -0
  241. package/dist/tests/setup.test.js +131 -0
  242. package/dist/tests/source-add.test.js +11 -0
  243. package/dist/tests/source-clone.test.js +254 -0
  244. package/dist/tests/source-manage.test.js +366 -0
  245. package/dist/tests/source-providers/filesystem.test.js +82 -0
  246. package/dist/tests/source-providers/git.test.js +252 -0
  247. package/dist/tests/source-providers/website.test.js +128 -0
  248. package/dist/tests/source-qa-fixes.test.js +286 -0
  249. package/dist/tests/source-registry.test.js +350 -0
  250. package/dist/tests/source-resolve.test.js +100 -0
  251. package/dist/tests/source-source.test.js +281 -0
  252. package/dist/tests/source.test.js +533 -0
  253. package/dist/tests/tar-utils-scan.test.js +73 -0
  254. package/dist/tests/toggle-components.test.js +73 -0
  255. package/dist/tests/usage-telemetry.test.js +265 -0
  256. package/dist/tests/utility-scoring.test.js +558 -0
  257. package/dist/tests/vault-load-error.test.js +78 -0
  258. package/dist/tests/vault-qa-fixes.test.js +194 -0
  259. package/dist/tests/vault.test.js +429 -0
  260. package/dist/tests/vector-search.test.js +608 -0
  261. package/dist/tests/walker.test.js +252 -0
  262. package/dist/tests/wave2-cluster-bc.test.js +228 -0
  263. package/dist/tests/wave2-cluster-d.test.js +180 -0
  264. package/dist/tests/wave2-cluster-e.test.js +179 -0
  265. package/dist/tests/wiki-qa-fixes.test.js +270 -0
  266. package/dist/tests/wiki.test.js +529 -0
  267. package/dist/tests/workflow-cli.test.js +271 -0
  268. package/dist/tests/workflow-markdown.test.js +171 -0
  269. package/dist/tests/workflow-path-escape.test.js +132 -0
  270. package/dist/tests/workflow-qa-fixes.test.js +395 -0
  271. package/dist/tests/workflows/indexer-rejection.test.js +213 -0
  272. package/docs/README.md +8 -0
  273. package/docs/migration/release-notes/0.7.0.md +244 -0
  274. package/package.json +2 -2
  275. package/dist/core/warn.js +0 -27
  276. package/dist/output/shapes.js +0 -212
  277. package/dist/output/text.js +0 -520
  278. /package/dist/{commands → src/commands}/completions.js +0 -0
  279. /package/dist/{commands → src/commands}/curate.js +0 -0
  280. /package/dist/{commands → src/commands}/info.js +0 -0
  281. /package/dist/{commands → src/commands}/init.js +0 -0
  282. /package/dist/{commands → src/commands}/install-audit.js +0 -0
  283. /package/dist/{commands → src/commands}/migration-help.js +0 -0
  284. /package/dist/{commands → src/commands}/source-clone.js +0 -0
  285. /package/dist/{commands → src/commands}/vault.js +0 -0
  286. /package/dist/{core → src/core}/asset-registry.js +0 -0
  287. /package/dist/{core → src/core}/frontmatter.js +0 -0
  288. /package/dist/{core → src/core}/markdown.js +0 -0
  289. /package/dist/{core → src/core}/paths.js +0 -0
  290. /package/dist/{indexer → src/indexer}/manifest.js +0 -0
  291. /package/dist/{indexer → src/indexer}/search-fields.js +0 -0
  292. /package/dist/{indexer → src/indexer}/semantic-status.js +0 -0
  293. /package/dist/{indexer → src/indexer}/usage-events.js +0 -0
  294. /package/dist/{indexer → src/indexer}/walker.js +0 -0
  295. /package/dist/{llm → src/llm}/embedder.js +0 -0
  296. /package/dist/{llm → src/llm}/embedders/cache.js +0 -0
  297. /package/dist/{llm → src/llm}/embedders/local.js +0 -0
  298. /package/dist/{llm → src/llm}/embedders/types.js +0 -0
  299. /package/dist/{llm → src/llm}/metadata-enhance.js +0 -0
  300. /package/dist/{output → src/output}/context.js +0 -0
  301. /package/dist/{registry → src/registry}/create-provider-registry.js +0 -0
  302. /package/dist/{registry → src/registry}/origin-resolve.js +0 -0
  303. /package/dist/{registry → src/registry}/providers/index.js +0 -0
  304. /package/dist/{registry → src/registry}/providers/skills-sh.js +0 -0
  305. /package/dist/{registry → src/registry}/providers/types.js +0 -0
  306. /package/dist/{registry → src/registry}/types.js +0 -0
  307. /package/dist/{setup → src/setup}/detect.js +0 -0
  308. /package/dist/{setup → src/setup}/ripgrep-install.js +0 -0
  309. /package/dist/{setup → src/setup}/ripgrep-resolve.js +0 -0
  310. /package/dist/{setup → src/setup}/steps.js +0 -0
  311. /package/dist/{sources → src/sources}/include.js +0 -0
  312. /package/dist/{sources → src/sources}/provider-factory.js +0 -0
  313. /package/dist/{sources → src/sources}/provider.js +0 -0
  314. /package/dist/{sources → src/sources}/providers/filesystem.js +0 -0
  315. /package/dist/{sources → src/sources}/providers/index.js +0 -0
  316. /package/dist/{sources → src/sources}/providers/install-types.js +0 -0
  317. /package/dist/{sources → src/sources}/providers/npm.js +0 -0
  318. /package/dist/{sources → src/sources}/providers/provider-utils.js +0 -0
  319. /package/dist/{sources → src/sources}/providers/sync-from-ref.js +0 -0
  320. /package/dist/{sources → src/sources}/providers/tar-utils.js +0 -0
  321. /package/dist/{sources → src/sources}/providers/website.js +0 -0
  322. /package/dist/{sources → src/sources}/resolve.js +0 -0
  323. /package/dist/{sources → src/sources}/types.js +0 -0
  324. /package/dist/{templates → src/templates}/wiki-templates.js +0 -0
  325. /package/dist/{version.js → src/version.js} +0 -0
  326. /package/dist/{workflows → src/workflows}/authoring.js +0 -0
  327. /package/dist/{workflows → src/workflows}/cli.js +0 -0
  328. /package/dist/{workflows → src/workflows}/db.js +0 -0
  329. /package/dist/{workflows → src/workflows}/document-cache.js +0 -0
  330. /package/dist/{workflows → src/workflows}/parser.js +0 -0
  331. /package/dist/{workflows → src/workflows}/renderer.js +0 -0
  332. /package/dist/{workflows → src/workflows}/schema.js +0 -0
  333. /package/dist/{workflows → src/workflows}/validator.js +0 -0
@@ -0,0 +1,238 @@
1
+ /**
2
+ * Phase 6 (v1 architecture refactor) — coverage for the `static-index`
3
+ * registry provider exercised through the full `RegistryProvider` interface
4
+ * (`search`, `searchKits`, `searchAssets`, `getKit`, `canHandle`).
5
+ *
6
+ * These tests are intentionally siloed from the orchestrator-level tests in
7
+ * `tests/registry-search.test.ts`: they hit the provider directly to make sure
8
+ * the v1-spec §3.1 surface contracts hold for the default provider.
9
+ */
10
+ import { afterAll, afterEach, beforeEach, describe, expect, test } from "bun:test";
11
+ import fs from "node:fs";
12
+ import os from "node:os";
13
+ import path from "node:path";
14
+ import { resolveProviderFactory } from "../../src/registry/factory";
15
+ // Trigger self-registration
16
+ import "../../src/registry/providers/static-index";
17
+ // ── Fixtures ────────────────────────────────────────────────────────────────
18
+ const FIXTURE_INDEX = {
19
+ version: 3,
20
+ updatedAt: "2026-04-25T00:00:00Z",
21
+ stashes: [
22
+ {
23
+ id: "github:vercel-labs/agent-skills",
24
+ name: "agent-skills",
25
+ description: "Production-ready agent skills",
26
+ ref: "vercel-labs/agent-skills",
27
+ source: "github",
28
+ tags: ["agent", "skills"],
29
+ assetTypes: ["skill"],
30
+ assets: [
31
+ { type: "skill", name: "deploy", description: "Deploy to Vercel" },
32
+ { type: "skill", name: "rollback" },
33
+ ],
34
+ author: "vercel-labs",
35
+ latestVersion: "1.0.0",
36
+ },
37
+ {
38
+ id: "npm:@itlackey/openkit",
39
+ name: "@itlackey/openkit",
40
+ description: "OpenCode starter",
41
+ ref: "@itlackey/openkit",
42
+ source: "npm",
43
+ tags: ["opencode", "starter"],
44
+ author: "itlackey",
45
+ },
46
+ ],
47
+ };
48
+ // ── Helpers ─────────────────────────────────────────────────────────────────
49
+ const createdTmpDirs = [];
50
+ const servers = [];
51
+ function createTmpDir(prefix = "akm-static-index-") {
52
+ const dir = fs.mkdtempSync(path.join(os.tmpdir(), prefix));
53
+ createdTmpDirs.push(dir);
54
+ return dir;
55
+ }
56
+ function serveJson(body) {
57
+ const server = Bun.serve({
58
+ port: 0,
59
+ fetch() {
60
+ return new Response(JSON.stringify(body), {
61
+ headers: { "Content-Type": "application/json" },
62
+ });
63
+ },
64
+ });
65
+ servers.push(server);
66
+ return {
67
+ url: `http://localhost:${server.port}/index.json`,
68
+ close: () => server.stop(true),
69
+ };
70
+ }
71
+ function makeProvider(url, name = "official") {
72
+ const factory = resolveProviderFactory("static-index");
73
+ if (!factory)
74
+ throw new Error("static-index provider not registered");
75
+ return factory({ url, name });
76
+ }
77
+ const originalXdgCacheHome = process.env.XDG_CACHE_HOME;
78
+ beforeEach(() => {
79
+ process.env.XDG_CACHE_HOME = createTmpDir("akm-si-cache-");
80
+ });
81
+ afterEach(() => {
82
+ for (const s of servers) {
83
+ try {
84
+ s.stop(true);
85
+ }
86
+ catch {
87
+ /* already stopped */
88
+ }
89
+ }
90
+ servers.length = 0;
91
+ if (originalXdgCacheHome === undefined) {
92
+ delete process.env.XDG_CACHE_HOME;
93
+ }
94
+ else {
95
+ process.env.XDG_CACHE_HOME = originalXdgCacheHome;
96
+ }
97
+ });
98
+ afterAll(() => {
99
+ for (const dir of createdTmpDirs) {
100
+ fs.rmSync(dir, { recursive: true, force: true });
101
+ }
102
+ });
103
+ // ── Tests ───────────────────────────────────────────────────────────────────
104
+ describe("StaticIndexProvider", () => {
105
+ test("factory is registered", () => {
106
+ expect(resolveProviderFactory("static-index")).not.toBeNull();
107
+ });
108
+ describe("searchKits (v1-spec §3.1)", () => {
109
+ test("returns KitResult entries with installRef", async () => {
110
+ const srv = serveJson(FIXTURE_INDEX);
111
+ const provider = makeProvider(srv.url);
112
+ const kits = await provider.searchKits({ text: "skills", limit: 10 });
113
+ expect(kits.length).toBeGreaterThan(0);
114
+ for (const kit of kits) {
115
+ expect(typeof kit.id).toBe("string");
116
+ expect(typeof kit.title).toBe("string");
117
+ expect(typeof kit.installRef).toBe("string");
118
+ }
119
+ });
120
+ test("kit installRef is a valid akm add target", async () => {
121
+ const srv = serveJson(FIXTURE_INDEX);
122
+ const provider = makeProvider(srv.url);
123
+ const kits = await provider.searchKits({ text: "agent", limit: 10 });
124
+ const githubKit = kits.find((k) => k.id.startsWith("github:"));
125
+ expect(githubKit?.installRef).toBe("github:vercel-labs/agent-skills");
126
+ });
127
+ test("respects limit", async () => {
128
+ const srv = serveJson(FIXTURE_INDEX);
129
+ const provider = makeProvider(srv.url);
130
+ const kits = await provider.searchKits({ text: "agent skills opencode", limit: 1 });
131
+ expect(kits.length).toBeLessThanOrEqual(1);
132
+ });
133
+ });
134
+ describe("searchAssets (v1-spec §3.1)", () => {
135
+ test("returns AssetPreview entries scoped to kits", async () => {
136
+ const srv = serveJson(FIXTURE_INDEX);
137
+ const provider = makeProvider(srv.url);
138
+ const assets = await provider.searchAssets?.({ text: "deploy", limit: 10 });
139
+ expect(assets).toBeDefined();
140
+ const deploy = assets?.find((a) => a.name === "deploy");
141
+ expect(deploy?.type).toBe("skill");
142
+ expect(deploy?.kitId).toBe("github:vercel-labs/agent-skills");
143
+ expect(deploy?.cloneRef).toBe("github:vercel-labs/agent-skills");
144
+ });
145
+ });
146
+ describe("getKit (v1-spec §3.1)", () => {
147
+ test("returns a KitManifest for a known id", async () => {
148
+ const srv = serveJson(FIXTURE_INDEX);
149
+ const provider = makeProvider(srv.url);
150
+ const manifest = await provider.getKit("github:vercel-labs/agent-skills");
151
+ expect(manifest).not.toBeNull();
152
+ expect(manifest?.id).toBe("github:vercel-labs/agent-skills");
153
+ expect(manifest?.installRef).toBe("github:vercel-labs/agent-skills");
154
+ expect(manifest?.assets?.length ?? 0).toBeGreaterThan(0);
155
+ });
156
+ test("returns null for an unknown id", async () => {
157
+ const srv = serveJson(FIXTURE_INDEX);
158
+ const provider = makeProvider(srv.url);
159
+ const manifest = await provider.getKit("github:does-not-exist/anywhere");
160
+ expect(manifest).toBeNull();
161
+ });
162
+ test("preserves npm install refs", async () => {
163
+ const srv = serveJson(FIXTURE_INDEX);
164
+ const provider = makeProvider(srv.url);
165
+ const manifest = await provider.getKit("npm:@itlackey/openkit");
166
+ expect(manifest?.installRef).toBe("npm:@itlackey/openkit");
167
+ });
168
+ });
169
+ describe("canHandle (plan §9 item 2)", () => {
170
+ test("claims github refs", () => {
171
+ const srv = serveJson({ stashes: [] });
172
+ const provider = makeProvider(srv.url);
173
+ const ref = {
174
+ source: "github",
175
+ ref: "owner/repo",
176
+ id: "github:owner/repo",
177
+ owner: "owner",
178
+ repo: "repo",
179
+ };
180
+ expect(provider.canHandle(ref)).toBe(true);
181
+ });
182
+ test("claims npm refs (default catch-all)", () => {
183
+ const srv = serveJson({ stashes: [] });
184
+ const provider = makeProvider(srv.url);
185
+ const ref = {
186
+ source: "npm",
187
+ ref: "npm:foo",
188
+ id: "npm:foo",
189
+ packageName: "foo",
190
+ };
191
+ expect(provider.canHandle(ref)).toBe(true);
192
+ });
193
+ });
194
+ describe("backwards-compat search", () => {
195
+ test("legacy search() still returns RegistrySearchHit shape", async () => {
196
+ const srv = serveJson(FIXTURE_INDEX);
197
+ const provider = makeProvider(srv.url);
198
+ const result = await provider.search({ query: "agent", limit: 10 });
199
+ expect(Array.isArray(result.hits)).toBe(true);
200
+ const hit = result.hits.find((h) => h.id === "github:vercel-labs/agent-skills");
201
+ expect(hit?.installRef).toBe("github:vercel-labs/agent-skills");
202
+ });
203
+ });
204
+ describe("registry version contract", () => {
205
+ test("version 3 index parses without warnings (canonical format)", async () => {
206
+ const srv = serveJson(FIXTURE_INDEX); // FIXTURE_INDEX has version: 3
207
+ const provider = makeProvider(srv.url);
208
+ const result = await provider.search({ query: "agent", limit: 10 });
209
+ expect(result.warnings ?? []).toHaveLength(0);
210
+ expect(result.hits.length).toBeGreaterThan(0);
211
+ });
212
+ test("version 2 index parses without warnings (live official registry format)", async () => {
213
+ const v2Index = { ...FIXTURE_INDEX, version: 2 };
214
+ const srv = serveJson(v2Index);
215
+ const provider = makeProvider(srv.url);
216
+ const result = await provider.search({ query: "agent", limit: 10 });
217
+ expect(result.warnings ?? []).toHaveLength(0);
218
+ expect(result.hits.length).toBeGreaterThan(0);
219
+ });
220
+ test("version 2 index returns correct kit hits", async () => {
221
+ const v2Index = { ...FIXTURE_INDEX, version: 2 };
222
+ const srv = serveJson(v2Index);
223
+ const provider = makeProvider(srv.url);
224
+ const kits = await provider.searchKits({ text: "agent", limit: 10 });
225
+ expect(kits.length).toBeGreaterThan(0);
226
+ expect(kits.some((k) => k.id === "github:vercel-labs/agent-skills")).toBe(true);
227
+ });
228
+ test("version 1 index returns null (unsupported)", async () => {
229
+ // version 1 is explicitly unsupported per schema comment
230
+ const v1Index = { ...FIXTURE_INDEX, version: 1 };
231
+ const srv = serveJson(v1Index);
232
+ const provider = makeProvider(srv.url);
233
+ const result = await provider.search({ query: "agent", limit: 10 });
234
+ // No hits because the parser returns null for unsupported versions
235
+ expect(result.hits).toHaveLength(0);
236
+ });
237
+ });
238
+ });
@@ -0,0 +1,126 @@
1
+ import { afterEach, beforeEach, describe, expect, test } from "bun:test";
2
+ import { UsageError } from "../src/core/errors";
3
+ import { trustedNpmTarballHosts, UntrustedNpmTarballError, validateGitRef, validateGitUrl, validateNpmTarballUrl, } from "../src/registry/resolve";
4
+ // ── validateGitUrl ───────────────────────────────────────────────────────────
5
+ describe("validateGitUrl", () => {
6
+ // ── Rejected schemes ──────────────────────────────────────────────────────
7
+ test("rejects ext:: protocol helper (arbitrary command execution)", () => {
8
+ expect(() => validateGitUrl("ext::evil-command arg")).toThrow(UsageError);
9
+ });
10
+ test("rejects fd:: protocol helper", () => {
11
+ expect(() => validateGitUrl("fd::5")).toThrow(UsageError);
12
+ });
13
+ test("rejects file:// scheme (local path traversal)", () => {
14
+ expect(() => validateGitUrl("file:///etc/passwd")).toThrow(UsageError);
15
+ });
16
+ test("rejects ftp:// scheme", () => {
17
+ expect(() => validateGitUrl("ftp://example.com/repo")).toThrow(UsageError);
18
+ });
19
+ test("rejects completely invalid URLs", () => {
20
+ expect(() => validateGitUrl("not a url at all !!")).toThrow(UsageError);
21
+ });
22
+ // ── Accepted schemes ──────────────────────────────────────────────────────
23
+ test("accepts https:// URLs", () => {
24
+ expect(() => validateGitUrl("https://github.com/valid/repo.git")).not.toThrow();
25
+ });
26
+ test("accepts http:// URLs", () => {
27
+ expect(() => validateGitUrl("http://internal.example.com/repo.git")).not.toThrow();
28
+ });
29
+ test("accepts ssh:// URLs", () => {
30
+ expect(() => validateGitUrl("ssh://git@github.com/org/repo.git")).not.toThrow();
31
+ });
32
+ test("accepts git:// URLs", () => {
33
+ expect(() => validateGitUrl("git://github.com/org/repo.git")).not.toThrow();
34
+ });
35
+ test("accepts git@ SSH shorthand", () => {
36
+ expect(() => validateGitUrl("git@github.com:org/repo.git")).not.toThrow();
37
+ });
38
+ test("accepts git@ SSH shorthand with subdomain", () => {
39
+ expect(() => validateGitUrl("git@gitlab.example.com:group/subgroup/repo.git")).not.toThrow();
40
+ });
41
+ });
42
+ // ── validateGitRef ───────────────────────────────────────────────────────────
43
+ describe("validateGitRef", () => {
44
+ // ── Rejected patterns ─────────────────────────────────────────────────────
45
+ test("rejects refs with semicolons", () => {
46
+ expect(() => validateGitRef("main;rm -rf /")).toThrow(UsageError);
47
+ });
48
+ test("rejects refs with spaces", () => {
49
+ expect(() => validateGitRef("main branch")).toThrow(UsageError);
50
+ });
51
+ test("rejects refs with shell special chars", () => {
52
+ expect(() => validateGitRef("$(evil)")).toThrow(UsageError);
53
+ expect(() => validateGitRef("`evil`")).toThrow(UsageError);
54
+ expect(() => validateGitRef("main&evil")).toThrow(UsageError);
55
+ });
56
+ test("rejects empty string", () => {
57
+ expect(() => validateGitRef("")).toThrow(UsageError);
58
+ });
59
+ // ── Accepted patterns ─────────────────────────────────────────────────────
60
+ test("accepts branch names", () => {
61
+ expect(() => validateGitRef("main")).not.toThrow();
62
+ expect(() => validateGitRef("feat/my-feature")).not.toThrow();
63
+ expect(() => validateGitRef("v1.2.3")).not.toThrow();
64
+ expect(() => validateGitRef("release-candidate_1")).not.toThrow();
65
+ });
66
+ test("accepts full git SHA hashes", () => {
67
+ expect(() => validateGitRef("abc1234def567890abc1234def567890abc12345")).not.toThrow();
68
+ });
69
+ test("accepts short SHA hashes", () => {
70
+ expect(() => validateGitRef("abc1234")).not.toThrow();
71
+ });
72
+ });
73
+ // ── validateNpmTarballUrl ───────────────────────────────────────────────────
74
+ describe("validateNpmTarballUrl", () => {
75
+ const originalRegistry = process.env.AKM_NPM_REGISTRY;
76
+ beforeEach(() => {
77
+ delete process.env.AKM_NPM_REGISTRY;
78
+ });
79
+ afterEach(() => {
80
+ if (originalRegistry === undefined) {
81
+ delete process.env.AKM_NPM_REGISTRY;
82
+ }
83
+ else {
84
+ process.env.AKM_NPM_REGISTRY = originalRegistry;
85
+ }
86
+ });
87
+ test("accepts public registry tarball", () => {
88
+ expect(() => validateNpmTarballUrl("https://registry.npmjs.org/@scope/pkg/-/pkg-1.0.0.tgz", "@scope/pkg@1.0.0")).not.toThrow();
89
+ });
90
+ test("rejects tarball on attacker-controlled host", () => {
91
+ let caught;
92
+ try {
93
+ validateNpmTarballUrl("https://evil.example.com/pkg-1.0.0.tgz", "pkg@1.0.0");
94
+ }
95
+ catch (err) {
96
+ caught = err;
97
+ }
98
+ expect(caught).toBeInstanceOf(UntrustedNpmTarballError);
99
+ expect(caught.code).toBe("UNTRUSTED_NPM_TARBALL");
100
+ expect(caught?.message).toContain("evil.example.com");
101
+ });
102
+ test("rejects malformed tarball URL", () => {
103
+ expect(() => validateNpmTarballUrl("not-a-url", "pkg@1.0.0")).toThrow(UntrustedNpmTarballError);
104
+ });
105
+ test("rejects disallowed scheme", () => {
106
+ expect(() => validateNpmTarballUrl("ftp://registry.npmjs.org/pkg.tgz", "pkg@1.0.0")).toThrow(UntrustedNpmTarballError);
107
+ });
108
+ test("accepts operator-configured private registry", () => {
109
+ process.env.AKM_NPM_REGISTRY = "https://npm.internal.example.com";
110
+ expect(() => validateNpmTarballUrl("https://npm.internal.example.com/pkg/-/pkg-2.0.0.tgz", "pkg@2.0.0")).not.toThrow();
111
+ });
112
+ test("still accepts public registry alongside operator override", () => {
113
+ process.env.AKM_NPM_REGISTRY = "https://npm.internal.example.com";
114
+ expect(() => validateNpmTarballUrl("https://registry.npmjs.org/pkg/-/pkg-1.0.0.tgz", "pkg@1.0.0")).not.toThrow();
115
+ });
116
+ test("rejects untrusted host even with override set", () => {
117
+ process.env.AKM_NPM_REGISTRY = "https://npm.internal.example.com";
118
+ expect(() => validateNpmTarballUrl("https://evil.example.com/pkg.tgz", "pkg@1.0.0")).toThrow(UntrustedNpmTarballError);
119
+ });
120
+ test("ignores unparseable AKM_NPM_REGISTRY override", () => {
121
+ process.env.AKM_NPM_REGISTRY = "this is not a url";
122
+ const hosts = trustedNpmTarballHosts();
123
+ expect(hosts.has("registry.npmjs.org")).toBe(true);
124
+ expect(hosts.size).toBe(1);
125
+ });
126
+ });