2026-05-13 15:04:50 +02:00
|
|
|
import { mkdtemp, readFile, rm, writeFile } from 'node:fs/promises';
|
2026-05-10 23:12:26 +02:00
|
|
|
import { tmpdir } from 'node:os';
|
|
|
|
|
import { join } from 'node:path';
|
chore(workspace): gate dead-code with knip production mode (#196)
* 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.
2026-05-21 15:28:58 +02:00
|
|
|
import type { LookerClient } from './context/ingest/adapters/looker/client.js';
|
|
|
|
|
import type { MetabaseRuntimeClient } from './context/ingest/adapters/metabase/client-port.js';
|
|
|
|
|
import type { NotionClient } from './context/ingest/adapters/notion/notion-client.js';
|
|
|
|
|
import { initKtxProject } from './context/project/project.js';
|
|
|
|
|
import { parseKtxProjectConfig, serializeKtxProjectConfig } from './context/project/config.js';
|
|
|
|
|
import type { KtxConnectionDriver, KtxScanConnector } from './context/scan/types.js';
|
2026-05-10 23:12:26 +02:00
|
|
|
import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest';
|
2026-05-10 23:51:24 +02:00
|
|
|
import { runKtxConnection } from './connection.js';
|
2026-05-10 23:12:26 +02:00
|
|
|
|
2026-05-14 16:21:18 +02:00
|
|
|
function stripAnsi(s: string): string {
|
|
|
|
|
return s.replace(/\[[0-9;]*m/g, '');
|
|
|
|
|
}
|
|
|
|
|
|
2026-05-13 15:04:50 +02:00
|
|
|
function makeIo() {
|
2026-05-10 23:12:26 +02:00
|
|
|
let stdout = '';
|
|
|
|
|
let stderr = '';
|
|
|
|
|
return {
|
|
|
|
|
io: {
|
|
|
|
|
stdout: {
|
|
|
|
|
write: (chunk: string) => {
|
|
|
|
|
stdout += chunk;
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
stderr: {
|
|
|
|
|
write: (chunk: string) => {
|
|
|
|
|
stderr += chunk;
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
stdout: () => stdout,
|
|
|
|
|
stderr: () => stderr,
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
2026-05-14 16:21:18 +02:00
|
|
|
function nativeConnector(
|
|
|
|
|
driver: KtxConnectionDriver,
|
|
|
|
|
testResult: { success: true } | { success: false; error: string } = { success: true },
|
|
|
|
|
) {
|
|
|
|
|
const testConnection = vi.fn(async () => testResult);
|
2026-05-10 23:12:26 +02:00
|
|
|
const cleanup = vi.fn(async () => undefined);
|
2026-05-10 23:51:24 +02:00
|
|
|
const connector: KtxScanConnector = {
|
2026-05-10 23:12:26 +02:00
|
|
|
id: `${driver}:warehouse`,
|
|
|
|
|
driver,
|
|
|
|
|
capabilities: {
|
|
|
|
|
structuralIntrospection: true,
|
|
|
|
|
tableSampling: false,
|
|
|
|
|
columnSampling: false,
|
|
|
|
|
columnStats: false,
|
|
|
|
|
readOnlySql: false,
|
|
|
|
|
nestedAnalysis: false,
|
|
|
|
|
eventStreamDiscovery: false,
|
|
|
|
|
formalForeignKeys: false,
|
|
|
|
|
estimatedRowCounts: false,
|
|
|
|
|
},
|
2026-05-14 16:21:18 +02:00
|
|
|
introspect: vi.fn(async () => {
|
|
|
|
|
throw new Error('introspect should not be called from connection test');
|
|
|
|
|
}),
|
|
|
|
|
testConnection,
|
2026-05-10 23:12:26 +02:00
|
|
|
cleanup,
|
|
|
|
|
};
|
2026-05-14 16:21:18 +02:00
|
|
|
return { connector, testConnection, cleanup };
|
2026-05-10 23:12:26 +02:00
|
|
|
}
|
|
|
|
|
|
2026-05-10 23:51:24 +02:00
|
|
|
describe('runKtxConnection', () => {
|
2026-05-10 23:12:26 +02:00
|
|
|
let tempDir: string;
|
|
|
|
|
|
|
|
|
|
beforeEach(async () => {
|
2026-05-10 23:51:24 +02:00
|
|
|
tempDir = await mkdtemp(join(tmpdir(), 'ktx-cli-connection-'));
|
2026-05-10 23:12:26 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
|
|
afterEach(async () => {
|
|
|
|
|
await rm(tempDir, { recursive: true, force: true });
|
|
|
|
|
});
|
|
|
|
|
|
2026-05-13 15:04:50 +02:00
|
|
|
async function writeConnections(
|
|
|
|
|
projectDir: string,
|
|
|
|
|
connections: ReturnType<typeof parseKtxProjectConfig>['connections'],
|
|
|
|
|
): Promise<void> {
|
|
|
|
|
const config = parseKtxProjectConfig(await readFile(join(projectDir, 'ktx.yaml'), 'utf-8'));
|
|
|
|
|
await writeFile(join(projectDir, 'ktx.yaml'), serializeKtxProjectConfig({ ...config, connections }), 'utf-8');
|
|
|
|
|
}
|
2026-05-10 23:12:26 +02:00
|
|
|
|
2026-05-13 15:04:50 +02:00
|
|
|
it('lists configured connections without resolving secrets', async () => {
|
2026-05-10 23:12:26 +02:00
|
|
|
const projectDir = join(tempDir, 'project');
|
2026-05-14 17:39:31 +02:00
|
|
|
await initKtxProject({ projectDir });
|
2026-05-13 15:04:50 +02:00
|
|
|
await writeConnections(projectDir, {
|
2026-05-13 19:37:25 +02:00
|
|
|
warehouse: { driver: 'postgres', url: 'env:DATABASE_URL' },
|
2026-05-13 15:04:50 +02:00
|
|
|
docs: { driver: 'notion', auth_token_ref: 'env:NOTION_TOKEN', crawl_mode: 'all_accessible' },
|
2026-05-10 23:12:26 +02:00
|
|
|
});
|
|
|
|
|
const io = makeIo();
|
|
|
|
|
|
2026-05-13 15:04:50 +02:00
|
|
|
await expect(runKtxConnection({ command: 'list', projectDir }, io.io)).resolves.toBe(0);
|
2026-05-10 23:12:26 +02:00
|
|
|
|
2026-05-13 15:04:50 +02:00
|
|
|
expect(io.stdout()).toContain('warehouse');
|
|
|
|
|
expect(io.stdout()).toContain('postgres');
|
|
|
|
|
expect(io.stdout()).toContain('docs');
|
|
|
|
|
expect(io.stdout()).toContain('notion');
|
2026-05-10 23:12:26 +02:00
|
|
|
expect(io.stderr()).toBe('');
|
|
|
|
|
});
|
|
|
|
|
|
2026-05-13 15:04:50 +02:00
|
|
|
it('prints an empty-state message that points at setup instead of removed connection add', async () => {
|
2026-05-10 23:12:26 +02:00
|
|
|
const projectDir = join(tempDir, 'project');
|
2026-05-14 17:39:31 +02:00
|
|
|
await initKtxProject({ projectDir });
|
2026-05-10 23:12:26 +02:00
|
|
|
const io = makeIo();
|
|
|
|
|
|
2026-05-13 15:04:50 +02:00
|
|
|
await expect(runKtxConnection({ command: 'list', projectDir }, io.io)).resolves.toBe(0);
|
2026-05-10 23:12:26 +02:00
|
|
|
|
2026-05-13 15:04:50 +02:00
|
|
|
expect(io.stdout()).toContain('No connections configured. Run `ktx setup` to add one.');
|
|
|
|
|
expect(io.stdout()).not.toContain('ktx connection add');
|
2026-05-10 23:12:26 +02:00
|
|
|
});
|
|
|
|
|
|
2026-05-14 16:21:18 +02:00
|
|
|
it('tests a native connection by calling connector.testConnection (not introspect)', async () => {
|
2026-05-10 23:12:26 +02:00
|
|
|
const projectDir = join(tempDir, 'project');
|
2026-05-14 17:39:31 +02:00
|
|
|
await initKtxProject({ projectDir });
|
2026-05-13 15:04:50 +02:00
|
|
|
await writeConnections(projectDir, {
|
2026-05-13 19:37:25 +02:00
|
|
|
warehouse: { driver: 'sqlite' },
|
2026-05-13 15:04:50 +02:00
|
|
|
});
|
2026-05-14 16:21:18 +02:00
|
|
|
const { connector, testConnection, cleanup } = nativeConnector('sqlite');
|
2026-05-10 23:12:26 +02:00
|
|
|
const createScanConnector = vi.fn(async () => connector);
|
|
|
|
|
const io = makeIo();
|
|
|
|
|
|
|
|
|
|
await expect(
|
2026-05-10 23:51:24 +02:00
|
|
|
runKtxConnection({ command: 'test', projectDir, connectionId: 'warehouse' }, io.io, {
|
2026-05-10 23:12:26 +02:00
|
|
|
createScanConnector,
|
|
|
|
|
}),
|
|
|
|
|
).resolves.toBe(0);
|
|
|
|
|
|
|
|
|
|
expect(createScanConnector).toHaveBeenCalledWith(expect.objectContaining({ projectDir }), 'warehouse');
|
2026-05-14 16:21:18 +02:00
|
|
|
expect(testConnection).toHaveBeenCalledTimes(1);
|
|
|
|
|
expect(connector.introspect).not.toHaveBeenCalled();
|
2026-05-10 23:12:26 +02:00
|
|
|
expect(cleanup).toHaveBeenCalledTimes(1);
|
|
|
|
|
expect(io.stdout()).toContain('Connection test passed: warehouse');
|
|
|
|
|
expect(io.stdout()).toContain('Driver: sqlite');
|
2026-05-14 16:21:18 +02:00
|
|
|
expect(io.stdout()).toContain('Status: ok');
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it('reports the connector error and still cleans up when native testConnection fails', async () => {
|
|
|
|
|
const projectDir = join(tempDir, 'project');
|
2026-05-14 17:39:31 +02:00
|
|
|
await initKtxProject({ projectDir });
|
2026-05-14 16:21:18 +02:00
|
|
|
await writeConnections(projectDir, {
|
|
|
|
|
warehouse: { driver: 'sqlite' },
|
|
|
|
|
});
|
|
|
|
|
const { connector, cleanup } = nativeConnector('sqlite', { success: false, error: 'database file is unreadable' });
|
|
|
|
|
const io = makeIo();
|
|
|
|
|
|
|
|
|
|
await expect(
|
|
|
|
|
runKtxConnection({ command: 'test', projectDir, connectionId: 'warehouse' }, io.io, {
|
|
|
|
|
createScanConnector: vi.fn(async () => connector),
|
|
|
|
|
}),
|
|
|
|
|
).resolves.toBe(1);
|
|
|
|
|
|
|
|
|
|
expect(cleanup).toHaveBeenCalledTimes(1);
|
|
|
|
|
expect(io.stderr()).toContain('database file is unreadable');
|
2026-05-10 23:12:26 +02:00
|
|
|
});
|
|
|
|
|
|
2026-05-12 10:25:54 +02:00
|
|
|
it('tests a configured Metabase connection through the Metabase runtime client', async () => {
|
|
|
|
|
const projectDir = join(tempDir, 'project');
|
2026-05-14 17:39:31 +02:00
|
|
|
await initKtxProject({ projectDir });
|
2026-05-13 15:04:50 +02:00
|
|
|
await writeConnections(projectDir, {
|
|
|
|
|
prod_metabase: {
|
|
|
|
|
driver: 'metabase',
|
|
|
|
|
api_url: 'http://metabase.example.test',
|
2026-05-13 19:49:25 +02:00
|
|
|
api_key: 'mb_test', // pragma: allowlist secret
|
2026-05-13 15:04:50 +02:00
|
|
|
},
|
|
|
|
|
});
|
2026-05-12 10:25:54 +02:00
|
|
|
const testConnection = vi.fn(async () => ({ success: true as const }));
|
|
|
|
|
const getDatabases = vi.fn(async () => [
|
|
|
|
|
{ id: 1, name: 'Analytics', engine: 'postgres', details: {}, is_sample: false },
|
|
|
|
|
{ id: 2, name: 'Sample Database', engine: 'h2', details: {}, is_sample: true },
|
|
|
|
|
]);
|
|
|
|
|
const cleanup = vi.fn(async () => undefined);
|
|
|
|
|
const createMetabaseClient = vi.fn(
|
|
|
|
|
async (): Promise<Pick<MetabaseRuntimeClient, 'testConnection' | 'getDatabases' | 'cleanup'>> => ({
|
|
|
|
|
testConnection,
|
|
|
|
|
getDatabases,
|
|
|
|
|
cleanup,
|
|
|
|
|
}),
|
|
|
|
|
);
|
|
|
|
|
const createScanConnector = vi.fn(async () => {
|
|
|
|
|
throw new Error('native scanner should not be used for Metabase');
|
|
|
|
|
});
|
|
|
|
|
const io = makeIo();
|
|
|
|
|
|
|
|
|
|
await expect(
|
|
|
|
|
runKtxConnection({ command: 'test', projectDir, connectionId: 'prod_metabase' }, io.io, {
|
|
|
|
|
createScanConnector,
|
|
|
|
|
createMetabaseClient,
|
|
|
|
|
}),
|
|
|
|
|
).resolves.toBe(0);
|
|
|
|
|
|
|
|
|
|
expect(createScanConnector).not.toHaveBeenCalled();
|
|
|
|
|
expect(createMetabaseClient).toHaveBeenCalledWith(expect.objectContaining({ projectDir }), 'prod_metabase');
|
|
|
|
|
expect(testConnection).toHaveBeenCalledTimes(1);
|
|
|
|
|
expect(getDatabases).toHaveBeenCalledTimes(1);
|
|
|
|
|
expect(cleanup).toHaveBeenCalledTimes(1);
|
|
|
|
|
expect(io.stdout()).toContain('Connection test passed: prod_metabase');
|
|
|
|
|
expect(io.stdout()).toContain('Driver: metabase');
|
|
|
|
|
expect(io.stdout()).toContain('Databases: 1');
|
|
|
|
|
expect(io.stderr()).toBe('');
|
|
|
|
|
});
|
|
|
|
|
|
2026-05-14 16:21:18 +02:00
|
|
|
it('tests a Looker connection through the Looker client', async () => {
|
2026-05-10 23:12:26 +02:00
|
|
|
const projectDir = join(tempDir, 'project');
|
2026-05-14 17:39:31 +02:00
|
|
|
await initKtxProject({ projectDir });
|
2026-05-13 15:04:50 +02:00
|
|
|
await writeConnections(projectDir, {
|
2026-05-14 16:21:18 +02:00
|
|
|
bi_looker: {
|
|
|
|
|
driver: 'looker',
|
|
|
|
|
base_url: 'https://looker.example.test',
|
|
|
|
|
client_id: 'cid',
|
|
|
|
|
client_secret: 'csecret', // pragma: allowlist secret
|
|
|
|
|
},
|
2026-05-13 15:04:50 +02:00
|
|
|
});
|
2026-05-14 16:21:18 +02:00
|
|
|
const testConnection = vi.fn(async () => ({
|
|
|
|
|
success: true as const,
|
|
|
|
|
metadata: { displayName: 'Alice Analyst', userId: '42' },
|
|
|
|
|
}));
|
|
|
|
|
const createLookerClient = vi.fn(async (): Promise<Pick<LookerClient, 'testConnection'>> => ({ testConnection }));
|
|
|
|
|
const io = makeIo();
|
|
|
|
|
|
|
|
|
|
await expect(
|
|
|
|
|
runKtxConnection({ command: 'test', projectDir, connectionId: 'bi_looker' }, io.io, { createLookerClient }),
|
|
|
|
|
).resolves.toBe(0);
|
|
|
|
|
|
|
|
|
|
expect(createLookerClient).toHaveBeenCalledWith(expect.objectContaining({ projectDir }), 'bi_looker');
|
|
|
|
|
expect(testConnection).toHaveBeenCalledTimes(1);
|
|
|
|
|
expect(io.stdout()).toContain('Connection test passed: bi_looker');
|
|
|
|
|
expect(io.stdout()).toContain('Driver: looker');
|
|
|
|
|
expect(io.stdout()).toContain('User: Alice Analyst');
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it('falls back to userId when Looker metadata has no display name', async () => {
|
|
|
|
|
const projectDir = join(tempDir, 'project');
|
2026-05-14 17:39:31 +02:00
|
|
|
await initKtxProject({ projectDir });
|
2026-05-14 16:21:18 +02:00
|
|
|
await writeConnections(projectDir, {
|
|
|
|
|
bi_looker: {
|
|
|
|
|
driver: 'looker',
|
|
|
|
|
base_url: 'https://looker.example.test',
|
|
|
|
|
client_id: 'cid',
|
|
|
|
|
client_secret: 'csecret', // pragma: allowlist secret
|
2026-05-10 23:12:26 +02:00
|
|
|
},
|
2026-05-14 16:21:18 +02:00
|
|
|
});
|
|
|
|
|
const createLookerClient = vi.fn(async (): Promise<Pick<LookerClient, 'testConnection'>> => ({
|
|
|
|
|
testConnection: vi.fn(async () => ({
|
|
|
|
|
success: true as const,
|
|
|
|
|
metadata: { displayName: null, userId: '42' },
|
|
|
|
|
})),
|
|
|
|
|
}));
|
2026-05-10 23:12:26 +02:00
|
|
|
const io = makeIo();
|
|
|
|
|
|
|
|
|
|
await expect(
|
2026-05-14 16:21:18 +02:00
|
|
|
runKtxConnection({ command: 'test', projectDir, connectionId: 'bi_looker' }, io.io, { createLookerClient }),
|
|
|
|
|
).resolves.toBe(0);
|
|
|
|
|
expect(io.stdout()).toContain('User: 42');
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it('reports the Looker error when testConnection fails', async () => {
|
|
|
|
|
const projectDir = join(tempDir, 'project');
|
2026-05-14 17:39:31 +02:00
|
|
|
await initKtxProject({ projectDir });
|
2026-05-14 16:21:18 +02:00
|
|
|
await writeConnections(projectDir, {
|
|
|
|
|
bi_looker: {
|
|
|
|
|
driver: 'looker',
|
|
|
|
|
base_url: 'https://looker.example.test',
|
|
|
|
|
client_id: 'cid',
|
|
|
|
|
client_secret: 'csecret', // pragma: allowlist secret
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
const createLookerClient = vi.fn(async (): Promise<Pick<LookerClient, 'testConnection'>> => ({
|
|
|
|
|
testConnection: vi.fn(async () => ({ success: false as const, error: 'invalid client_id' })),
|
|
|
|
|
}));
|
|
|
|
|
const io = makeIo();
|
|
|
|
|
|
|
|
|
|
await expect(
|
|
|
|
|
runKtxConnection({ command: 'test', projectDir, connectionId: 'bi_looker' }, io.io, { createLookerClient }),
|
2026-05-10 23:12:26 +02:00
|
|
|
).resolves.toBe(1);
|
2026-05-14 16:21:18 +02:00
|
|
|
expect(io.stderr()).toContain('Looker connection test failed: invalid client_id');
|
|
|
|
|
});
|
2026-05-10 23:12:26 +02:00
|
|
|
|
2026-05-14 16:21:18 +02:00
|
|
|
it('tests a Notion connection by retrieving the bot user', async () => {
|
|
|
|
|
const projectDir = join(tempDir, 'project');
|
2026-05-14 17:39:31 +02:00
|
|
|
await initKtxProject({ projectDir });
|
2026-05-14 16:21:18 +02:00
|
|
|
await writeConnections(projectDir, {
|
|
|
|
|
docs: {
|
|
|
|
|
driver: 'notion',
|
|
|
|
|
auth_token: 'secret_token', // pragma: allowlist secret
|
|
|
|
|
crawl_mode: 'all_accessible',
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
const retrieveBotUser = vi.fn(async () => ({ id: 'bot-1', name: 'Analytics Bot' }));
|
|
|
|
|
const createNotionClient = vi.fn(async (): Promise<Pick<NotionClient, 'retrieveBotUser'>> => ({ retrieveBotUser }));
|
|
|
|
|
const io = makeIo();
|
|
|
|
|
|
|
|
|
|
await expect(
|
|
|
|
|
runKtxConnection({ command: 'test', projectDir, connectionId: 'docs' }, io.io, { createNotionClient }),
|
|
|
|
|
).resolves.toBe(0);
|
|
|
|
|
|
|
|
|
|
expect(createNotionClient).toHaveBeenCalledWith(expect.objectContaining({ projectDir }), 'docs');
|
|
|
|
|
expect(retrieveBotUser).toHaveBeenCalledTimes(1);
|
|
|
|
|
expect(io.stdout()).toContain('Connection test passed: docs');
|
|
|
|
|
expect(io.stdout()).toContain('Driver: notion');
|
|
|
|
|
expect(io.stdout()).toContain('Bot: Analytics Bot');
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it('falls back to bot id when Notion bot has no name', async () => {
|
|
|
|
|
const projectDir = join(tempDir, 'project');
|
2026-05-14 17:39:31 +02:00
|
|
|
await initKtxProject({ projectDir });
|
2026-05-14 16:21:18 +02:00
|
|
|
await writeConnections(projectDir, {
|
|
|
|
|
docs: {
|
|
|
|
|
driver: 'notion',
|
|
|
|
|
auth_token: 'secret_token', // pragma: allowlist secret
|
|
|
|
|
crawl_mode: 'all_accessible',
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
const createNotionClient = vi.fn(async (): Promise<Pick<NotionClient, 'retrieveBotUser'>> => ({
|
|
|
|
|
retrieveBotUser: vi.fn(async () => ({ id: 'bot-1', name: null })),
|
|
|
|
|
}));
|
|
|
|
|
const io = makeIo();
|
|
|
|
|
|
|
|
|
|
await expect(
|
|
|
|
|
runKtxConnection({ command: 'test', projectDir, connectionId: 'docs' }, io.io, { createNotionClient }),
|
|
|
|
|
).resolves.toBe(0);
|
|
|
|
|
expect(io.stdout()).toContain('Bot: bot-1');
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it('tests a dbt connection via testRepoConnection (success)', async () => {
|
|
|
|
|
const projectDir = join(tempDir, 'project');
|
2026-05-14 17:39:31 +02:00
|
|
|
await initKtxProject({ projectDir });
|
2026-05-14 16:21:18 +02:00
|
|
|
process.env.DBT_TOKEN = 'gh_token_abc'; // pragma: allowlist secret
|
|
|
|
|
await writeConnections(projectDir, {
|
|
|
|
|
'dbt-main': {
|
|
|
|
|
driver: 'dbt',
|
|
|
|
|
repo_url: 'https://github.com/example/dbt-project',
|
|
|
|
|
auth_token_ref: 'env:DBT_TOKEN',
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
const testRepoConnection = vi.fn(async () => ({ ok: true as const }));
|
|
|
|
|
const io = makeIo();
|
|
|
|
|
|
|
|
|
|
try {
|
|
|
|
|
await expect(
|
|
|
|
|
runKtxConnection({ command: 'test', projectDir, connectionId: 'dbt-main' }, io.io, { testRepoConnection }),
|
|
|
|
|
).resolves.toBe(0);
|
|
|
|
|
|
|
|
|
|
expect(testRepoConnection).toHaveBeenCalledWith({
|
|
|
|
|
repoUrl: 'https://github.com/example/dbt-project',
|
|
|
|
|
authToken: 'gh_token_abc',
|
|
|
|
|
});
|
|
|
|
|
expect(io.stdout()).toContain('Connection test passed: dbt-main');
|
|
|
|
|
expect(io.stdout()).toContain('Driver: dbt');
|
|
|
|
|
expect(io.stdout()).toContain('Repo: https://github.com/example/dbt-project');
|
|
|
|
|
} finally {
|
|
|
|
|
delete process.env.DBT_TOKEN;
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it('reports the git error when testRepoConnection fails for dbt', async () => {
|
|
|
|
|
const projectDir = join(tempDir, 'project');
|
2026-05-14 17:39:31 +02:00
|
|
|
await initKtxProject({ projectDir });
|
2026-05-14 16:21:18 +02:00
|
|
|
await writeConnections(projectDir, {
|
|
|
|
|
'dbt-main': {
|
|
|
|
|
driver: 'dbt',
|
|
|
|
|
repo_url: 'https://github.com/example/dbt-project',
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
const testRepoConnection = vi.fn(async () => ({ ok: false as const, error: 'fatal: auth failed' }));
|
|
|
|
|
const io = makeIo();
|
|
|
|
|
|
|
|
|
|
await expect(
|
|
|
|
|
runKtxConnection({ command: 'test', projectDir, connectionId: 'dbt-main' }, io.io, { testRepoConnection }),
|
|
|
|
|
).resolves.toBe(1);
|
|
|
|
|
|
|
|
|
|
expect(testRepoConnection).toHaveBeenCalledWith({
|
|
|
|
|
repoUrl: 'https://github.com/example/dbt-project',
|
|
|
|
|
authToken: null,
|
|
|
|
|
});
|
|
|
|
|
expect(io.stderr()).toContain('dbt repository check failed: fatal: auth failed');
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it('tests a LookML connection via testRepoConnection with camelCase repoUrl', async () => {
|
|
|
|
|
const projectDir = join(tempDir, 'project');
|
2026-05-14 17:39:31 +02:00
|
|
|
await initKtxProject({ projectDir });
|
2026-05-14 16:21:18 +02:00
|
|
|
await writeConnections(projectDir, {
|
|
|
|
|
lookml_main: {
|
|
|
|
|
driver: 'lookml',
|
|
|
|
|
repoUrl: 'https://github.com/example/lookml',
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
const testRepoConnection = vi.fn(async () => ({ ok: true as const }));
|
|
|
|
|
const io = makeIo();
|
|
|
|
|
|
|
|
|
|
await expect(
|
|
|
|
|
runKtxConnection({ command: 'test', projectDir, connectionId: 'lookml_main' }, io.io, { testRepoConnection }),
|
|
|
|
|
).resolves.toBe(0);
|
|
|
|
|
expect(testRepoConnection).toHaveBeenCalledWith({
|
|
|
|
|
repoUrl: 'https://github.com/example/lookml',
|
|
|
|
|
authToken: null,
|
|
|
|
|
});
|
|
|
|
|
expect(io.stdout()).toContain('Driver: lookml');
|
|
|
|
|
expect(io.stdout()).toContain('Repo: https://github.com/example/lookml');
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it('tests a MetricFlow connection via the nested metricflow block', async () => {
|
|
|
|
|
const projectDir = join(tempDir, 'project');
|
2026-05-14 17:39:31 +02:00
|
|
|
await initKtxProject({ projectDir });
|
2026-05-14 16:21:18 +02:00
|
|
|
await writeConnections(projectDir, {
|
|
|
|
|
mf_main: {
|
|
|
|
|
driver: 'metricflow',
|
|
|
|
|
metricflow: { repoUrl: 'https://github.com/example/metricflow' },
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
const testRepoConnection = vi.fn(async () => ({ ok: true as const }));
|
|
|
|
|
const io = makeIo();
|
|
|
|
|
|
|
|
|
|
await expect(
|
|
|
|
|
runKtxConnection({ command: 'test', projectDir, connectionId: 'mf_main' }, io.io, { testRepoConnection }),
|
|
|
|
|
).resolves.toBe(0);
|
|
|
|
|
expect(testRepoConnection).toHaveBeenCalledWith({
|
|
|
|
|
repoUrl: 'https://github.com/example/metricflow',
|
|
|
|
|
authToken: null,
|
|
|
|
|
});
|
|
|
|
|
expect(io.stdout()).toContain('Driver: metricflow');
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it('--all: prints a single coherent list with one row per connection', async () => {
|
|
|
|
|
const projectDir = join(tempDir, 'project');
|
2026-05-14 17:39:31 +02:00
|
|
|
await initKtxProject({ projectDir });
|
2026-05-14 16:21:18 +02:00
|
|
|
await writeConnections(projectDir, {
|
|
|
|
|
warehouse: { driver: 'sqlite' },
|
|
|
|
|
docs: { driver: 'notion', auth_token: 'secret_token', crawl_mode: 'all_accessible' }, // pragma: allowlist secret
|
|
|
|
|
});
|
|
|
|
|
const { connector } = nativeConnector('sqlite');
|
|
|
|
|
const createScanConnector = vi.fn(async () => connector);
|
|
|
|
|
const createNotionClient = vi.fn(async (): Promise<Pick<NotionClient, 'retrieveBotUser'>> => ({
|
|
|
|
|
retrieveBotUser: vi.fn(async () => ({ id: 'bot-1', name: 'Docs Bot' })),
|
|
|
|
|
}));
|
|
|
|
|
const io = makeIo();
|
|
|
|
|
|
|
|
|
|
await expect(
|
|
|
|
|
runKtxConnection({ command: 'test-all', projectDir }, io.io, { createScanConnector, createNotionClient }),
|
|
|
|
|
).resolves.toBe(0);
|
|
|
|
|
|
|
|
|
|
const out = stripAnsi(io.stdout());
|
|
|
|
|
expect(out).toContain('connection test --all');
|
|
|
|
|
expect(out).toMatch(/docs\s+notion\s+✓ ok\s+Bot: Docs Bot/);
|
|
|
|
|
expect(out).toMatch(/warehouse\s+sqlite\s+✓ ok\s+Status: ok/);
|
|
|
|
|
expect(out).toContain('2 tested');
|
|
|
|
|
expect(out).toContain('2 passed');
|
|
|
|
|
expect(out).not.toContain('failed');
|
|
|
|
|
expect(io.stderr()).toBe('');
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it('--all: marks failing connections, keeps passing ones, and returns non-zero', async () => {
|
|
|
|
|
const projectDir = join(tempDir, 'project');
|
2026-05-14 17:39:31 +02:00
|
|
|
await initKtxProject({ projectDir });
|
2026-05-14 16:21:18 +02:00
|
|
|
await writeConnections(projectDir, {
|
|
|
|
|
warehouse: { driver: 'sqlite' },
|
|
|
|
|
broken: { driver: 'sqlite' },
|
|
|
|
|
});
|
|
|
|
|
const okConnector = nativeConnector('sqlite').connector;
|
|
|
|
|
const failConnector = nativeConnector('sqlite', { success: false, error: 'database file is unreadable' }).connector;
|
|
|
|
|
const createScanConnector = vi.fn(async (_p, connectionId: string) =>
|
|
|
|
|
connectionId === 'broken' ? failConnector : okConnector,
|
|
|
|
|
);
|
|
|
|
|
const io = makeIo();
|
|
|
|
|
|
|
|
|
|
await expect(
|
|
|
|
|
runKtxConnection({ command: 'test-all', projectDir }, io.io, { createScanConnector }),
|
|
|
|
|
).resolves.toBe(1);
|
|
|
|
|
|
|
|
|
|
const out = stripAnsi(io.stdout());
|
|
|
|
|
expect(out).toMatch(/broken\s+sqlite\s+✗ failed\s+database file is unreadable/);
|
|
|
|
|
expect(out).toMatch(/warehouse\s+sqlite\s+✓ ok\s+Status: ok/);
|
|
|
|
|
expect(out).toContain('1 passed');
|
|
|
|
|
expect(out).toContain('1 failed');
|
|
|
|
|
expect(io.stderr()).toBe('');
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it('--all: shows an empty-state message when no connections are configured', async () => {
|
|
|
|
|
const projectDir = join(tempDir, 'project');
|
2026-05-14 17:39:31 +02:00
|
|
|
await initKtxProject({ projectDir });
|
2026-05-14 16:21:18 +02:00
|
|
|
const io = makeIo();
|
|
|
|
|
|
|
|
|
|
await expect(runKtxConnection({ command: 'test-all', projectDir }, io.io)).resolves.toBe(0);
|
|
|
|
|
|
|
|
|
|
const out = stripAnsi(io.stdout());
|
|
|
|
|
expect(out).toContain('connection test --all');
|
|
|
|
|
expect(out).toContain('No connections configured. Run `ktx setup` to add one.');
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it('rejects unknown drivers with a helpful error', async () => {
|
|
|
|
|
const projectDir = join(tempDir, 'project');
|
2026-05-14 17:39:31 +02:00
|
|
|
await initKtxProject({ projectDir });
|
2026-05-15 00:08:11 +02:00
|
|
|
await writeFile(
|
|
|
|
|
join(projectDir, 'ktx.yaml'),
|
|
|
|
|
'connections:\n mystery:\n driver: duckdb\n',
|
|
|
|
|
'utf-8',
|
|
|
|
|
);
|
2026-05-14 16:21:18 +02:00
|
|
|
const io = makeIo();
|
|
|
|
|
|
|
|
|
|
await expect(
|
|
|
|
|
runKtxConnection({ command: 'test', projectDir, connectionId: 'mystery' }, io.io),
|
|
|
|
|
).resolves.toBe(1);
|
2026-05-15 00:08:11 +02:00
|
|
|
expect(io.stderr()).toContain('connections.mystery.driver');
|
|
|
|
|
expect(io.stderr()).toContain('postgres');
|
2026-05-10 23:12:26 +02:00
|
|
|
});
|
|
|
|
|
});
|