2026-05-10 23:12:26 +02:00
|
|
|
import assert from 'node:assert/strict';
|
|
|
|
|
import { readFile } from 'node:fs/promises';
|
|
|
|
|
import { dirname } from 'node:path';
|
|
|
|
|
import { describe, it } from 'node:test';
|
|
|
|
|
import {
|
|
|
|
|
buildOrbitReportArgv,
|
|
|
|
|
buildOrbitScanArgv,
|
|
|
|
|
defaultOrbitVerificationProjectDir,
|
|
|
|
|
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',
|
|
|
|
|
],
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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-10 23:51:24 +02:00
|
|
|
it('builds the current KTX launcher arguments for scan and JSON report commands', () => {
|
2026-05-10 23:12:26 +02:00
|
|
|
assert.deepEqual(buildOrbitScanArgv({ connectionId: 'orbit', projectDir: '/tmp/orbit-project' }), [
|
|
|
|
|
'dev',
|
|
|
|
|
'scan',
|
|
|
|
|
'orbit',
|
|
|
|
|
'--enrich',
|
|
|
|
|
'--project-dir',
|
|
|
|
|
'/tmp/orbit-project',
|
|
|
|
|
]);
|
|
|
|
|
assert.deepEqual(buildOrbitReportArgv({ projectDir: '/tmp/orbit-project', runId: 'scan-orbit-1' }), [
|
|
|
|
|
'dev',
|
|
|
|
|
'scan',
|
|
|
|
|
'report',
|
|
|
|
|
'--json',
|
|
|
|
|
'--project-dir',
|
|
|
|
|
'/tmp/orbit-project',
|
|
|
|
|
'scan-orbit-1',
|
|
|
|
|
]);
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it('uses the checked-in Orbit verification project by default', async () => {
|
|
|
|
|
const calls = [];
|
|
|
|
|
const envs = [];
|
|
|
|
|
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-10 23:51:24 +02:00
|
|
|
runWorkspaceKtx: async (argv, options) => {
|
2026-05-10 23:12:26 +02:00
|
|
|
calls.push(argv);
|
|
|
|
|
envs.push(options.env);
|
|
|
|
|
if (argv[2] === 'report') {
|
|
|
|
|
options.stdout.write(successReportJson());
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
2026-05-10 23:51:24 +02:00
|
|
|
options.stdout.write('KTX scan completed\nRun: scan-orbit-1\nConnection: orbit\n');
|
2026-05-10 23:12:26 +02:00
|
|
|
return 0;
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
assert.equal(result.status, 'success');
|
|
|
|
|
assert.deepEqual(calls, [
|
|
|
|
|
['dev', 'scan', 'orbit', '--enrich', '--project-dir', defaultProjectDir],
|
|
|
|
|
['dev', 'scan', 'report', '--json', '--project-dir', defaultProjectDir, 'scan-orbit-1'],
|
|
|
|
|
]);
|
|
|
|
|
assert.equal(envs[0].GIT_CEILING_DIRECTORIES, dirname(defaultProjectDir));
|
|
|
|
|
assert.equal(envs[1].GIT_CEILING_DIRECTORIES, dirname(defaultProjectDir));
|
|
|
|
|
assert.equal(writes.length, 1);
|
|
|
|
|
assert.match(writes[0].content, new RegExp(defaultProjectDir.replace(/[.*+?^${}()|[\]\\]/g, '\\$&')));
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
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-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-10 23:51:24 +02:00
|
|
|
scanCommand: 'pnpm run ktx -- dev scan orbit --enrich --project-dir /tmp/orbit-project',
|
|
|
|
|
reportCommand: 'pnpm run ktx -- dev scan report --json --project-dir /tmp/orbit-project scan-orbit-1',
|
2026-05-10 23:12:26 +02:00
|
|
|
scanExitCode: 0,
|
|
|
|
|
reportExitCode: 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-10 23:51:24 +02:00
|
|
|
scanCommand: 'pnpm run ktx -- dev scan orbit --enrich --project-dir /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/);
|
|
|
|
|
assert.match(markdown, /Orbit verification was not executed because the current local Orbit scan command failed/);
|
|
|
|
|
assert.doesNotMatch(markdown, /scan\.enrichment\.mode is required/);
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it('runs scan then JSON report and writes success Markdown', async () => {
|
|
|
|
|
const calls = [];
|
|
|
|
|
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-10 23:51:24 +02:00
|
|
|
runWorkspaceKtx: async (argv, options) => {
|
2026-05-10 23:12:26 +02:00
|
|
|
calls.push(argv);
|
|
|
|
|
if (argv[2] === 'report') {
|
|
|
|
|
options.stdout.write(successReportJson());
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
2026-05-10 23:51:24 +02:00
|
|
|
options.stdout.write('KTX scan completed\nRun: scan-orbit-1\nConnection: orbit\n');
|
2026-05-10 23:12:26 +02:00
|
|
|
return 0;
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
assert.equal(result.status, 'success');
|
|
|
|
|
assert.deepEqual(calls, [
|
|
|
|
|
['dev', 'scan', 'orbit', '--enrich', '--project-dir', '/tmp/orbit-project'],
|
|
|
|
|
['dev', 'scan', 'report', '--json', '--project-dir', '/tmp/orbit-project', 'scan-orbit-1'],
|
|
|
|
|
]);
|
|
|
|
|
assert.equal(writes.length, 1);
|
|
|
|
|
assert.equal(writes[0].path, '/tmp/orbit-report.md');
|
|
|
|
|
assert.match(writes[0].content, /Accepted: 14/);
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it('writes blocked Markdown when the scan command fails before a run id exists', async () => {
|
|
|
|
|
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-10 23:51:24 +02:00
|
|
|
runWorkspaceKtx: async (_argv, options) => {
|
2026-05-10 23:12:26 +02:00
|
|
|
options.stderr.write('Connection "orbit" was not found\n');
|
|
|
|
|
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/);
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it('runs the workspace launcher in buffered mode so real scan errors are captured', async () => {
|
|
|
|
|
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-10 23:51:24 +02:00
|
|
|
runWorkspaceKtx: async (_argv, options) => {
|
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
|
|
|
});
|
|
|
|
|
});
|