mirror of
https://github.com/Kaelio/ktx.git
synced 2026-06-13 08:15:14 +02:00
* refactor(workspace): relocate @ktx/llm source into packages/cli/src/llm * refactor(workspace): rewrite @ktx/llm imports to relative paths * refactor(workspace): fold internal packages into cli * chore(workspace): gate dead-code with knip production mode Turn on production-mode knip plus an autofix run in pre-commit and the `pnpm dead-code` script, document the `/** @internal */` convention for test-only exports in AGENTS.md, annotate test-only exports across the CLI with that JSDoc, and drop dead exports/wrappers the new gate surfaced (e.g. `cli-project.ts`, `lookerRuntimeSourceToFileAdapterSource`, `createLocalScanEnrichmentProvidersFromConfig`, `PGLITE_OWNER_PROCESS_BACKEND_CAPABILITIES`, stale type re-exports). Replace the loose `ignoreIssues` allowlist in `knip.json` with explicit production entries so cross-package barrel leaks are caught. * refactor(cli): delete internal barrel index.ts files The 34 `index.ts` re-export barrels inside `packages/cli/src/` were holdovers from the pre-fold multi-workspace structure. Post-fold-in they served no production purpose: external consumers go through the single package main entry, and in-repo callers mostly imported through them only because the path was short. Internally, knip flagged most barrel re-exports as production-dead (only reached via tests). This change: - Deletes every internal barrel except `packages/cli/src/index.ts` (the published package entry). - Rewrites ~270 source/test files to import each name directly from the file that defines it. - Moves `tools/warehouse-verification/index.ts` to `create-warehouse-verification-tools.ts` (the function it defined locally) and updates its single consumer. - Renames `search/backend-conformance.ts` → `.test-utils.ts` to match the existing test-helper file convention. - Deletes 13 dead test-only chains (dbt-descriptions/*, live-database/extracted-schema, live-database/structural-sync, relationship-* feedback/review chain) plus their tests and a cascading orphan integration test. - Updates test mocks that pointed at deleted barrel paths (notion-client, connector barrels in scan/local-scan-connectors tests) to mock the source files instead. - Points the maintainer benchmark script (`scripts/relationship-benchmark-report.mjs`) at source files instead of `dist/context/scan/index.js`. - Drops the barrel `!` entries from `knip.json`; adds explicit production entries only for the benchmark code reached via dist by the maintainer script. Net: 413 files changed, ~1.2k insertions, ~9.4k deletions. `pnpm run dead-code` (Biome + knip default + knip production) and `pnpm run type-check` are clean; 2277 tests pass. * refactor(workspace): rename @ktx/cli to @kaelio/ktx and pack it directly Promote the CLI workspace package to the public name `@kaelio/ktx` and drop the separate `scripts/build-public-npm-package.mjs` wrapper. The CLI package is now publishable in place (`publishConfig.access: public`, `provenance: true`), so artifact packing uses `pnpm pack` against `packages/cli/` instead of assembling a parallel package tree. Updates all workspace filter invocations, docs, tests, and release readiness checks to reference the new package name, and folds the tarball-name helper into `scripts/public-npm-release-metadata.mjs`. * docs: align "agent clients" and "data agents" terminology Replace "client agents" with "agent clients" and "database agents" with "data agents" across AGENTS.md, README.md, the docs-site copy, and the matching setup-agents test description, matching the canonical vocabulary in docs/terminology.md. Also moves packages/cli/tsconfig.json's tsBuildInfoFile from node_modules/.cache/ to dist/.tsbuildinfo so incremental builds survive node_modules reinstalls. * refactor(release): single source of truth for package version Make packages/cli/package.json the single source of truth for the @kaelio/ktx version. publicNpmPackageVersion() now reads it directly, so artifact filenames, release-readiness checks, and the Python wheel version all derive from one field. The duplicate release-policy.json.publicNpmPackageVersion is removed. Previously the two fields could drift: tarballs were named kaelio-ktx-0.4.1.tgz while internally containing @kaelio/ktx@0.0.0-private. - update-public-release-version.mjs rewrites both Python pyproject.toml files (ktx-daemon, ktx-sl) alongside the npm package.jsons, normalizing the version for PEP 440 (e.g. 0.1.0-rc.2 -> 0.1.0rc2). - semantic-release-config.cjs adds the two pyproject.toml files to @semantic-release/git assets so the release commit back to main carries every version source in lockstep. - The six "?? '0.0.0-private'" fallback literals across the CLI are replaced with "?? getKtxCliPackageInfo().version", and createDefaultKtxMcpServer makes its version arg required. - docs/release.md describes the actual commit-back model: the dev tree always reflects the most recent release; no sentinel pin to maintain. Verified: pnpm run artifacts:build now produces kaelio-ktx-0.4.1.tgz and kaelio_ktx-0.4.1-py3-none-any.whl with @kaelio/ktx@0.4.1 inside. Full type-check, dead-code, and 2287 vitests + 173 script tests pass. * refactor(cli): inject embedding provider resolution and detect sentence-transformers runtime Make resolveProjectEmbeddingProvider and runtimeIo injectable in ingest and scan command entrypoints so tests can stub them, and teach resolvePublicIngestRuntimeRequirements to flag the local-embeddings runtime feature when ktx.yaml selects sentence-transformers. * chore(cli): mark buildLocalStatsStatus and LocalStatsStatus as @internal Both symbols are consumed only by status-project.test.ts. Annotating with /** @internal */ keeps knip's production-mode check clean without changing runtime behavior. * fix(cli): use real package metadata in print-command-tree The stubbed package name embedded a forbidden product identifier that tripped the boundary check in CI. Read the metadata from package.json instead — keeps the rendered tree unchanged and removes a duplicate source of truth. * feat(cli): show embedding coverage in `ktx status`, drop duplicate disk counts Inline `(N embedded)` next to the Wiki scope counts and Semantic-layer source counts, computed with `SUM(embedding_json IS NOT NULL)` over `knowledge_pages` and `local_sl_sources`. Rename the "Knowledge" label to "Wiki" (canonical per `docs/terminology.md`) and rename the matching `localStats.knowledgePages` field to `localStats.wikiPages`. Drop `wiki=N md` and `semantic-layer=N yaml` from the Disk row — those duplicated the per-surface rows above. Disk now reports only actual byte usage (db, cache, raw-sources). The unused `wikiGlobalMarkdownCount` / `semanticLayerYamlCount` fields, the `isMarkdownEntry` / `isYamlEntry` helpers, and the `filter` arg on `summarizeDir` are removed.
865 lines
26 KiB
TypeScript
865 lines
26 KiB
TypeScript
import { mkdtemp, rm, writeFile } from 'node:fs/promises';
|
|
import { tmpdir } from 'node:os';
|
|
import { basename, join } from 'node:path';
|
|
import { afterEach, beforeEach, describe, expect, it } from 'vitest';
|
|
import {
|
|
formatDoctorReport,
|
|
runKtxDoctor,
|
|
runSetupDoctorChecks,
|
|
type DoctorCheck,
|
|
} from './doctor.js';
|
|
|
|
function makeIo() {
|
|
let stdout = '';
|
|
let stderr = '';
|
|
return {
|
|
io: {
|
|
stdout: {
|
|
write: (chunk: string) => {
|
|
stdout += chunk;
|
|
},
|
|
},
|
|
stderr: {
|
|
write: (chunk: string) => {
|
|
stderr += chunk;
|
|
},
|
|
},
|
|
},
|
|
stdout: () => stdout,
|
|
stderr: () => stderr,
|
|
};
|
|
}
|
|
|
|
describe('formatDoctorReport', () => {
|
|
it('shows the failing check and its fix in plain output', () => {
|
|
const checks: DoctorCheck[] = [
|
|
{ id: 'node', label: 'Node 22+', status: 'pass', detail: 'v22.16.0 ABI 127', group: 'toolchain' },
|
|
{
|
|
id: 'native-sqlite',
|
|
label: 'Native SQLite',
|
|
status: 'fail',
|
|
detail: 'Cannot load better-sqlite3',
|
|
fix: 'Run: pnpm run native:rebuild',
|
|
group: 'toolchain',
|
|
},
|
|
];
|
|
|
|
const output = formatDoctorReport({ title: 'KTX status', checks });
|
|
expect(output).toContain('KTX status');
|
|
expect(output).toContain('✗ Environment');
|
|
expect(output).toContain('1 of 2 need attention');
|
|
expect(output).toContain('✗ Native SQLite: Cannot load better-sqlite3');
|
|
expect(output).toContain('→ Run: pnpm run native:rebuild');
|
|
expect(output).toContain('1 issue to fix.');
|
|
});
|
|
|
|
it('lists what was checked when a group has all passing checks', () => {
|
|
const checks: DoctorCheck[] = [
|
|
{ id: 'node', label: 'Node 22+', status: 'pass', detail: 'v22.16.0', group: 'toolchain' },
|
|
{ id: 'pnpm', label: 'pnpm 10.20+', status: 'pass', detail: '10.28.0', group: 'toolchain' },
|
|
];
|
|
|
|
const output = formatDoctorReport({ title: 'KTX status', checks });
|
|
expect(output).toContain('✓ Environment');
|
|
expect(output).toContain('Node 22+ · pnpm 10.20+');
|
|
expect(output).not.toContain('v22.16.0');
|
|
expect(output).toContain('Everything ready.');
|
|
expect(output).toContain('ktx status --json');
|
|
expect(output).toContain('ktx sl');
|
|
expect(output).toContain('ktx wiki');
|
|
expect(output).not.toContain('ktx scan');
|
|
expect(output).not.toContain('ktx sl ask');
|
|
});
|
|
|
|
it('shows the underlying detail for a single-check group on the group line', () => {
|
|
const checks: DoctorCheck[] = [
|
|
{
|
|
id: 'semantic-search-embeddings',
|
|
label: 'Semantic search embeddings',
|
|
status: 'pass',
|
|
detail: 'openai/text-embedding-3-small (1536d) probe succeeded',
|
|
group: 'search',
|
|
},
|
|
];
|
|
|
|
const output = formatDoctorReport({ title: 'KTX status', checks });
|
|
expect(output).toContain('✓ Semantic search');
|
|
expect(output).toContain('openai/text-embedding-3-small (1536d) probe succeeded');
|
|
});
|
|
|
|
it('lists every check in verbose mode', () => {
|
|
const checks: DoctorCheck[] = [
|
|
{ id: 'node', label: 'Node 22+', status: 'pass', detail: 'v22.16.0', group: 'toolchain' },
|
|
];
|
|
|
|
const output = formatDoctorReport({ title: 'KTX status', checks }, { verbose: true });
|
|
expect(output).toContain('✓ Node 22+: v22.16.0');
|
|
});
|
|
});
|
|
|
|
describe('runSetupDoctorChecks', () => {
|
|
it('returns pass checks when injected commands and file checks succeed', async () => {
|
|
const checks = await runSetupDoctorChecks({
|
|
env: { PATH: '/bin' },
|
|
workspaceRoot: '/workspace/ktx',
|
|
execText: async (command, args) => {
|
|
if (command === 'pnpm' && args[0] === '--version') return '10.28.0';
|
|
if (command === 'corepack' && args[0] === '--version') return '0.32.0';
|
|
if (command === 'uv' && args[0] === '--version') return 'uv 0.9.5';
|
|
if (command === process.execPath && args.includes('--version')) return '@kaelio/ktx 0.0.0-private';
|
|
throw new Error(`${command} ${args.join(' ')}`);
|
|
},
|
|
pathExists: async () => true,
|
|
importBetterSqlite3: async () => ({ default: function Database() {} }),
|
|
});
|
|
|
|
expect(checks.map((check) => [check.id, check.status])).toEqual([
|
|
['node', 'pass'],
|
|
['pnpm', 'pass'],
|
|
['corepack', 'pass'],
|
|
['uv', 'pass'],
|
|
['native-sqlite', 'pass'],
|
|
['package-build', 'pass'],
|
|
['workspace-cli', 'pass'],
|
|
]);
|
|
});
|
|
|
|
it('returns exact fixes when setup checks fail', async () => {
|
|
const checks = await runSetupDoctorChecks({
|
|
env: {},
|
|
workspaceRoot: '/workspace/ktx',
|
|
execText: async (command) => {
|
|
throw new Error(`${command} not found`);
|
|
},
|
|
pathExists: async () => false,
|
|
importBetterSqlite3: async () => {
|
|
throw new Error('Cannot find module better-sqlite3');
|
|
},
|
|
});
|
|
|
|
expect(checks).toContainEqual({
|
|
id: 'pnpm',
|
|
label: 'pnpm 10.20+',
|
|
status: 'fail',
|
|
detail: 'pnpm not found',
|
|
fix: 'Run: corepack enable && corepack prepare pnpm@10.28.0 --activate',
|
|
group: 'toolchain',
|
|
});
|
|
expect(checks).toContainEqual({
|
|
id: 'package-build',
|
|
label: 'TypeScript package build',
|
|
status: 'fail',
|
|
detail: 'Missing packages/cli/dist/bin.js',
|
|
fix: 'Run: pnpm run build',
|
|
group: 'toolchain',
|
|
});
|
|
});
|
|
|
|
it('treats missing corepack as a warning so setup doctor can still pass', async () => {
|
|
const checks = await runSetupDoctorChecks({
|
|
env: { PATH: '/bin' },
|
|
workspaceRoot: '/workspace/ktx',
|
|
execText: async (command, args) => {
|
|
if (command === 'pnpm' && args[0] === '--version') return '10.28.0';
|
|
if (command === 'corepack' && args[0] === '--version') throw new Error('spawn corepack ENOENT');
|
|
if (command === 'uv' && args[0] === '--version') return 'uv 0.9.5';
|
|
if (command === process.execPath && args.includes('--version')) return '@kaelio/ktx 0.0.0-private';
|
|
throw new Error(`${command} ${args.join(' ')}`);
|
|
},
|
|
pathExists: async () => true,
|
|
importBetterSqlite3: async () => ({ default: function Database() {} }),
|
|
});
|
|
const testIo = makeIo();
|
|
|
|
await expect(
|
|
runKtxDoctor(
|
|
{ command: 'setup', outputMode: 'plain', inputMode: 'disabled', verbose: true },
|
|
testIo.io,
|
|
{ runSetupChecks: async () => checks },
|
|
),
|
|
).resolves.toBe(0);
|
|
|
|
expect(checks).toContainEqual({
|
|
id: 'corepack',
|
|
label: 'Corepack',
|
|
status: 'warn',
|
|
detail: 'spawn corepack ENOENT',
|
|
fix: 'Run: corepack enable',
|
|
group: 'toolchain',
|
|
});
|
|
expect(testIo.stdout()).toContain('⚠ Corepack: spawn corepack ENOENT');
|
|
expect(testIo.stderr()).toBe('');
|
|
});
|
|
});
|
|
|
|
describe('runKtxDoctor', () => {
|
|
let tempDir: string;
|
|
|
|
beforeEach(async () => {
|
|
tempDir = await mkdtemp(join(tmpdir(), 'ktx-doctor-'));
|
|
});
|
|
|
|
afterEach(async () => {
|
|
await rm(tempDir, { recursive: true, force: true });
|
|
});
|
|
|
|
it('prints setup report and exits nonzero when a check fails', async () => {
|
|
const testIo = makeIo();
|
|
|
|
await expect(
|
|
runKtxDoctor(
|
|
{ command: 'setup', outputMode: 'plain', inputMode: 'disabled' },
|
|
testIo.io,
|
|
{
|
|
runSetupChecks: async () => [
|
|
{ id: 'node', label: 'Node 22+', status: 'pass', detail: 'v22.16.0 ABI 127' },
|
|
{
|
|
id: 'package-build',
|
|
label: 'TypeScript package build',
|
|
status: 'fail',
|
|
detail: 'Missing packages/cli/dist/bin.js',
|
|
fix: 'Run: pnpm run build',
|
|
},
|
|
],
|
|
},
|
|
),
|
|
).resolves.toBe(1);
|
|
|
|
expect(testIo.stdout()).toContain('KTX status');
|
|
expect(testIo.stdout()).toContain('No project here yet.');
|
|
expect(testIo.stdout()).toContain('Before you can run');
|
|
expect(testIo.stdout()).toContain('✗ TypeScript package build: Missing packages/cli/dist/bin.js');
|
|
expect(testIo.stdout()).toContain('→ Run: pnpm run build');
|
|
expect(testIo.stderr()).toBe('');
|
|
});
|
|
|
|
it('leads with `ktx setup` and hides toolchain warnings when no project exists', async () => {
|
|
const testIo = makeIo();
|
|
|
|
await expect(
|
|
runKtxDoctor(
|
|
{ command: 'setup', outputMode: 'plain', inputMode: 'disabled' },
|
|
testIo.io,
|
|
{
|
|
runSetupChecks: async () => [
|
|
{ id: 'node', label: 'Node 22+', status: 'pass', detail: 'v22.16.0', group: 'toolchain' },
|
|
{
|
|
id: 'corepack',
|
|
label: 'Corepack',
|
|
status: 'warn',
|
|
detail: 'spawn corepack ENOENT',
|
|
fix: 'Run: corepack enable',
|
|
group: 'toolchain',
|
|
},
|
|
],
|
|
},
|
|
),
|
|
).resolves.toBe(0);
|
|
|
|
const out = testIo.stdout();
|
|
expect(out).toContain('No project here yet.');
|
|
expect(out).toContain('Run');
|
|
expect(out).toContain('ktx setup');
|
|
expect(out).not.toContain('Corepack');
|
|
expect(out).not.toContain('Node 22+');
|
|
});
|
|
|
|
it('prints JSON setup report', async () => {
|
|
const testIo = makeIo();
|
|
|
|
await expect(
|
|
runKtxDoctor(
|
|
{ command: 'setup', outputMode: 'json', inputMode: 'disabled' },
|
|
testIo.io,
|
|
{
|
|
runSetupChecks: async () => [
|
|
{ id: 'node', label: 'Node 22+', status: 'pass', detail: 'v22.16.0 ABI 127' },
|
|
],
|
|
},
|
|
),
|
|
).resolves.toBe(0);
|
|
|
|
expect(JSON.parse(testIo.stdout())).toEqual({
|
|
title: 'KTX status',
|
|
checks: [{ id: 'node', label: 'Node 22+', status: 'pass', detail: 'v22.16.0 ABI 127' }],
|
|
});
|
|
});
|
|
|
|
it('prints a friendly message when ktx.yaml is missing at the project dir', async () => {
|
|
const originalEnvProjectDir = process.env.KTX_PROJECT_DIR;
|
|
process.env.KTX_PROJECT_DIR = tempDir;
|
|
const testIo = makeIo();
|
|
|
|
await expect(
|
|
runKtxDoctor(
|
|
{ command: 'project', projectDir: tempDir, outputMode: 'plain', inputMode: 'disabled' },
|
|
testIo.io,
|
|
{},
|
|
),
|
|
).resolves.toBe(1);
|
|
|
|
const out = testIo.stdout();
|
|
expect(out).toContain('KTX status');
|
|
expect(out).toContain('No KTX project here yet.');
|
|
expect(out).toContain('ktx setup');
|
|
expect(out).toContain('KTX_PROJECT_DIR');
|
|
expect(out).not.toContain('ENOENT');
|
|
expect(testIo.stderr()).toBe('');
|
|
|
|
if (originalEnvProjectDir === undefined) {
|
|
delete process.env.KTX_PROJECT_DIR;
|
|
} else {
|
|
process.env.KTX_PROJECT_DIR = originalEnvProjectDir;
|
|
}
|
|
});
|
|
|
|
it('emits a structured JSON error when ktx.yaml is missing and JSON output is requested', async () => {
|
|
const testIo = makeIo();
|
|
|
|
await expect(
|
|
runKtxDoctor(
|
|
{ command: 'project', projectDir: tempDir, outputMode: 'json', inputMode: 'disabled' },
|
|
testIo.io,
|
|
{},
|
|
),
|
|
).resolves.toBe(1);
|
|
|
|
const parsed = JSON.parse(testIo.stdout()) as { error: string; projectDir: string };
|
|
expect(parsed.error).toBe('missing_project');
|
|
expect(parsed.projectDir).toBe(tempDir);
|
|
});
|
|
|
|
it('prints schema issues and exits 1 when ktx.yaml fails Zod validation', async () => {
|
|
await writeFile(
|
|
join(tempDir, 'ktx.yaml'),
|
|
[
|
|
'storrage:',
|
|
' state: sqlite',
|
|
'ingest:',
|
|
' llm:',
|
|
' backend: anthropic',
|
|
'',
|
|
].join('\n'),
|
|
'utf-8',
|
|
);
|
|
const testIo = makeIo();
|
|
|
|
await expect(
|
|
runKtxDoctor(
|
|
{ command: 'project', projectDir: tempDir, outputMode: 'plain', inputMode: 'disabled' },
|
|
testIo.io,
|
|
{},
|
|
),
|
|
).resolves.toBe(1);
|
|
|
|
const out = testIo.stdout();
|
|
expect(out).toContain('KTX status');
|
|
expect(out).toContain('Config');
|
|
expect(out).toContain('Unsupported storrage: unknown field');
|
|
expect(out).toContain('Unsupported ingest.llm: use top-level llm.provider');
|
|
expect(out).toContain('ktx.yaml');
|
|
});
|
|
|
|
it('emits structured JSON when ktx.yaml fails Zod validation', async () => {
|
|
await writeFile(
|
|
join(tempDir, 'ktx.yaml'),
|
|
['storrage: {}', ''].join('\n'),
|
|
'utf-8',
|
|
);
|
|
const testIo = makeIo();
|
|
|
|
await expect(
|
|
runKtxDoctor(
|
|
{ command: 'project', projectDir: tempDir, outputMode: 'json', inputMode: 'disabled' },
|
|
testIo.io,
|
|
{},
|
|
),
|
|
).resolves.toBe(1);
|
|
|
|
const parsed = JSON.parse(testIo.stdout()) as {
|
|
error: string;
|
|
projectDir: string;
|
|
issues: Array<{ path: string; message: string }>;
|
|
};
|
|
expect(parsed.error).toBe('invalid_config');
|
|
expect(parsed.projectDir).toBe(tempDir);
|
|
expect(parsed.issues.some((issue) => issue.path === 'storrage')).toBe(true);
|
|
});
|
|
|
|
it('shows a Config row labelled "ktx.yaml schema valid" on the happy path', async () => {
|
|
process.env.ANTHROPIC_API_KEY = 'test-key'; // pragma: allowlist secret
|
|
await writeFile(
|
|
join(tempDir, 'ktx.yaml'),
|
|
[
|
|
'connections:',
|
|
' warehouse:',
|
|
' driver: sqlite',
|
|
' path: ./warehouse.db',
|
|
'llm:',
|
|
' provider:',
|
|
' backend: anthropic',
|
|
'',
|
|
].join('\n'),
|
|
'utf-8',
|
|
);
|
|
const testIo = makeIo();
|
|
|
|
await expect(
|
|
runKtxDoctor(
|
|
{ command: 'project', projectDir: tempDir, outputMode: 'plain', inputMode: 'disabled' },
|
|
testIo.io,
|
|
{},
|
|
),
|
|
).resolves.toBe(0);
|
|
|
|
expect(testIo.stdout()).toContain('ktx.yaml schema valid');
|
|
delete process.env.ANTHROPIC_API_KEY;
|
|
});
|
|
|
|
it('runs project checks against a valid ktx.yaml', async () => {
|
|
process.env.ANTHROPIC_API_KEY = 'test-key'; // pragma: allowlist secret
|
|
await writeFile(
|
|
join(tempDir, 'ktx.yaml'),
|
|
[
|
|
'connections:',
|
|
' warehouse:',
|
|
' driver: sqlite',
|
|
' path: ./warehouse.db',
|
|
'llm:',
|
|
' provider:',
|
|
' backend: anthropic',
|
|
' models:',
|
|
' default: claude-sonnet-4-5',
|
|
'ingest:',
|
|
' adapters:',
|
|
' - live-database',
|
|
' embeddings:',
|
|
' backend: openai',
|
|
' model: text-embedding-3-small',
|
|
' dimensions: 1536',
|
|
'',
|
|
].join('\n'),
|
|
'utf-8',
|
|
);
|
|
process.env.OPENAI_API_KEY = 'test-key'; // pragma: allowlist secret
|
|
const testIo = makeIo();
|
|
|
|
await expect(
|
|
runKtxDoctor(
|
|
{ command: 'project', projectDir: tempDir, outputMode: 'plain', inputMode: 'disabled' },
|
|
testIo.io,
|
|
{},
|
|
),
|
|
).resolves.toBe(0);
|
|
|
|
const out = testIo.stdout();
|
|
expect(out).toContain('KTX status');
|
|
expect(out).toContain(`· ${basename(tempDir)}`);
|
|
expect(out).toContain('Connections (1)');
|
|
expect(out).toContain('LLM');
|
|
expect(out).toContain('anthropic');
|
|
expect(out).toContain('Embeddings');
|
|
expect(out).toContain('Ready.');
|
|
delete process.env.ANTHROPIC_API_KEY;
|
|
delete process.env.OPENAI_API_KEY;
|
|
});
|
|
|
|
it('reports Claude Code auth failures and ignored prompt-caching fields in project doctor output', async () => {
|
|
await writeFile(
|
|
join(tempDir, 'ktx.yaml'),
|
|
[
|
|
'llm:',
|
|
' provider:',
|
|
' backend: claude-code',
|
|
' models:',
|
|
' default: sonnet',
|
|
' promptCaching:',
|
|
' enabled: true',
|
|
' systemTtl: 1h',
|
|
' toolsTtl: 1h',
|
|
' historyTtl: 5m',
|
|
'',
|
|
].join('\n'),
|
|
'utf-8',
|
|
);
|
|
const testIo = makeIo();
|
|
|
|
await expect(
|
|
runKtxDoctor(
|
|
{ command: 'project', projectDir: tempDir, outputMode: 'plain', inputMode: 'disabled' },
|
|
testIo.io,
|
|
{
|
|
claudeCodeAuthProbe: async () => ({
|
|
ok: false as const,
|
|
message: 'Authenticate Claude Code locally.',
|
|
}),
|
|
},
|
|
),
|
|
).resolves.toBe(1);
|
|
|
|
expect(testIo.stdout()).toContain('claude-code');
|
|
expect(testIo.stdout()).toContain('Authenticate Claude Code locally');
|
|
expect(testIo.stdout()).toContain('claude-code ignores llm.promptCaching');
|
|
});
|
|
|
|
it('includes Postgres query-history readiness in project doctor output', async () => {
|
|
process.env.ANTHROPIC_API_KEY = 'test-key'; // pragma: allowlist secret
|
|
process.env.OPENAI_API_KEY = 'test-key'; // pragma: allowlist secret
|
|
process.env.WAREHOUSE_DATABASE_URL = 'postgresql://reader@example.test/warehouse';
|
|
await writeFile(
|
|
join(tempDir, 'ktx.yaml'),
|
|
[
|
|
'connections:',
|
|
' warehouse:',
|
|
' driver: postgres',
|
|
' url: env:WAREHOUSE_DATABASE_URL',
|
|
' context:',
|
|
' queryHistory:',
|
|
' enabled: true',
|
|
'llm:',
|
|
' provider:',
|
|
' backend: anthropic',
|
|
'ingest:',
|
|
' adapters:',
|
|
' - live-database',
|
|
' - historic-sql',
|
|
' embeddings:',
|
|
' backend: openai',
|
|
' model: text-embedding-3-small',
|
|
' dimensions: 1536',
|
|
'',
|
|
].join('\n'),
|
|
'utf-8',
|
|
);
|
|
const testIo = makeIo();
|
|
let probeCalls = 0;
|
|
|
|
await expect(
|
|
runKtxDoctor(
|
|
{ command: 'project', projectDir: tempDir, outputMode: 'plain', inputMode: 'disabled' },
|
|
testIo.io,
|
|
{
|
|
postgresQueryHistoryProbe: async () => {
|
|
probeCalls += 1;
|
|
return {
|
|
pgServerVersion: 'PostgreSQL 16.4',
|
|
warnings: [],
|
|
info: [
|
|
'pg_stat_statements.max is 1000; set it to at least 5000 to reduce query-template eviction churn',
|
|
],
|
|
};
|
|
},
|
|
},
|
|
),
|
|
).resolves.toBe(0);
|
|
|
|
const out = testIo.stdout();
|
|
expect(probeCalls).toBe(1);
|
|
expect(out).toContain('Query history');
|
|
expect(out).toContain('warehouse');
|
|
expect(out).toContain('pg_stat_statements ready (PostgreSQL 16.4)');
|
|
expect(out).toContain('info: pg_stat_statements.max is 1000');
|
|
expect(out).not.toContain('Update the Postgres parameter group or config');
|
|
expect(out).toContain('ktx status --json');
|
|
expect(out).toContain('ktx sl');
|
|
expect(out).toContain('ktx wiki');
|
|
expect(out).not.toContain('ktx scan');
|
|
expect(out).not.toContain('ktx sl ask');
|
|
delete process.env.ANTHROPIC_API_KEY;
|
|
delete process.env.OPENAI_API_KEY;
|
|
delete process.env.WAREHOUSE_DATABASE_URL;
|
|
});
|
|
|
|
it('returns blocked verdict when LLM is not configured', async () => {
|
|
await writeFile(
|
|
join(tempDir, 'ktx.yaml'),
|
|
[
|
|
'connections:',
|
|
' warehouse:',
|
|
' driver: sqlite',
|
|
' path: ./warehouse.db',
|
|
'',
|
|
].join('\n'),
|
|
'utf-8',
|
|
);
|
|
const testIo = makeIo();
|
|
|
|
await expect(
|
|
runKtxDoctor(
|
|
{ command: 'project', projectDir: tempDir, outputMode: 'plain', inputMode: 'disabled' },
|
|
testIo.io,
|
|
{},
|
|
),
|
|
).resolves.toBe(1);
|
|
|
|
expect(testIo.stdout()).toContain('no LLM configured');
|
|
expect(testIo.stdout()).not.toContain('ktx ask');
|
|
expect(testIo.stdout()).toContain('ktx setup');
|
|
});
|
|
|
|
it('warns about stale and unsupported per-driver connection fields', async () => {
|
|
process.env.ANTHROPIC_API_KEY = 'test-key'; // pragma: allowlist secret
|
|
process.env.WAREHOUSE_DATABASE_URL = 'postgresql://reader@example.test/warehouse';
|
|
process.env.NOTION_TOKEN = 'notion-secret';
|
|
await writeFile(
|
|
join(tempDir, 'ktx.yaml'),
|
|
[
|
|
'connections:',
|
|
' warehouse:',
|
|
' driver: postgres',
|
|
' url: env:WAREHOUSE_DATABASE_URL',
|
|
' readonly: true',
|
|
' historicSql:',
|
|
' enabled: true',
|
|
' dialect: postgres',
|
|
' windowDays: 30',
|
|
' concurrency: 4',
|
|
' local:',
|
|
' driver: sqlite',
|
|
' file_path: ./warehouse.db',
|
|
' docs:',
|
|
' driver: notion',
|
|
' auth_token_ref: env:NOTION_TOKEN',
|
|
' crawl_mode: all_accessible',
|
|
' last_successful_cursor: \'{"phase":"all_accessible_pages","cursor":"cursor-1"}\'',
|
|
'ingest:',
|
|
' adapters:',
|
|
' - live-database',
|
|
'llm:',
|
|
' provider:',
|
|
' backend: anthropic',
|
|
'',
|
|
].join('\n'),
|
|
'utf-8',
|
|
);
|
|
const testIo = makeIo();
|
|
|
|
await expect(
|
|
runKtxDoctor(
|
|
{ command: 'project', projectDir: tempDir, outputMode: 'plain', inputMode: 'disabled' },
|
|
testIo.io,
|
|
{
|
|
postgresQueryHistoryProbe: async () => ({
|
|
pgServerVersion: 'PostgreSQL 16.4',
|
|
warnings: [],
|
|
info: [],
|
|
}),
|
|
},
|
|
),
|
|
).resolves.toBe(0);
|
|
|
|
const out = testIo.stdout();
|
|
expect(out).toContain('Warnings');
|
|
expect(out).toContain('connections.warehouse.readonly is no longer used.');
|
|
expect(out).toContain('connections.warehouse.historicSql.concurrency is no longer used.');
|
|
expect(out).toContain('connections.warehouse.historicSql.windowDays does not constrain pg_stat_statements.');
|
|
expect(out).toContain('connections.local.file_path was removed.');
|
|
expect(out).toContain('connections.docs.last_successful_cursor is local sync state.');
|
|
delete process.env.ANTHROPIC_API_KEY;
|
|
delete process.env.WAREHOUSE_DATABASE_URL;
|
|
delete process.env.NOTION_TOKEN;
|
|
});
|
|
|
|
it('warns when semantic-search embeddings are not configured', async () => {
|
|
process.env.ANTHROPIC_API_KEY = 'test-key'; // pragma: allowlist secret
|
|
await writeFile(
|
|
join(tempDir, 'ktx.yaml'),
|
|
[
|
|
'connections:',
|
|
' warehouse:',
|
|
' driver: sqlite',
|
|
' path: ./warehouse.db',
|
|
'llm:',
|
|
' provider:',
|
|
' backend: anthropic',
|
|
'ingest:',
|
|
' adapters:',
|
|
' - live-database',
|
|
' embeddings:',
|
|
' backend: none',
|
|
' dimensions: 8',
|
|
'',
|
|
].join('\n'),
|
|
'utf-8',
|
|
);
|
|
const testIo = makeIo();
|
|
|
|
await expect(
|
|
runKtxDoctor(
|
|
{ command: 'project', projectDir: tempDir, outputMode: 'plain', inputMode: 'disabled' },
|
|
testIo.io,
|
|
{},
|
|
),
|
|
).resolves.toBe(0);
|
|
|
|
expect(testIo.stdout()).toContain('Embeddings');
|
|
expect(testIo.stdout()).toContain('none');
|
|
expect(testIo.stdout()).toContain('semantic search will be skipped');
|
|
delete process.env.ANTHROPIC_API_KEY;
|
|
});
|
|
|
|
describe('command: validate', () => {
|
|
it('prints a success line and exits 0 when ktx.yaml is schema-valid', async () => {
|
|
await writeFile(
|
|
join(tempDir, 'ktx.yaml'),
|
|
[
|
|
'connections:',
|
|
' warehouse:',
|
|
' driver: sqlite',
|
|
' path: ./warehouse.db',
|
|
'llm:',
|
|
' provider:',
|
|
' backend: anthropic',
|
|
'',
|
|
].join('\n'),
|
|
'utf-8',
|
|
);
|
|
const testIo = makeIo();
|
|
|
|
await expect(
|
|
runKtxDoctor(
|
|
{ command: 'validate', projectDir: tempDir, outputMode: 'plain', inputMode: 'disabled' },
|
|
testIo.io,
|
|
{},
|
|
),
|
|
).resolves.toBe(0);
|
|
|
|
const out = testIo.stdout();
|
|
expect(out).toContain('KTX status');
|
|
expect(out).toContain('Config');
|
|
expect(out).toContain('ktx.yaml schema valid');
|
|
expect(out).not.toContain('LLM');
|
|
expect(out).not.toContain('Connections');
|
|
expect(out).not.toContain('Pipeline');
|
|
});
|
|
|
|
it('emits {ok: true} JSON when ktx.yaml is schema-valid', async () => {
|
|
await writeFile(
|
|
join(tempDir, 'ktx.yaml'),
|
|
[
|
|
'connections:',
|
|
' warehouse:',
|
|
' driver: sqlite',
|
|
' path: ./warehouse.db',
|
|
'llm:',
|
|
' provider:',
|
|
' backend: anthropic',
|
|
'',
|
|
].join('\n'),
|
|
'utf-8',
|
|
);
|
|
const testIo = makeIo();
|
|
|
|
await expect(
|
|
runKtxDoctor(
|
|
{ command: 'validate', projectDir: tempDir, outputMode: 'json', inputMode: 'disabled' },
|
|
testIo.io,
|
|
{},
|
|
),
|
|
).resolves.toBe(0);
|
|
|
|
expect(JSON.parse(testIo.stdout())).toEqual({ ok: true, projectDir: tempDir });
|
|
});
|
|
|
|
it('prints schema issues and exits 1 when ktx.yaml fails Zod validation', async () => {
|
|
await writeFile(
|
|
join(tempDir, 'ktx.yaml'),
|
|
[
|
|
'storrage:',
|
|
' state: sqlite',
|
|
'ingest:',
|
|
' llm:',
|
|
' backend: anthropic',
|
|
'',
|
|
].join('\n'),
|
|
'utf-8',
|
|
);
|
|
const testIo = makeIo();
|
|
|
|
await expect(
|
|
runKtxDoctor(
|
|
{ command: 'validate', projectDir: tempDir, outputMode: 'plain', inputMode: 'disabled' },
|
|
testIo.io,
|
|
{},
|
|
),
|
|
).resolves.toBe(1);
|
|
|
|
const out = testIo.stdout();
|
|
expect(out).toContain('Unsupported storrage: unknown field');
|
|
expect(out).toContain('Unsupported ingest.llm: use top-level llm.provider');
|
|
});
|
|
|
|
it('emits structured JSON issues when validation fails', async () => {
|
|
await writeFile(
|
|
join(tempDir, 'ktx.yaml'),
|
|
['storrage: {}', ''].join('\n'),
|
|
'utf-8',
|
|
);
|
|
const testIo = makeIo();
|
|
|
|
await expect(
|
|
runKtxDoctor(
|
|
{ command: 'validate', projectDir: tempDir, outputMode: 'json', inputMode: 'disabled' },
|
|
testIo.io,
|
|
{},
|
|
),
|
|
).resolves.toBe(1);
|
|
|
|
const parsed = JSON.parse(testIo.stdout()) as { error: string; issues: Array<{ path: string }> };
|
|
expect(parsed.error).toBe('invalid_config');
|
|
expect(parsed.issues.some((issue) => issue.path === 'storrage')).toBe(true);
|
|
});
|
|
|
|
it('prints the missing-project message and exits 1 when ktx.yaml is absent', async () => {
|
|
const testIo = makeIo();
|
|
|
|
await expect(
|
|
runKtxDoctor(
|
|
{ command: 'validate', projectDir: tempDir, outputMode: 'plain', inputMode: 'disabled' },
|
|
testIo.io,
|
|
{},
|
|
),
|
|
).resolves.toBe(1);
|
|
|
|
expect(testIo.stdout()).toContain('No KTX project here yet.');
|
|
});
|
|
|
|
it('does not invoke the Postgres query-history probe in validate mode', async () => {
|
|
await writeFile(
|
|
join(tempDir, 'ktx.yaml'),
|
|
[
|
|
'connections:',
|
|
' warehouse:',
|
|
' driver: postgres',
|
|
' url: env:WAREHOUSE_DATABASE_URL',
|
|
' context:',
|
|
' queryHistory:',
|
|
' enabled: true',
|
|
'llm:',
|
|
' provider:',
|
|
' backend: anthropic',
|
|
'',
|
|
].join('\n'),
|
|
'utf-8',
|
|
);
|
|
const testIo = makeIo();
|
|
let probeCalls = 0;
|
|
|
|
await expect(
|
|
runKtxDoctor(
|
|
{ command: 'validate', projectDir: tempDir, outputMode: 'plain', inputMode: 'disabled' },
|
|
testIo.io,
|
|
{
|
|
postgresQueryHistoryProbe: async () => {
|
|
probeCalls += 1;
|
|
return { pgServerVersion: 'PostgreSQL 16.4', warnings: [], info: [] };
|
|
},
|
|
},
|
|
),
|
|
).resolves.toBe(0);
|
|
|
|
expect(probeCalls).toBe(0);
|
|
expect(testIo.stdout()).toContain('ktx.yaml schema valid');
|
|
});
|
|
});
|
|
});
|