2026-05-10 23:12:26 +02:00
|
|
|
import assert from 'node:assert/strict';
|
|
|
|
|
import { readFile } from 'node:fs/promises';
|
|
|
|
|
import { describe, it } from 'node:test';
|
|
|
|
|
import {
|
2026-05-14 01:43:06 +02:00
|
|
|
buildOrbitScanArgs,
|
2026-05-10 23:12:26 +02:00
|
|
|
defaultOrbitVerificationProjectDir,
|
2026-05-13 12:00:08 +02:00
|
|
|
extractReportPath,
|
2026-05-10 23:12:26 +02:00
|
|
|
extractRunId,
|
|
|
|
|
formatOrbitVerificationMarkdown,
|
|
|
|
|
runOrbitVerification,
|
|
|
|
|
} from './relationship-orbit-verification.mjs';
|
|
|
|
|
|
|
|
|
|
function successReportJson() {
|
|
|
|
|
return JSON.stringify({
|
|
|
|
|
runId: 'scan-orbit-1',
|
|
|
|
|
connectionId: 'orbit',
|
|
|
|
|
mode: 'enriched',
|
|
|
|
|
syncId: '2026-05-07-100000-scan-enriched-1',
|
|
|
|
|
relationships: {
|
|
|
|
|
accepted: 14,
|
|
|
|
|
review: 8,
|
|
|
|
|
rejected: 91,
|
|
|
|
|
skipped: 0,
|
|
|
|
|
},
|
|
|
|
|
enrichment: {
|
|
|
|
|
deterministicRelationships: 'completed',
|
|
|
|
|
statisticalValidation: 'completed',
|
|
|
|
|
llmRelationshipValidation: 'skipped',
|
|
|
|
|
},
|
|
|
|
|
warnings: [
|
|
|
|
|
{
|
|
|
|
|
code: 'scan_enrichment_backend_not_configured',
|
|
|
|
|
message:
|
|
|
|
|
'Skipping description and embedding enrichment because scan.enrichment.mode is not configured; relationship discovery still ran.',
|
|
|
|
|
recoverable: true,
|
|
|
|
|
},
|
|
|
|
|
],
|
|
|
|
|
artifactPaths: {
|
|
|
|
|
reportPath: 'raw-sources/orbit/live-database/2026-05-07-100000-scan-enriched-1/reports/scan-report.json',
|
|
|
|
|
rawSourcesDir: 'raw-sources/orbit/live-database/2026-05-07-100000-scan-enriched-1',
|
|
|
|
|
manifestShards: ['semantic-layer/orbit/_schema/orbit_analytics.yaml'],
|
|
|
|
|
enrichmentArtifacts: [
|
|
|
|
|
'raw-sources/orbit/live-database/2026-05-07-100000-scan-enriched-1/enrichment/relationships.json',
|
|
|
|
|
'raw-sources/orbit/live-database/2026-05-07-100000-scan-enriched-1/enrichment/relationship-profile.json',
|
|
|
|
|
'raw-sources/orbit/live-database/2026-05-07-100000-scan-enriched-1/enrichment/relationship-diagnostics.json',
|
|
|
|
|
],
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
2026-05-14 01:43:06 +02:00
|
|
|
function successfulRunKtxScan(calls = []) {
|
|
|
|
|
return async (args, io) => {
|
|
|
|
|
calls.push(args);
|
|
|
|
|
io.stdout.write('KTX scan completed\nRun: scan-orbit-1\nConnection: orbit\n Report: reports/scan-report.json\n');
|
|
|
|
|
return 0;
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
2026-05-10 23:12:26 +02:00
|
|
|
describe('relationship Orbit verification helper', () => {
|
2026-05-10 23:51:24 +02:00
|
|
|
it('exposes the Orbit verification command from the KTX workspace package', async () => {
|
2026-05-10 23:12:26 +02:00
|
|
|
const packageJson = JSON.parse(await readFile(new URL('../package.json', import.meta.url), 'utf8'));
|
|
|
|
|
|
|
|
|
|
assert.equal(
|
|
|
|
|
packageJson.scripts['relationships:verify-orbit'],
|
|
|
|
|
'node scripts/relationship-orbit-verification.mjs',
|
|
|
|
|
);
|
|
|
|
|
});
|
|
|
|
|
|
2026-05-14 01:43:06 +02:00
|
|
|
it('builds the internal relationship scan arguments', () => {
|
|
|
|
|
assert.deepEqual(buildOrbitScanArgs({ connectionId: 'orbit', projectDir: '/tmp/orbit-project' }), {
|
|
|
|
|
command: 'run',
|
|
|
|
|
projectDir: '/tmp/orbit-project',
|
|
|
|
|
connectionId: 'orbit',
|
|
|
|
|
mode: 'relationships',
|
|
|
|
|
detectRelationships: true,
|
|
|
|
|
dryRun: false,
|
|
|
|
|
});
|
2026-05-10 23:12:26 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it('uses the checked-in Orbit verification project by default', async () => {
|
2026-05-14 01:43:06 +02:00
|
|
|
const scanCalls = [];
|
2026-05-10 23:12:26 +02:00
|
|
|
const writes = [];
|
|
|
|
|
const defaultProjectDir = defaultOrbitVerificationProjectDir();
|
|
|
|
|
|
|
|
|
|
const result = await runOrbitVerification({
|
|
|
|
|
reportPath: '/tmp/orbit-report.md',
|
|
|
|
|
now: () => new Date('2026-05-07T10:00:00.000Z'),
|
|
|
|
|
mkdir: async () => {},
|
|
|
|
|
writeFile: async (path, content) => {
|
|
|
|
|
writes.push({ path, content });
|
|
|
|
|
},
|
2026-05-14 01:43:06 +02:00
|
|
|
runKtxScan: successfulRunKtxScan(scanCalls),
|
2026-05-13 12:00:08 +02:00
|
|
|
readFile: async () => successReportJson(),
|
2026-05-10 23:12:26 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
|
|
assert.equal(result.status, 'success');
|
2026-05-14 01:43:06 +02:00
|
|
|
assert.deepEqual(scanCalls, [
|
|
|
|
|
{
|
|
|
|
|
command: 'run',
|
|
|
|
|
projectDir: defaultProjectDir,
|
|
|
|
|
connectionId: 'orbit',
|
|
|
|
|
mode: 'relationships',
|
|
|
|
|
detectRelationships: true,
|
|
|
|
|
dryRun: false,
|
|
|
|
|
},
|
2026-05-10 23:12:26 +02:00
|
|
|
]);
|
|
|
|
|
assert.equal(writes.length, 1);
|
|
|
|
|
assert.match(writes[0].content, new RegExp(defaultProjectDir.replace(/[.*+?^${}()|[\]\\]/g, '\\$&')));
|
|
|
|
|
});
|
|
|
|
|
|
2026-05-12 11:21:37 +02:00
|
|
|
it('uses KTX_PROJECT_DIR for the Orbit verification project override', async () => {
|
|
|
|
|
const previousProjectDir = process.env.KTX_PROJECT_DIR;
|
2026-05-14 01:43:06 +02:00
|
|
|
const scanCalls = [];
|
2026-05-12 11:21:37 +02:00
|
|
|
|
|
|
|
|
try {
|
|
|
|
|
process.env.KTX_PROJECT_DIR = '/tmp/orbit-project-from-env';
|
|
|
|
|
|
|
|
|
|
const result = await runOrbitVerification({
|
|
|
|
|
reportPath: '/tmp/orbit-report.md',
|
|
|
|
|
now: () => new Date('2026-05-07T10:00:00.000Z'),
|
|
|
|
|
mkdir: async () => {},
|
|
|
|
|
writeFile: async () => {},
|
2026-05-14 01:43:06 +02:00
|
|
|
runKtxScan: successfulRunKtxScan(scanCalls),
|
2026-05-13 12:00:08 +02:00
|
|
|
readFile: async () => successReportJson(),
|
2026-05-12 11:21:37 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
|
|
assert.equal(result.projectDir, '/tmp/orbit-project-from-env');
|
2026-05-14 01:43:06 +02:00
|
|
|
assert.deepEqual(scanCalls, [
|
|
|
|
|
{
|
|
|
|
|
command: 'run',
|
|
|
|
|
projectDir: '/tmp/orbit-project-from-env',
|
|
|
|
|
connectionId: 'orbit',
|
|
|
|
|
mode: 'relationships',
|
|
|
|
|
detectRelationships: true,
|
|
|
|
|
dryRun: false,
|
|
|
|
|
},
|
2026-05-12 11:21:37 +02:00
|
|
|
]);
|
|
|
|
|
} finally {
|
|
|
|
|
if (previousProjectDir === undefined) {
|
|
|
|
|
delete process.env.KTX_PROJECT_DIR;
|
|
|
|
|
} else {
|
|
|
|
|
process.env.KTX_PROJECT_DIR = previousProjectDir;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
2026-05-10 23:12:26 +02:00
|
|
|
it('extracts the run id from human scan output', () => {
|
2026-05-10 23:51:24 +02:00
|
|
|
assert.equal(extractRunId(`KTX scan completed\nStatus: done\nRun: scan-orbit-1\nConnection: orbit\n`), 'scan-orbit-1');
|
|
|
|
|
assert.equal(extractRunId('KTX scan completed without a run line\n'), null);
|
2026-05-13 12:00:08 +02:00
|
|
|
assert.equal(extractReportPath('Artifacts\n Report: reports/scan-report.json\n'), 'reports/scan-report.json');
|
2026-05-10 23:12:26 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it('formats successful Orbit verification evidence from the JSON report', () => {
|
|
|
|
|
const markdown = formatOrbitVerificationMarkdown({
|
|
|
|
|
status: 'success',
|
|
|
|
|
date: '2026-05-07',
|
|
|
|
|
connectionId: 'orbit',
|
|
|
|
|
projectDir: '/tmp/orbit-project',
|
2026-05-14 01:43:06 +02:00
|
|
|
scanCommand: 'internal runKtxScan connection=orbit mode=relationships projectDir=/tmp/orbit-project',
|
2026-05-13 12:00:08 +02:00
|
|
|
reportPath: '/tmp/orbit-project/reports/scan-report.json',
|
2026-05-10 23:12:26 +02:00
|
|
|
scanExitCode: 0,
|
2026-05-10 23:51:24 +02:00
|
|
|
scanStdout: 'KTX scan completed\nRun: scan-orbit-1\n',
|
2026-05-10 23:12:26 +02:00
|
|
|
scanStderr: '',
|
|
|
|
|
report: JSON.parse(successReportJson()),
|
|
|
|
|
});
|
|
|
|
|
|
2026-05-10 23:51:24 +02:00
|
|
|
assert.match(markdown, /# KTX Relationship Discovery Orbit Verification/);
|
2026-05-10 23:12:26 +02:00
|
|
|
assert.match(markdown, /Outcome/);
|
|
|
|
|
assert.match(markdown, /Exit code: 0/);
|
|
|
|
|
assert.match(markdown, /Accepted: 14/);
|
|
|
|
|
assert.match(markdown, /Review: 8/);
|
|
|
|
|
assert.match(markdown, /Rejected: 91/);
|
|
|
|
|
assert.match(markdown, /semantic-layer\/orbit\/_schema\/orbit_analytics\.yaml/);
|
|
|
|
|
assert.match(markdown, /relationship-diagnostics\.json/);
|
|
|
|
|
assert.match(markdown, /scan_enrichment_backend_not_configured/);
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it('formats blocked Orbit verification evidence from the current failing command', () => {
|
|
|
|
|
const markdown = formatOrbitVerificationMarkdown({
|
|
|
|
|
status: 'blocked',
|
|
|
|
|
date: '2026-05-07',
|
|
|
|
|
connectionId: 'orbit',
|
|
|
|
|
projectDir: '/tmp/orbit-project',
|
2026-05-14 01:43:06 +02:00
|
|
|
scanCommand: 'internal runKtxScan connection=orbit mode=relationships projectDir=/tmp/orbit-project',
|
2026-05-10 23:12:26 +02:00
|
|
|
scanExitCode: 1,
|
|
|
|
|
blocker: 'Connection "orbit" was not found',
|
|
|
|
|
scanStdout: '',
|
|
|
|
|
scanStderr: 'Connection "orbit" was not found\n',
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
assert.match(markdown, /Exit code: 1/);
|
|
|
|
|
assert.match(markdown, /Connection "orbit" was not found/);
|
2026-05-14 01:43:06 +02:00
|
|
|
assert.match(markdown, /Orbit verification was not executed because the current local Orbit relationship scan failed/);
|
2026-05-10 23:12:26 +02:00
|
|
|
assert.doesNotMatch(markdown, /scan\.enrichment\.mode is required/);
|
|
|
|
|
});
|
|
|
|
|
|
2026-05-13 12:00:08 +02:00
|
|
|
it('runs scan then reads the report artifact and writes success Markdown', async () => {
|
2026-05-14 01:43:06 +02:00
|
|
|
const scanCalls = [];
|
2026-05-10 23:12:26 +02:00
|
|
|
const writes = [];
|
|
|
|
|
const result = await runOrbitVerification({
|
|
|
|
|
connectionId: 'orbit',
|
|
|
|
|
projectDir: '/tmp/orbit-project',
|
|
|
|
|
reportPath: '/tmp/orbit-report.md',
|
|
|
|
|
now: () => new Date('2026-05-07T10:00:00.000Z'),
|
|
|
|
|
mkdir: async () => {},
|
|
|
|
|
writeFile: async (path, content) => {
|
|
|
|
|
writes.push({ path, content });
|
|
|
|
|
},
|
2026-05-14 01:43:06 +02:00
|
|
|
runKtxScan: successfulRunKtxScan(scanCalls),
|
2026-05-13 12:00:08 +02:00
|
|
|
readFile: async () => successReportJson(),
|
2026-05-10 23:12:26 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
|
|
assert.equal(result.status, 'success');
|
2026-05-14 01:43:06 +02:00
|
|
|
assert.deepEqual(scanCalls, [
|
|
|
|
|
{
|
|
|
|
|
command: 'run',
|
|
|
|
|
projectDir: '/tmp/orbit-project',
|
|
|
|
|
connectionId: 'orbit',
|
|
|
|
|
mode: 'relationships',
|
|
|
|
|
detectRelationships: true,
|
|
|
|
|
dryRun: false,
|
|
|
|
|
},
|
2026-05-10 23:12:26 +02:00
|
|
|
]);
|
|
|
|
|
assert.equal(writes.length, 1);
|
|
|
|
|
assert.equal(writes[0].path, '/tmp/orbit-report.md');
|
|
|
|
|
assert.match(writes[0].content, /Accepted: 14/);
|
|
|
|
|
});
|
|
|
|
|
|
2026-05-14 01:43:06 +02:00
|
|
|
it('writes blocked Markdown when the internal scan fails before a run id exists', async () => {
|
2026-05-10 23:12:26 +02:00
|
|
|
const writes = [];
|
|
|
|
|
const result = await runOrbitVerification({
|
|
|
|
|
connectionId: 'orbit',
|
|
|
|
|
projectDir: '/tmp/orbit-project',
|
|
|
|
|
reportPath: '/tmp/orbit-report.md',
|
|
|
|
|
now: () => new Date('2026-05-07T10:00:00.000Z'),
|
|
|
|
|
mkdir: async () => {},
|
|
|
|
|
writeFile: async (path, content) => {
|
|
|
|
|
writes.push({ path, content });
|
|
|
|
|
},
|
2026-05-14 01:43:06 +02:00
|
|
|
runKtxScan: async (_args, io) => {
|
|
|
|
|
io.stderr.write('Connection "orbit" was not found\n');
|
2026-05-10 23:12:26 +02:00
|
|
|
return 1;
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
assert.equal(result.status, 'blocked');
|
|
|
|
|
assert.equal(result.scanExitCode, 1);
|
|
|
|
|
assert.equal(writes.length, 1);
|
|
|
|
|
assert.match(writes[0].content, /Connection "orbit" was not found/);
|
|
|
|
|
});
|
|
|
|
|
|
2026-05-14 01:43:06 +02:00
|
|
|
it('runs the workspace launcher in buffered mode when preparing the internal scan module', async () => {
|
2026-05-10 23:12:26 +02:00
|
|
|
let sawExecFile = false;
|
|
|
|
|
const result = await runOrbitVerification({
|
|
|
|
|
connectionId: 'orbit',
|
|
|
|
|
projectDir: '/tmp/orbit-project',
|
|
|
|
|
reportPath: '/tmp/orbit-report.md',
|
|
|
|
|
now: () => new Date('2026-05-07T10:00:00.000Z'),
|
|
|
|
|
mkdir: async () => {},
|
|
|
|
|
writeFile: async () => {},
|
|
|
|
|
execFile: async () => ({ stdout: '', stderr: '' }),
|
2026-05-14 01:43:06 +02:00
|
|
|
runWorkspaceKtx: async (argv, options) => {
|
|
|
|
|
assert.deepEqual(argv, ['--version']);
|
2026-05-10 23:12:26 +02:00
|
|
|
sawExecFile = typeof options.execFile === 'function';
|
2026-05-10 23:51:24 +02:00
|
|
|
options.stderr.write('ENOENT: no such file or directory, open \'/tmp/orbit-project/ktx.yaml\'\n');
|
2026-05-10 23:12:26 +02:00
|
|
|
return 1;
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
assert.equal(sawExecFile, true);
|
2026-05-10 23:51:24 +02:00
|
|
|
assert.equal(result.blocker, "ENOENT: no such file or directory, open '/tmp/orbit-project/ktx.yaml'");
|
2026-05-10 23:12:26 +02:00
|
|
|
});
|
|
|
|
|
});
|