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.
- package/CHANGELOG.md +66 -0
- package/dist/{cli.js → src/cli.js} +672 -29
- package/dist/{commands → src/commands}/config-cli.js +5 -4
- package/dist/src/commands/distill.js +283 -0
- package/dist/src/commands/events.js +108 -0
- package/dist/src/commands/history.js +120 -0
- package/dist/{commands → src/commands}/installed-stashes.js +28 -2
- package/dist/src/commands/proposal.js +119 -0
- package/dist/src/commands/propose.js +171 -0
- package/dist/src/commands/reflect.js +193 -0
- package/dist/{commands → src/commands}/registry-search.js +2 -1
- package/dist/{commands → src/commands}/remember.js +12 -0
- package/dist/{commands → src/commands}/search.js +74 -1
- package/dist/{commands → src/commands}/self-update.js +4 -3
- package/dist/{commands → src/commands}/show.js +67 -2
- package/dist/{core → src/core}/asset-ref.js +5 -5
- package/dist/{core → src/core}/asset-spec.js +12 -0
- package/dist/{core → src/core}/common.js +1 -1
- package/dist/{core → src/core}/config.js +175 -121
- package/dist/{core → src/core}/errors.js +4 -0
- package/dist/src/core/events.js +239 -0
- package/dist/src/core/lesson-lint.js +86 -0
- package/dist/src/core/proposals.js +406 -0
- package/dist/src/core/warn.js +72 -0
- package/dist/{core → src/core}/write-source.js +80 -5
- package/dist/{indexer → src/indexer}/db-search.js +119 -27
- package/dist/{indexer → src/indexer}/db.js +76 -23
- package/dist/{indexer → src/indexer}/file-context.js +0 -3
- package/dist/src/indexer/graph-boost.js +179 -0
- package/dist/src/indexer/graph-extraction.js +212 -0
- package/dist/{indexer → src/indexer}/indexer.js +73 -6
- package/dist/src/indexer/memory-inference.js +263 -0
- package/dist/{indexer → src/indexer}/metadata.js +114 -11
- package/dist/src/integrations/agent/config.js +292 -0
- package/dist/src/integrations/agent/detect.js +94 -0
- package/dist/src/integrations/agent/index.js +17 -0
- package/dist/src/integrations/agent/profiles.js +65 -0
- package/dist/src/integrations/agent/prompts.js +167 -0
- package/dist/src/integrations/agent/spawn.js +221 -0
- package/dist/{integrations → src/integrations}/lockfile.js +0 -26
- package/dist/{llm → src/llm}/client.js +33 -2
- package/dist/src/llm/feature-gate.js +108 -0
- package/dist/src/llm/graph-extract.js +107 -0
- package/dist/src/llm/index-passes.js +35 -0
- package/dist/src/llm/memory-infer.js +86 -0
- package/dist/{output → src/output}/renderers.js +60 -1
- package/dist/src/output/shapes.js +516 -0
- package/dist/{output → src/output}/text.js +447 -4
- package/dist/{registry → src/registry}/build-index.js +14 -4
- package/dist/{registry → src/registry}/factory.js +0 -8
- package/dist/{registry → src/registry}/providers/static-index.js +3 -2
- package/dist/{registry → src/registry}/resolve.js +68 -2
- package/dist/{setup → src/setup}/setup.js +43 -5
- package/dist/{sources → src/sources}/providers/git.js +7 -15
- package/dist/{wiki → src/wiki}/wiki.js +9 -11
- package/dist/tests/add-website-source.test.js +119 -0
- package/dist/tests/agent/agent-config-loader.test.js +70 -0
- package/dist/tests/agent/agent-config.test.js +221 -0
- package/dist/tests/agent/agent-detect.test.js +100 -0
- package/dist/tests/agent/agent-spawn.test.js +234 -0
- package/dist/tests/agent-output.test.js +186 -0
- package/dist/tests/architecture/agent-no-llm-sdk-guard.test.js +103 -0
- package/dist/tests/architecture/agent-spawn-seam.test.js +193 -0
- package/dist/tests/architecture/llm-stateless-seam.test.js +112 -0
- package/dist/tests/asset-ref.test.js +192 -0
- package/dist/tests/asset-registry.test.js +103 -0
- package/dist/tests/asset-spec.test.js +241 -0
- package/dist/tests/bench/attribution.test.js +995 -0
- package/dist/tests/bench/cleanup-sigint.test.js +83 -0
- package/dist/tests/bench/cleanup.js +203 -0
- package/dist/tests/bench/cleanup.test.js +166 -0
- package/dist/tests/bench/cli.js +683 -0
- package/dist/tests/bench/cli.test.js +177 -0
- package/dist/tests/bench/compare.test.js +556 -0
- package/dist/tests/bench/corpus.js +314 -0
- package/dist/tests/bench/corpus.test.js +258 -0
- package/dist/tests/bench/driver.js +346 -0
- package/dist/tests/bench/driver.test.js +443 -0
- package/dist/tests/bench/evolve-metrics.js +179 -0
- package/dist/tests/bench/evolve-metrics.test.js +187 -0
- package/dist/tests/bench/evolve.js +580 -0
- package/dist/tests/bench/evolve.test.js +616 -0
- package/dist/tests/bench/failure-modes.test.js +300 -0
- package/dist/tests/bench/feedback-integrity.test.js +456 -0
- package/dist/tests/bench/leakage.test.js +125 -0
- package/dist/tests/bench/learning-curve.test.js +133 -0
- package/dist/tests/bench/metrics.js +2319 -0
- package/dist/tests/bench/metrics.test.js +1144 -0
- package/dist/tests/bench/no-os-tmpdir-invariant.test.js +43 -0
- package/dist/tests/bench/report.js +1821 -0
- package/dist/tests/bench/report.test.js +989 -0
- package/dist/tests/bench/runner.js +536 -0
- package/dist/tests/bench/runner.test.js +958 -0
- package/dist/tests/bench/search-bridge.test.js +331 -0
- package/dist/tests/bench/tmp.js +41 -0
- package/dist/tests/bench/trajectory.js +116 -0
- package/dist/tests/bench/trajectory.test.js +127 -0
- package/dist/tests/bench/verifier.js +109 -0
- package/dist/tests/bench/verifier.test.js +118 -0
- package/dist/tests/bench/workflow-evaluator.js +557 -0
- package/dist/tests/bench/workflow-evaluator.test.js +421 -0
- package/dist/tests/bench/workflow-spec.js +358 -0
- package/dist/tests/bench/workflow-spec.test.js +363 -0
- package/dist/tests/bench/workflow-trace.js +438 -0
- package/dist/tests/bench/workflow-trace.test.js +254 -0
- package/dist/tests/benchmark-search-quality.js +536 -0
- package/dist/tests/benchmark-suite.js +1441 -0
- package/dist/tests/capture-cli.test.js +112 -0
- package/dist/tests/cli-errors.test.js +203 -0
- package/dist/tests/commands/events.test.js +370 -0
- package/dist/tests/commands/history.test.js +223 -0
- package/dist/tests/commands/import.test.js +103 -0
- package/dist/tests/commands/proposal-cli.test.js +209 -0
- package/dist/tests/commands/reflect-propose-cli.test.js +333 -0
- package/dist/tests/commands/remember.test.js +97 -0
- package/dist/tests/commands/scope-flags.test.js +300 -0
- package/dist/tests/commands/search.test.js +537 -0
- package/dist/tests/commands/show-indexer-parity.test.js +117 -0
- package/dist/tests/commands/show.test.js +294 -0
- package/dist/tests/common.test.js +266 -0
- package/dist/tests/completions.test.js +142 -0
- package/dist/tests/config-cli.test.js +193 -0
- package/dist/tests/config-llm-features.test.js +139 -0
- package/dist/tests/config.test.js +544 -0
- package/dist/tests/contracts/migration-baseline.test.js +43 -0
- package/dist/tests/contracts/reflect-propose-envelope.test.js +139 -0
- package/dist/tests/contracts/spec-helpers.js +46 -0
- package/dist/tests/contracts/v1-spec-section-11-proposal-queue.test.js +228 -0
- package/dist/tests/contracts/v1-spec-section-12-agent-config.test.js +56 -0
- package/dist/tests/contracts/v1-spec-section-13-lesson-type.test.js +34 -0
- package/dist/tests/contracts/v1-spec-section-14-llm-features.test.js +94 -0
- package/dist/tests/contracts/v1-spec-section-4-1-asset-types.test.js +39 -0
- package/dist/tests/contracts/v1-spec-section-4-2-quality-rules.test.js +44 -0
- package/dist/tests/contracts/v1-spec-section-5-configuration.test.js +47 -0
- package/dist/tests/contracts/v1-spec-section-6-orchestration.test.js +40 -0
- package/dist/tests/contracts/v1-spec-section-7-module-layout.test.js +58 -0
- package/dist/tests/contracts/v1-spec-section-8-extension-points.test.js +34 -0
- package/dist/tests/contracts/v1-spec-section-9-4-cli-surface.test.js +75 -0
- package/dist/tests/contracts/v1-spec-section-9-7-llm-agent-boundary.test.js +36 -0
- package/dist/tests/core/write-source.test.js +366 -0
- package/dist/tests/curate-command.test.js +87 -0
- package/dist/tests/db-scoring.test.js +201 -0
- package/dist/tests/db.test.js +654 -0
- package/dist/tests/distill-cli-flag.test.js +208 -0
- package/dist/tests/distill.test.js +515 -0
- package/dist/tests/docker-install.test.js +120 -0
- package/dist/tests/e2e.test.js +1398 -0
- package/dist/tests/embedder.test.js +340 -0
- package/dist/tests/embedding-model-config.test.js +379 -0
- package/dist/tests/feedback-command.test.js +172 -0
- package/dist/tests/file-context.test.js +552 -0
- package/dist/tests/fixtures/scripts/git/summarize-diff.js +9 -0
- package/dist/tests/fixtures/scripts/lint/eslint-check.js +7 -0
- package/dist/tests/fixtures/stashes/load.js +166 -0
- package/dist/tests/fixtures/stashes/load.test.js +88 -0
- package/dist/tests/fixtures/stashes/ranking-baseline/scripts/mem0-search.js +12 -0
- package/dist/tests/frontmatter.test.js +190 -0
- package/dist/tests/fts-field-weighting.test.js +254 -0
- package/dist/tests/fuzzy-search.test.js +230 -0
- package/dist/tests/git-provider-clone.test.js +45 -0
- package/dist/tests/github.test.js +161 -0
- package/dist/tests/graph-boost-ranking.test.js +305 -0
- package/dist/tests/graph-extraction.test.js +282 -0
- package/dist/tests/helpers/usage-events.js +8 -0
- package/dist/tests/index-pass-llm.test.js +161 -0
- package/dist/tests/indexer.test.js +559 -0
- package/dist/tests/info-command.test.js +166 -0
- package/dist/tests/init.test.js +69 -0
- package/dist/tests/install-script.test.js +246 -0
- package/dist/tests/integration/agent-real-profile.test.js +94 -0
- package/dist/tests/issue-36-repro.test.js +304 -0
- package/dist/tests/issues-191-194.test.js +160 -0
- package/dist/tests/lesson-lint.test.js +111 -0
- package/dist/tests/llm-client.test.js +115 -0
- package/dist/tests/llm-feature-gate.test.js +151 -0
- package/dist/tests/llm.test.js +139 -0
- package/dist/tests/lockfile.test.js +216 -0
- package/dist/tests/manifest.test.js +205 -0
- package/dist/tests/markdown.test.js +126 -0
- package/dist/tests/matchers-unit.test.js +189 -0
- package/dist/tests/memory-inference.test.js +299 -0
- package/dist/tests/merge-scoring.test.js +136 -0
- package/dist/tests/metadata.test.js +313 -0
- package/dist/tests/migration-help.test.js +89 -0
- package/dist/tests/origin-resolve.test.js +124 -0
- package/dist/tests/output-baseline.test.js +217 -0
- package/dist/tests/output-shapes-unit.test.js +476 -0
- package/dist/tests/parallel-search.test.js +272 -0
- package/dist/tests/parameter-metadata.test.js +365 -0
- package/dist/tests/paths.test.js +177 -0
- package/dist/tests/progressive-disclosure.test.js +280 -0
- package/dist/tests/proposals.test.js +279 -0
- package/dist/tests/proposed-quality.test.js +271 -0
- package/dist/tests/provider-registry.test.js +32 -0
- package/dist/tests/ranking-regression.test.js +548 -0
- package/dist/tests/reflect-propose.test.js +455 -0
- package/dist/tests/registry-build-index.test.js +378 -0
- package/dist/tests/registry-cli.test.js +290 -0
- package/dist/tests/registry-index-v2.test.js +430 -0
- package/dist/tests/registry-install.test.js +728 -0
- package/dist/tests/registry-providers/parity.test.js +189 -0
- package/dist/tests/registry-providers/skills-sh.test.js +309 -0
- package/dist/tests/registry-providers/static-index.test.js +204 -0
- package/dist/tests/registry-resolve.test.js +126 -0
- package/dist/tests/registry-search.test.js +723 -0
- package/dist/tests/remember-frontmatter.test.js +380 -0
- package/dist/tests/remember-unit.test.js +123 -0
- package/dist/tests/ripgrep-install.test.js +251 -0
- package/dist/tests/ripgrep-resolve.test.js +108 -0
- package/dist/tests/ripgrep.test.js +163 -0
- package/dist/tests/save-command.test.js +94 -0
- package/dist/tests/save-trust-qa-fixes.test.js +270 -0
- package/dist/tests/scoring-pipeline.test.js +648 -0
- package/dist/tests/search-include-proposed-cli.test.js +118 -0
- package/dist/tests/self-update.test.js +442 -0
- package/dist/tests/semantic-search-e2e.test.js +512 -0
- package/dist/tests/semantic-status.test.js +471 -0
- package/dist/tests/setup-run.integration.js +877 -0
- package/dist/tests/setup-wizard.test.js +198 -0
- package/dist/tests/setup.test.js +131 -0
- package/dist/tests/source-add.test.js +11 -0
- package/dist/tests/source-clone.test.js +254 -0
- package/dist/tests/source-manage.test.js +366 -0
- package/dist/tests/source-providers/filesystem.test.js +82 -0
- package/dist/tests/source-providers/git.test.js +252 -0
- package/dist/tests/source-providers/website.test.js +128 -0
- package/dist/tests/source-qa-fixes.test.js +268 -0
- package/dist/tests/source-registry.test.js +350 -0
- package/dist/tests/source-resolve.test.js +100 -0
- package/dist/tests/source-source.test.js +221 -0
- package/dist/tests/source.test.js +533 -0
- package/dist/tests/tar-utils-scan.test.js +73 -0
- package/dist/tests/toggle-components.test.js +73 -0
- package/dist/tests/usage-telemetry.test.js +265 -0
- package/dist/tests/utility-scoring.test.js +558 -0
- package/dist/tests/vault-load-error.test.js +78 -0
- package/dist/tests/vault-qa-fixes.test.js +194 -0
- package/dist/tests/vault.test.js +429 -0
- package/dist/tests/vector-search.test.js +608 -0
- package/dist/tests/walker.test.js +252 -0
- package/dist/tests/wave2-cluster-bc.test.js +228 -0
- package/dist/tests/wave2-cluster-d.test.js +180 -0
- package/dist/tests/wave2-cluster-e.test.js +179 -0
- package/dist/tests/wiki-qa-fixes.test.js +270 -0
- package/dist/tests/wiki.test.js +529 -0
- package/dist/tests/workflow-cli.test.js +271 -0
- package/dist/tests/workflow-markdown.test.js +171 -0
- package/dist/tests/workflow-path-escape.test.js +132 -0
- package/dist/tests/workflow-qa-fixes.test.js +377 -0
- package/dist/tests/workflows/indexer-rejection.test.js +213 -0
- package/docs/README.md +8 -0
- package/docs/migration/release-notes/0.7.0.md +244 -0
- package/package.json +2 -2
- package/dist/core/warn.js +0 -27
- package/dist/output/shapes.js +0 -212
- /package/dist/{commands → src/commands}/completions.js +0 -0
- /package/dist/{commands → src/commands}/curate.js +0 -0
- /package/dist/{commands → src/commands}/info.js +0 -0
- /package/dist/{commands → src/commands}/init.js +0 -0
- /package/dist/{commands → src/commands}/install-audit.js +0 -0
- /package/dist/{commands → src/commands}/migration-help.js +0 -0
- /package/dist/{commands → src/commands}/source-add.js +0 -0
- /package/dist/{commands → src/commands}/source-clone.js +0 -0
- /package/dist/{commands → src/commands}/source-manage.js +0 -0
- /package/dist/{commands → src/commands}/vault.js +0 -0
- /package/dist/{core → src/core}/asset-registry.js +0 -0
- /package/dist/{core → src/core}/frontmatter.js +0 -0
- /package/dist/{core → src/core}/markdown.js +0 -0
- /package/dist/{core → src/core}/paths.js +0 -0
- /package/dist/{indexer → src/indexer}/manifest.js +0 -0
- /package/dist/{indexer → src/indexer}/matchers.js +0 -0
- /package/dist/{indexer → src/indexer}/search-fields.js +0 -0
- /package/dist/{indexer → src/indexer}/search-source.js +0 -0
- /package/dist/{indexer → src/indexer}/semantic-status.js +0 -0
- /package/dist/{indexer → src/indexer}/usage-events.js +0 -0
- /package/dist/{indexer → src/indexer}/walker.js +0 -0
- /package/dist/{integrations → src/integrations}/github.js +0 -0
- /package/dist/{llm → src/llm}/embedder.js +0 -0
- /package/dist/{llm → src/llm}/embedders/cache.js +0 -0
- /package/dist/{llm → src/llm}/embedders/local.js +0 -0
- /package/dist/{llm → src/llm}/embedders/remote.js +0 -0
- /package/dist/{llm → src/llm}/embedders/types.js +0 -0
- /package/dist/{llm → src/llm}/metadata-enhance.js +0 -0
- /package/dist/{output → src/output}/cli-hints.js +0 -0
- /package/dist/{output → src/output}/context.js +0 -0
- /package/dist/{registry → src/registry}/create-provider-registry.js +0 -0
- /package/dist/{registry → src/registry}/origin-resolve.js +0 -0
- /package/dist/{registry → src/registry}/providers/index.js +0 -0
- /package/dist/{registry → src/registry}/providers/skills-sh.js +0 -0
- /package/dist/{registry → src/registry}/providers/types.js +0 -0
- /package/dist/{registry → src/registry}/types.js +0 -0
- /package/dist/{setup → src/setup}/detect.js +0 -0
- /package/dist/{setup → src/setup}/ripgrep-install.js +0 -0
- /package/dist/{setup → src/setup}/ripgrep-resolve.js +0 -0
- /package/dist/{setup → src/setup}/steps.js +0 -0
- /package/dist/{sources → src/sources}/include.js +0 -0
- /package/dist/{sources → src/sources}/provider-factory.js +0 -0
- /package/dist/{sources → src/sources}/provider.js +0 -0
- /package/dist/{sources → src/sources}/providers/filesystem.js +0 -0
- /package/dist/{sources → src/sources}/providers/index.js +0 -0
- /package/dist/{sources → src/sources}/providers/install-types.js +0 -0
- /package/dist/{sources → src/sources}/providers/npm.js +0 -0
- /package/dist/{sources → src/sources}/providers/provider-utils.js +0 -0
- /package/dist/{sources → src/sources}/providers/sync-from-ref.js +0 -0
- /package/dist/{sources → src/sources}/providers/tar-utils.js +0 -0
- /package/dist/{sources → src/sources}/providers/website.js +0 -0
- /package/dist/{sources → src/sources}/resolve.js +0 -0
- /package/dist/{sources → src/sources}/types.js +0 -0
- /package/dist/{templates → src/templates}/wiki-templates.js +0 -0
- /package/dist/{version.js → src/version.js} +0 -0
- /package/dist/{workflows → src/workflows}/authoring.js +0 -0
- /package/dist/{workflows → src/workflows}/cli.js +0 -0
- /package/dist/{workflows → src/workflows}/db.js +0 -0
- /package/dist/{workflows → src/workflows}/document-cache.js +0 -0
- /package/dist/{workflows → src/workflows}/parser.js +0 -0
- /package/dist/{workflows → src/workflows}/renderer.js +0 -0
- /package/dist/{workflows → src/workflows}/runs.js +0 -0
- /package/dist/{workflows → src/workflows}/schema.js +0 -0
- /package/dist/{workflows → src/workflows}/validator.js +0 -0
|
@@ -0,0 +1,251 @@
|
|
|
1
|
+
import { afterEach, 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
|
+
// We test the module by importing and mocking its dependencies.
|
|
7
|
+
// Since ensureRg calls spawnSync and resolveRg, we mock at the module level.
|
|
8
|
+
// ── Helpers ─────────────────────────────────────────────────────────────────
|
|
9
|
+
let tmpDirs = [];
|
|
10
|
+
function makeTmpDir() {
|
|
11
|
+
const dir = fs.mkdtempSync(path.join(os.tmpdir(), "akm-rg-install-test-"));
|
|
12
|
+
tmpDirs.push(dir);
|
|
13
|
+
return dir;
|
|
14
|
+
}
|
|
15
|
+
function makeToolchainDir() {
|
|
16
|
+
const dir = makeTmpDir();
|
|
17
|
+
fs.writeFileSync(path.join(dir, "curl"), '#!/bin/sh\nout=\'\'\nwhile [ $# -gt 0 ]; do\n if [ "$1" = "-o" ]; then\n out=$2\n shift 2\n continue\n fi\n shift\ndone\n/bin/cp "$FAKE_CURL_SOURCE" "$out"\n');
|
|
18
|
+
fs.chmodSync(path.join(dir, "curl"), 0o755);
|
|
19
|
+
fs.symlinkSync("/usr/bin/tar", path.join(dir, "tar"));
|
|
20
|
+
fs.symlinkSync("/usr/bin/gzip", path.join(dir, "gzip"));
|
|
21
|
+
return dir;
|
|
22
|
+
}
|
|
23
|
+
function makeFailingCurlDir() {
|
|
24
|
+
const dir = makeTmpDir();
|
|
25
|
+
fs.writeFileSync(path.join(dir, "curl"), "#!/bin/sh\necho 'fake curl failure' >&2\nexit 1\n");
|
|
26
|
+
fs.chmodSync(path.join(dir, "curl"), 0o755);
|
|
27
|
+
return dir;
|
|
28
|
+
}
|
|
29
|
+
function makeRipgrepTarball() {
|
|
30
|
+
const root = makeTmpDir();
|
|
31
|
+
const packageDir = path.join(root, "ripgrep-14.1.1-x86_64-unknown-linux-musl");
|
|
32
|
+
fs.mkdirSync(packageDir, { recursive: true });
|
|
33
|
+
fs.writeFileSync(path.join(packageDir, "rg"), "#!/bin/sh\necho 'ripgrep 14.1.1'\n");
|
|
34
|
+
fs.chmodSync(path.join(packageDir, "rg"), 0o755);
|
|
35
|
+
const tarballPath = path.join(root, "ripgrep.tar.gz");
|
|
36
|
+
const result = spawnSync("tar", ["czf", tarballPath, "-C", root, path.basename(packageDir)], { encoding: "utf8" });
|
|
37
|
+
if (result.status !== 0) {
|
|
38
|
+
throw new Error(result.stderr || result.error?.message || "Failed to create ripgrep tarball");
|
|
39
|
+
}
|
|
40
|
+
return tarballPath;
|
|
41
|
+
}
|
|
42
|
+
afterEach(() => {
|
|
43
|
+
for (const dir of tmpDirs) {
|
|
44
|
+
fs.rmSync(dir, { recursive: true, force: true });
|
|
45
|
+
}
|
|
46
|
+
tmpDirs = [];
|
|
47
|
+
});
|
|
48
|
+
// ── ensureRg – already available ────────────────────────────────────────────
|
|
49
|
+
describe("ensureRg", () => {
|
|
50
|
+
test("returns existing rg when already available in binDir", async () => {
|
|
51
|
+
// Create a fake rg binary so resolveRg finds it
|
|
52
|
+
const binDir = makeTmpDir();
|
|
53
|
+
const rgPath = path.join(binDir, "rg");
|
|
54
|
+
fs.writeFileSync(rgPath, "#!/bin/sh\necho 'ripgrep 14.1.1'\n");
|
|
55
|
+
fs.chmodSync(rgPath, 0o755);
|
|
56
|
+
// We need to isolate PATH so only our binDir is searched
|
|
57
|
+
const origPath = process.env.PATH;
|
|
58
|
+
const origXdgCache = process.env.XDG_CACHE_HOME;
|
|
59
|
+
const origHome = process.env.HOME;
|
|
60
|
+
process.env.PATH = "";
|
|
61
|
+
process.env.XDG_CACHE_HOME = makeTmpDir();
|
|
62
|
+
try {
|
|
63
|
+
const { ensureRg } = await import("../src/setup/ripgrep-install");
|
|
64
|
+
const result = ensureRg(binDir);
|
|
65
|
+
expect(result.rgPath).toBe(rgPath);
|
|
66
|
+
expect(result.installed).toBe(false);
|
|
67
|
+
}
|
|
68
|
+
finally {
|
|
69
|
+
process.env.PATH = origPath;
|
|
70
|
+
if (origXdgCache === undefined)
|
|
71
|
+
delete process.env.XDG_CACHE_HOME;
|
|
72
|
+
else
|
|
73
|
+
process.env.XDG_CACHE_HOME = origXdgCache;
|
|
74
|
+
if (origHome === undefined)
|
|
75
|
+
delete process.env.HOME;
|
|
76
|
+
else
|
|
77
|
+
process.env.HOME = origHome;
|
|
78
|
+
}
|
|
79
|
+
});
|
|
80
|
+
});
|
|
81
|
+
// ── getRgPlatformTarget (tested via ensureRg behavior) ──────────────────────
|
|
82
|
+
describe("platform detection", () => {
|
|
83
|
+
// We can test this indirectly: ensureRg will throw for unsupported platforms
|
|
84
|
+
// We test current platform should be supported (we're running on linux/x64 or similar)
|
|
85
|
+
test("current platform is recognized (does not throw unsupported)", async () => {
|
|
86
|
+
const binDir = makeTmpDir();
|
|
87
|
+
const origPath = process.env.PATH;
|
|
88
|
+
const origXdgCache = process.env.XDG_CACHE_HOME;
|
|
89
|
+
process.env.PATH = makeFailingCurlDir();
|
|
90
|
+
process.env.XDG_CACHE_HOME = makeTmpDir();
|
|
91
|
+
try {
|
|
92
|
+
const { ensureRg } = await import("../src/setup/ripgrep-install");
|
|
93
|
+
// This will try to actually download, so we expect a network error or curl error,
|
|
94
|
+
// NOT an "Unsupported platform" error.
|
|
95
|
+
try {
|
|
96
|
+
ensureRg(binDir);
|
|
97
|
+
}
|
|
98
|
+
catch (err) {
|
|
99
|
+
const message = err.message;
|
|
100
|
+
// Should NOT be the unsupported platform error
|
|
101
|
+
expect(message).not.toContain("Unsupported platform");
|
|
102
|
+
// It should be a download/extraction error since we're not actually downloading
|
|
103
|
+
expect(message.includes("Failed to download") ||
|
|
104
|
+
message.includes("Failed to extract") ||
|
|
105
|
+
message.includes("not found at")).toBe(true);
|
|
106
|
+
}
|
|
107
|
+
}
|
|
108
|
+
finally {
|
|
109
|
+
process.env.PATH = origPath;
|
|
110
|
+
if (origXdgCache === undefined)
|
|
111
|
+
delete process.env.XDG_CACHE_HOME;
|
|
112
|
+
else
|
|
113
|
+
process.env.XDG_CACHE_HOME = origXdgCache;
|
|
114
|
+
}
|
|
115
|
+
});
|
|
116
|
+
});
|
|
117
|
+
// ── getRgVersion (tested via ensureRg result) ───────────────────────────────
|
|
118
|
+
describe("getRgVersion", () => {
|
|
119
|
+
test("extracts version from rg binary output", async () => {
|
|
120
|
+
const binDir = makeTmpDir();
|
|
121
|
+
const rgPath = path.join(binDir, "rg");
|
|
122
|
+
// Create a script that mimics rg --version output
|
|
123
|
+
fs.writeFileSync(rgPath, '#!/bin/sh\necho "ripgrep 14.1.1 (rev abc123)"\n');
|
|
124
|
+
fs.chmodSync(rgPath, 0o755);
|
|
125
|
+
const origPath = process.env.PATH;
|
|
126
|
+
const origXdgCache = process.env.XDG_CACHE_HOME;
|
|
127
|
+
process.env.PATH = "";
|
|
128
|
+
process.env.XDG_CACHE_HOME = makeTmpDir();
|
|
129
|
+
try {
|
|
130
|
+
const { ensureRg } = await import("../src/setup/ripgrep-install");
|
|
131
|
+
const result = ensureRg(binDir);
|
|
132
|
+
expect(result.version).toBe("14.1.1");
|
|
133
|
+
expect(result.installed).toBe(false);
|
|
134
|
+
}
|
|
135
|
+
finally {
|
|
136
|
+
process.env.PATH = origPath;
|
|
137
|
+
if (origXdgCache === undefined)
|
|
138
|
+
delete process.env.XDG_CACHE_HOME;
|
|
139
|
+
else
|
|
140
|
+
process.env.XDG_CACHE_HOME = origXdgCache;
|
|
141
|
+
}
|
|
142
|
+
});
|
|
143
|
+
test("returns 'unknown' when rg binary does not output version format", async () => {
|
|
144
|
+
const binDir = makeTmpDir();
|
|
145
|
+
const rgPath = path.join(binDir, "rg");
|
|
146
|
+
fs.writeFileSync(rgPath, '#!/bin/sh\necho "something else"\n');
|
|
147
|
+
fs.chmodSync(rgPath, 0o755);
|
|
148
|
+
const origPath = process.env.PATH;
|
|
149
|
+
const origXdgCache = process.env.XDG_CACHE_HOME;
|
|
150
|
+
process.env.PATH = "";
|
|
151
|
+
process.env.XDG_CACHE_HOME = makeTmpDir();
|
|
152
|
+
try {
|
|
153
|
+
const { ensureRg } = await import("../src/setup/ripgrep-install");
|
|
154
|
+
const result = ensureRg(binDir);
|
|
155
|
+
expect(result.version).toBe("unknown");
|
|
156
|
+
}
|
|
157
|
+
finally {
|
|
158
|
+
process.env.PATH = origPath;
|
|
159
|
+
if (origXdgCache === undefined)
|
|
160
|
+
delete process.env.XDG_CACHE_HOME;
|
|
161
|
+
else
|
|
162
|
+
process.env.XDG_CACHE_HOME = origXdgCache;
|
|
163
|
+
}
|
|
164
|
+
});
|
|
165
|
+
});
|
|
166
|
+
// ── EnsureRgResult shape ────────────────────────────────────────────────────
|
|
167
|
+
describe("EnsureRgResult", () => {
|
|
168
|
+
test("result has correct shape for existing binary", async () => {
|
|
169
|
+
const binDir = makeTmpDir();
|
|
170
|
+
const rgPath = path.join(binDir, "rg");
|
|
171
|
+
fs.writeFileSync(rgPath, '#!/bin/sh\necho "ripgrep 14.0.0"\n');
|
|
172
|
+
fs.chmodSync(rgPath, 0o755);
|
|
173
|
+
const origPath = process.env.PATH;
|
|
174
|
+
const origXdgCache = process.env.XDG_CACHE_HOME;
|
|
175
|
+
process.env.PATH = "";
|
|
176
|
+
process.env.XDG_CACHE_HOME = makeTmpDir();
|
|
177
|
+
try {
|
|
178
|
+
const { ensureRg } = await import("../src/setup/ripgrep-install");
|
|
179
|
+
const result = ensureRg(binDir);
|
|
180
|
+
expect(typeof result.rgPath).toBe("string");
|
|
181
|
+
expect(typeof result.installed).toBe("boolean");
|
|
182
|
+
expect(typeof result.version).toBe("string");
|
|
183
|
+
}
|
|
184
|
+
finally {
|
|
185
|
+
process.env.PATH = origPath;
|
|
186
|
+
if (origXdgCache === undefined)
|
|
187
|
+
delete process.env.XDG_CACHE_HOME;
|
|
188
|
+
else
|
|
189
|
+
process.env.XDG_CACHE_HOME = origXdgCache;
|
|
190
|
+
}
|
|
191
|
+
});
|
|
192
|
+
});
|
|
193
|
+
// ── Download error handling ─────────────────────────────────────────────────
|
|
194
|
+
describe("download error handling", () => {
|
|
195
|
+
test("creates binDir if it does not exist", async () => {
|
|
196
|
+
const parentDir = makeTmpDir();
|
|
197
|
+
const binDir = path.join(parentDir, "nested", "bin");
|
|
198
|
+
const rgPath = path.join(binDir, "rg");
|
|
199
|
+
// Pre-create an rg binary so ensureRg finds it and doesn't try to download
|
|
200
|
+
fs.mkdirSync(binDir, { recursive: true });
|
|
201
|
+
fs.writeFileSync(rgPath, '#!/bin/sh\necho "ripgrep 14.1.1"\n');
|
|
202
|
+
fs.chmodSync(rgPath, 0o755);
|
|
203
|
+
const origPath = process.env.PATH;
|
|
204
|
+
const origXdgCache = process.env.XDG_CACHE_HOME;
|
|
205
|
+
process.env.PATH = "";
|
|
206
|
+
process.env.XDG_CACHE_HOME = makeTmpDir();
|
|
207
|
+
try {
|
|
208
|
+
const { ensureRg } = await import("../src/setup/ripgrep-install");
|
|
209
|
+
const result = ensureRg(binDir);
|
|
210
|
+
expect(result.rgPath).toBe(rgPath);
|
|
211
|
+
expect(result.installed).toBe(false);
|
|
212
|
+
}
|
|
213
|
+
finally {
|
|
214
|
+
process.env.PATH = origPath;
|
|
215
|
+
if (origXdgCache === undefined)
|
|
216
|
+
delete process.env.XDG_CACHE_HOME;
|
|
217
|
+
else
|
|
218
|
+
process.env.XDG_CACHE_HOME = origXdgCache;
|
|
219
|
+
}
|
|
220
|
+
});
|
|
221
|
+
test("ensureRg returns installed=true when it installs a new binary", async () => {
|
|
222
|
+
const binDir = makeTmpDir();
|
|
223
|
+
const origPath = process.env.PATH;
|
|
224
|
+
const origXdgCache = process.env.XDG_CACHE_HOME;
|
|
225
|
+
const origFakeCurlSource = process.env.FAKE_CURL_SOURCE;
|
|
226
|
+
process.env.XDG_CACHE_HOME = makeTmpDir();
|
|
227
|
+
process.env.FAKE_CURL_SOURCE = makeRipgrepTarball();
|
|
228
|
+
process.env.PATH = makeToolchainDir();
|
|
229
|
+
try {
|
|
230
|
+
const { ensureRg } = await import("../src/setup/ripgrep-install");
|
|
231
|
+
const rgInBin = path.join(binDir, "rg");
|
|
232
|
+
if (fs.existsSync(rgInBin))
|
|
233
|
+
fs.unlinkSync(rgInBin);
|
|
234
|
+
const result = ensureRg(binDir);
|
|
235
|
+
expect(result.installed).toBe(true);
|
|
236
|
+
expect(result.version).toBe("14.1.1");
|
|
237
|
+
expect(fs.existsSync(result.rgPath)).toBe(true);
|
|
238
|
+
}
|
|
239
|
+
finally {
|
|
240
|
+
process.env.PATH = origPath;
|
|
241
|
+
if (origXdgCache === undefined)
|
|
242
|
+
delete process.env.XDG_CACHE_HOME;
|
|
243
|
+
else
|
|
244
|
+
process.env.XDG_CACHE_HOME = origXdgCache;
|
|
245
|
+
if (origFakeCurlSource === undefined)
|
|
246
|
+
delete process.env.FAKE_CURL_SOURCE;
|
|
247
|
+
else
|
|
248
|
+
process.env.FAKE_CURL_SOURCE = origFakeCurlSource;
|
|
249
|
+
}
|
|
250
|
+
});
|
|
251
|
+
});
|
|
@@ -0,0 +1,108 @@
|
|
|
1
|
+
import { afterAll, afterEach, describe, expect, test } from "bun:test";
|
|
2
|
+
import fs from "node:fs";
|
|
3
|
+
import os from "node:os";
|
|
4
|
+
import path from "node:path";
|
|
5
|
+
import { isRgAvailable, resolveRg } from "../src/setup/ripgrep-resolve";
|
|
6
|
+
// ── Helpers ─────────────────────────────────────────────────────────────────
|
|
7
|
+
const tempDirs = [];
|
|
8
|
+
function makeTempDir() {
|
|
9
|
+
const dir = fs.mkdtempSync(path.join(os.tmpdir(), "akm-rg-"));
|
|
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 origPath = process.env.PATH;
|
|
19
|
+
const origXdgCacheHome = process.env.XDG_CACHE_HOME;
|
|
20
|
+
const origHome = process.env.HOME;
|
|
21
|
+
afterEach(() => {
|
|
22
|
+
if (origPath === undefined) {
|
|
23
|
+
delete process.env.PATH;
|
|
24
|
+
}
|
|
25
|
+
else {
|
|
26
|
+
process.env.PATH = origPath;
|
|
27
|
+
}
|
|
28
|
+
if (origXdgCacheHome === undefined) {
|
|
29
|
+
delete process.env.XDG_CACHE_HOME;
|
|
30
|
+
}
|
|
31
|
+
else {
|
|
32
|
+
process.env.XDG_CACHE_HOME = origXdgCacheHome;
|
|
33
|
+
}
|
|
34
|
+
if (origHome === undefined) {
|
|
35
|
+
delete process.env.HOME;
|
|
36
|
+
}
|
|
37
|
+
else {
|
|
38
|
+
process.env.HOME = origHome;
|
|
39
|
+
}
|
|
40
|
+
});
|
|
41
|
+
/** Isolate cache so getBinDir() never finds a real rg binary. */
|
|
42
|
+
function isolateCache() {
|
|
43
|
+
process.env.XDG_CACHE_HOME = makeTempDir();
|
|
44
|
+
}
|
|
45
|
+
// ── resolveRg ───────────────────────────────────────────────────────────────
|
|
46
|
+
describe("resolveRg", () => {
|
|
47
|
+
test("finds rg in provided bin directory", () => {
|
|
48
|
+
const binDir = makeTempDir();
|
|
49
|
+
const rgPath = path.join(binDir, "rg");
|
|
50
|
+
fs.writeFileSync(rgPath, "#!/bin/sh\necho rg\n");
|
|
51
|
+
fs.chmodSync(rgPath, 0o755);
|
|
52
|
+
const result = resolveRg(binDir);
|
|
53
|
+
expect(result).toBe(rgPath);
|
|
54
|
+
});
|
|
55
|
+
test("falls back to system PATH", () => {
|
|
56
|
+
const fakeBinDir = makeTempDir();
|
|
57
|
+
const fakeRg = path.join(fakeBinDir, "rg");
|
|
58
|
+
fs.writeFileSync(fakeRg, "#!/bin/sh\necho rg\n");
|
|
59
|
+
fs.chmodSync(fakeRg, 0o755);
|
|
60
|
+
// Put our fake bin dir at the front of PATH
|
|
61
|
+
process.env.PATH = `${fakeBinDir}${path.delimiter}${origPath}`;
|
|
62
|
+
// No bin dir provided -- should find from PATH
|
|
63
|
+
const result = resolveRg();
|
|
64
|
+
expect(result).toBeTruthy();
|
|
65
|
+
});
|
|
66
|
+
test("returns null when not found anywhere", () => {
|
|
67
|
+
const emptyDir = makeTempDir();
|
|
68
|
+
process.env.PATH = "";
|
|
69
|
+
isolateCache();
|
|
70
|
+
const result = resolveRg(emptyDir);
|
|
71
|
+
expect(result).toBeNull();
|
|
72
|
+
});
|
|
73
|
+
test("skips non-executable file in bin dir", () => {
|
|
74
|
+
const binDir = makeTempDir();
|
|
75
|
+
// Create an rg file that is NOT executable
|
|
76
|
+
const rgPath = path.join(binDir, "rg");
|
|
77
|
+
fs.writeFileSync(rgPath, "not executable");
|
|
78
|
+
fs.chmodSync(rgPath, 0o644);
|
|
79
|
+
process.env.PATH = "";
|
|
80
|
+
isolateCache();
|
|
81
|
+
const result = resolveRg(binDir);
|
|
82
|
+
expect(result).toBeNull();
|
|
83
|
+
});
|
|
84
|
+
});
|
|
85
|
+
// ── isRgAvailable ───────────────────────────────────────────────────────────
|
|
86
|
+
describe("isRgAvailable", () => {
|
|
87
|
+
test("returns true when resolveRg finds a binary", () => {
|
|
88
|
+
const binDir = makeTempDir();
|
|
89
|
+
const rgPath = path.join(binDir, "rg");
|
|
90
|
+
fs.writeFileSync(rgPath, "#!/bin/sh\necho rg\n");
|
|
91
|
+
fs.chmodSync(rgPath, 0o755);
|
|
92
|
+
expect(isRgAvailable(binDir)).toBe(true);
|
|
93
|
+
});
|
|
94
|
+
test("returns false when resolveRg finds nothing", () => {
|
|
95
|
+
const emptyDir = makeTempDir();
|
|
96
|
+
process.env.PATH = "";
|
|
97
|
+
isolateCache();
|
|
98
|
+
expect(isRgAvailable(emptyDir)).toBe(false);
|
|
99
|
+
});
|
|
100
|
+
test("boolean result matches resolveRg truthiness", () => {
|
|
101
|
+
const binDir = makeTempDir();
|
|
102
|
+
process.env.PATH = "";
|
|
103
|
+
isolateCache();
|
|
104
|
+
const resolved = resolveRg(binDir);
|
|
105
|
+
const available = isRgAvailable(binDir);
|
|
106
|
+
expect(available).toBe(resolved !== null);
|
|
107
|
+
});
|
|
108
|
+
});
|
|
@@ -0,0 +1,163 @@
|
|
|
1
|
+
import { afterAll, expect, test } from "bun:test";
|
|
2
|
+
import fs from "node:fs";
|
|
3
|
+
import os from "node:os";
|
|
4
|
+
import path from "node:path";
|
|
5
|
+
import { isRgAvailable, resolveRg } from "../src/setup/ripgrep-resolve";
|
|
6
|
+
const createdTmpDirs = [];
|
|
7
|
+
function expectDefined(value) {
|
|
8
|
+
expect(value).toBeDefined();
|
|
9
|
+
if (value === undefined || value === null) {
|
|
10
|
+
throw new Error("Expected value to be defined");
|
|
11
|
+
}
|
|
12
|
+
return value;
|
|
13
|
+
}
|
|
14
|
+
function tmpDir() {
|
|
15
|
+
const dir = fs.mkdtempSync(path.join(os.tmpdir(), "akm-rg-"));
|
|
16
|
+
createdTmpDirs.push(dir);
|
|
17
|
+
return dir;
|
|
18
|
+
}
|
|
19
|
+
afterAll(() => {
|
|
20
|
+
for (const dir of createdTmpDirs) {
|
|
21
|
+
fs.rmSync(dir, { recursive: true, force: true });
|
|
22
|
+
}
|
|
23
|
+
});
|
|
24
|
+
function writeFile(filePath, content = "") {
|
|
25
|
+
fs.mkdirSync(path.dirname(filePath), { recursive: true });
|
|
26
|
+
fs.writeFileSync(filePath, content);
|
|
27
|
+
}
|
|
28
|
+
// ── resolveRg ───────────────────────────────────────────────────────────────
|
|
29
|
+
test("resolveRg finds system ripgrep on PATH", () => {
|
|
30
|
+
const originalPath = process.env.PATH;
|
|
31
|
+
const stashDir = tmpDir();
|
|
32
|
+
const binDir = path.join(stashDir, "bin");
|
|
33
|
+
fs.mkdirSync(binDir, { recursive: true });
|
|
34
|
+
// Create a fake rg binary on PATH so the test does not depend on the host environment
|
|
35
|
+
const rgName = process.platform === "win32" ? "rg.cmd" : "rg";
|
|
36
|
+
const fakeRg = path.join(binDir, rgName);
|
|
37
|
+
const scriptContent = process.platform === "win32" ? "@echo off\r\necho fake rg\r\n" : "#!/bin/sh\necho fake rg\n";
|
|
38
|
+
fs.writeFileSync(fakeRg, scriptContent);
|
|
39
|
+
try {
|
|
40
|
+
// Make sure the fake rg is executable where that concept applies
|
|
41
|
+
if (process.platform !== "win32") {
|
|
42
|
+
fs.chmodSync(fakeRg, 0o755);
|
|
43
|
+
}
|
|
44
|
+
}
|
|
45
|
+
catch {
|
|
46
|
+
// Ignore chmod errors on platforms/filesystems that do not support it
|
|
47
|
+
}
|
|
48
|
+
// Prepend the fake rg directory to PATH for this test only
|
|
49
|
+
process.env.PATH = binDir + path.delimiter + (originalPath ?? "");
|
|
50
|
+
try {
|
|
51
|
+
const rg = resolveRg();
|
|
52
|
+
expect(expectDefined(rg)).toContain("rg");
|
|
53
|
+
}
|
|
54
|
+
finally {
|
|
55
|
+
process.env.PATH = originalPath;
|
|
56
|
+
}
|
|
57
|
+
});
|
|
58
|
+
test("resolveRg finds rg in provided bin directory", () => {
|
|
59
|
+
const binDir = tmpDir();
|
|
60
|
+
// Create a fake rg binary
|
|
61
|
+
const fakeRg = path.join(binDir, "rg");
|
|
62
|
+
fs.writeFileSync(fakeRg, "#!/bin/sh\necho fake rg\n");
|
|
63
|
+
fs.chmodSync(fakeRg, 0o755);
|
|
64
|
+
const rg = resolveRg(binDir);
|
|
65
|
+
expect(rg).toBe(fakeRg);
|
|
66
|
+
});
|
|
67
|
+
test("resolveRg skips non-executable files in bin dir", () => {
|
|
68
|
+
const binDir = tmpDir();
|
|
69
|
+
// Create a non-executable rg file
|
|
70
|
+
const fakeRg = path.join(binDir, "rg");
|
|
71
|
+
fs.writeFileSync(fakeRg, "not executable");
|
|
72
|
+
fs.chmodSync(fakeRg, 0o644);
|
|
73
|
+
const rg = resolveRg(binDir);
|
|
74
|
+
// Should fall through to system PATH
|
|
75
|
+
expect(rg).not.toBe(fakeRg);
|
|
76
|
+
});
|
|
77
|
+
// ── isRgAvailable ───────────────────────────────────────────────────────────
|
|
78
|
+
test("isRgAvailable returns true when rg is on PATH", () => {
|
|
79
|
+
const originalPath = process.env.PATH;
|
|
80
|
+
const stashDir = tmpDir();
|
|
81
|
+
const binDir = path.join(stashDir, "bin");
|
|
82
|
+
fs.mkdirSync(binDir, { recursive: true });
|
|
83
|
+
const rgName = process.platform === "win32" ? "rg.cmd" : "rg";
|
|
84
|
+
const fakeRg = path.join(binDir, rgName);
|
|
85
|
+
const scriptContent = process.platform === "win32" ? "@echo off\r\necho fake rg\r\n" : "#!/bin/sh\necho fake rg\n";
|
|
86
|
+
fs.writeFileSync(fakeRg, scriptContent);
|
|
87
|
+
if (process.platform !== "win32") {
|
|
88
|
+
fs.chmodSync(fakeRg, 0o755);
|
|
89
|
+
}
|
|
90
|
+
process.env.PATH = binDir + path.delimiter + (originalPath ?? "");
|
|
91
|
+
try {
|
|
92
|
+
expect(isRgAvailable()).toBe(true);
|
|
93
|
+
}
|
|
94
|
+
finally {
|
|
95
|
+
process.env.PATH = originalPath;
|
|
96
|
+
}
|
|
97
|
+
});
|
|
98
|
+
// ── Integration: indexed search pipeline ────────────────────────────────────
|
|
99
|
+
test("search pipeline returns ranked results when index exists", async () => {
|
|
100
|
+
const stashDir = tmpDir();
|
|
101
|
+
for (const sub of ["scripts", "skills", "commands", "agents"]) {
|
|
102
|
+
fs.mkdirSync(path.join(stashDir, sub), { recursive: true });
|
|
103
|
+
}
|
|
104
|
+
// Create scripts with .stash.json metadata
|
|
105
|
+
writeFile(path.join(stashDir, "scripts", "docker", "build.sh"), "#!/bin/bash\necho build\n");
|
|
106
|
+
writeFile(path.join(stashDir, "scripts", "docker", ".stash.json"), JSON.stringify({
|
|
107
|
+
entries: [
|
|
108
|
+
{
|
|
109
|
+
name: "docker-build",
|
|
110
|
+
type: "script",
|
|
111
|
+
description: "build docker images",
|
|
112
|
+
tags: ["docker", "container"],
|
|
113
|
+
filename: "build.sh",
|
|
114
|
+
},
|
|
115
|
+
],
|
|
116
|
+
}));
|
|
117
|
+
writeFile(path.join(stashDir, "scripts", "git", "diff.sh"), "#!/bin/bash\necho diff\n");
|
|
118
|
+
writeFile(path.join(stashDir, "scripts", "git", ".stash.json"), JSON.stringify({
|
|
119
|
+
entries: [
|
|
120
|
+
{
|
|
121
|
+
name: "git-diff",
|
|
122
|
+
type: "script",
|
|
123
|
+
description: "summarize git changes",
|
|
124
|
+
tags: ["git", "diff"],
|
|
125
|
+
filename: "diff.sh",
|
|
126
|
+
},
|
|
127
|
+
],
|
|
128
|
+
}));
|
|
129
|
+
// Isolation: ensure index cache and config are written to temp directories
|
|
130
|
+
const oldXdgCacheHome = process.env.XDG_CACHE_HOME;
|
|
131
|
+
const oldXdgConfigHome = process.env.XDG_CONFIG_HOME;
|
|
132
|
+
const oldAkmStashDir = process.env.AKM_STASH_DIR;
|
|
133
|
+
const tempCacheDir = tmpDir();
|
|
134
|
+
const tempConfigDir = tmpDir();
|
|
135
|
+
process.env.XDG_CACHE_HOME = tempCacheDir;
|
|
136
|
+
process.env.XDG_CONFIG_HOME = tempConfigDir;
|
|
137
|
+
try {
|
|
138
|
+
// Build index
|
|
139
|
+
process.env.AKM_STASH_DIR = stashDir;
|
|
140
|
+
const { akmIndex } = await import("../src/indexer/indexer");
|
|
141
|
+
await akmIndex({ stashDir });
|
|
142
|
+
// Search — TF-IDF should rank docker-related results first
|
|
143
|
+
const { akmSearch } = await import("../src/commands/search");
|
|
144
|
+
const result = await akmSearch({ query: "docker", type: "any" });
|
|
145
|
+
expect(result.hits.length).toBeGreaterThan(0);
|
|
146
|
+
// Docker-related result should be ranked first
|
|
147
|
+
expect(result.hits[0].name).toContain("docker");
|
|
148
|
+
}
|
|
149
|
+
finally {
|
|
150
|
+
if (oldXdgCacheHome === undefined)
|
|
151
|
+
delete process.env.XDG_CACHE_HOME;
|
|
152
|
+
else
|
|
153
|
+
process.env.XDG_CACHE_HOME = oldXdgCacheHome;
|
|
154
|
+
if (oldXdgConfigHome === undefined)
|
|
155
|
+
delete process.env.XDG_CONFIG_HOME;
|
|
156
|
+
else
|
|
157
|
+
process.env.XDG_CONFIG_HOME = oldXdgConfigHome;
|
|
158
|
+
if (oldAkmStashDir === undefined)
|
|
159
|
+
delete process.env.AKM_STASH_DIR;
|
|
160
|
+
else
|
|
161
|
+
process.env.AKM_STASH_DIR = oldAkmStashDir;
|
|
162
|
+
}
|
|
163
|
+
});
|
|
@@ -0,0 +1,94 @@
|
|
|
1
|
+
import { afterEach, 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
|
+
const CLI = path.join(__dirname, "..", "src", "cli.ts");
|
|
7
|
+
const tempDirs = [];
|
|
8
|
+
function makeTempDir(prefix) {
|
|
9
|
+
const dir = fs.mkdtempSync(path.join(os.tmpdir(), prefix));
|
|
10
|
+
tempDirs.push(dir);
|
|
11
|
+
return dir;
|
|
12
|
+
}
|
|
13
|
+
afterEach(() => {
|
|
14
|
+
for (const dir of tempDirs.splice(0)) {
|
|
15
|
+
fs.rmSync(dir, { recursive: true, force: true });
|
|
16
|
+
}
|
|
17
|
+
});
|
|
18
|
+
function runCli(args, stashDir) {
|
|
19
|
+
const xdgCache = makeTempDir("akm-save-cache-");
|
|
20
|
+
const xdgConfig = makeTempDir("akm-save-cfg-");
|
|
21
|
+
return spawnSync("bun", [CLI, ...args], {
|
|
22
|
+
encoding: "utf8",
|
|
23
|
+
timeout: 30_000,
|
|
24
|
+
env: {
|
|
25
|
+
...process.env,
|
|
26
|
+
AKM_STASH_DIR: stashDir,
|
|
27
|
+
XDG_CACHE_HOME: xdgCache,
|
|
28
|
+
XDG_CONFIG_HOME: xdgConfig,
|
|
29
|
+
},
|
|
30
|
+
});
|
|
31
|
+
}
|
|
32
|
+
function parseSaveOutput(stdout) {
|
|
33
|
+
return JSON.parse(stdout.trim());
|
|
34
|
+
}
|
|
35
|
+
/** Initialise a bare git repo in `dir` so akm save can commit. */
|
|
36
|
+
function initGitRepo(dir) {
|
|
37
|
+
fs.mkdirSync(dir, { recursive: true });
|
|
38
|
+
spawnSync("git", ["init", dir], { encoding: "utf8" });
|
|
39
|
+
spawnSync("git", ["-C", dir, "config", "commit.gpgsign", "false"], { encoding: "utf8" });
|
|
40
|
+
}
|
|
41
|
+
describe("akm save", () => {
|
|
42
|
+
test("returns skipped when stash is not a git repo", () => {
|
|
43
|
+
const stashDir = makeTempDir("akm-save-nongit-");
|
|
44
|
+
const result = runCli(["save"], stashDir);
|
|
45
|
+
expect(result.status).toBe(0);
|
|
46
|
+
const json = parseSaveOutput(result.stdout);
|
|
47
|
+
expect(json.skipped).toBe(true);
|
|
48
|
+
expect(json.committed).toBe(false);
|
|
49
|
+
expect(json.pushed).toBe(false);
|
|
50
|
+
});
|
|
51
|
+
test("reports nothing to commit on a clean git repo", () => {
|
|
52
|
+
const stashDir = makeTempDir("akm-save-clean-");
|
|
53
|
+
initGitRepo(stashDir);
|
|
54
|
+
// Create an initial commit so the repo is not bare
|
|
55
|
+
const f = path.join(stashDir, "README.md");
|
|
56
|
+
fs.writeFileSync(f, "hello");
|
|
57
|
+
spawnSync("git", ["-C", stashDir, "add", "-A"], { encoding: "utf8" });
|
|
58
|
+
spawnSync("git", ["-C", stashDir, "-c", "user.name=test", "-c", "user.email=t@t", "commit", "-m", "init"], {
|
|
59
|
+
encoding: "utf8",
|
|
60
|
+
});
|
|
61
|
+
const result = runCli(["save"], stashDir);
|
|
62
|
+
expect(result.status).toBe(0);
|
|
63
|
+
const json = parseSaveOutput(result.stdout);
|
|
64
|
+
expect(json.committed).toBe(false);
|
|
65
|
+
expect(json.skipped).toBe(false);
|
|
66
|
+
expect(json.output).toContain("nothing to commit");
|
|
67
|
+
});
|
|
68
|
+
test("commits changes in a git repo with no remote", () => {
|
|
69
|
+
const stashDir = makeTempDir("akm-save-commit-");
|
|
70
|
+
initGitRepo(stashDir);
|
|
71
|
+
// Write a file so there's something to commit
|
|
72
|
+
fs.writeFileSync(path.join(stashDir, "skill.md"), "# Test");
|
|
73
|
+
const result = runCli(["save", "-m", "test commit"], stashDir);
|
|
74
|
+
expect(result.status).toBe(0);
|
|
75
|
+
const json = parseSaveOutput(result.stdout);
|
|
76
|
+
expect(json.committed).toBe(true);
|
|
77
|
+
expect(json.pushed).toBe(false);
|
|
78
|
+
expect(json.skipped).toBe(false);
|
|
79
|
+
// Verify the commit actually landed
|
|
80
|
+
const log = spawnSync("git", ["-C", stashDir, "log", "--oneline"], { encoding: "utf8" });
|
|
81
|
+
expect(log.stdout).toContain("test commit");
|
|
82
|
+
});
|
|
83
|
+
test("uses timestamp message when -m is omitted", () => {
|
|
84
|
+
const stashDir = makeTempDir("akm-save-ts-");
|
|
85
|
+
initGitRepo(stashDir);
|
|
86
|
+
fs.writeFileSync(path.join(stashDir, "skill.md"), "# Test");
|
|
87
|
+
const result = runCli(["save"], stashDir);
|
|
88
|
+
expect(result.status).toBe(0);
|
|
89
|
+
const json = parseSaveOutput(result.stdout);
|
|
90
|
+
expect(json.committed).toBe(true);
|
|
91
|
+
const log = spawnSync("git", ["-C", stashDir, "log", "--oneline"], { encoding: "utf8" });
|
|
92
|
+
expect(log.stdout).toContain("akm save");
|
|
93
|
+
});
|
|
94
|
+
});
|