This commit captures uncommitted modifications that accumulated in the working tree across multiple in-progress workstreams. It is a snapshot to clear the deck before sf v3 work begins; individual workstreams should land separately on top of this. Notable additions: - trace-collector.ts, traces.ts, src/tests/trace-export.test.ts — trace export plumbing - biome.json — Biome linter configuration - .gitignore — exclude native/npm/**/*.node compiled binaries The bulk of the diff is across src/resources/extensions/sf/ (301 files) and src/resources/extensions/sf/tests/ (277 files), reflecting the ongoing sf extension work. Specific feature commits should follow this snapshot rather than being archaeology'd out of it. The 76MB native/npm/linux-x64-gnu/forge_engine.node compiled binary was left out of the commit — it's now gitignored and built locally. Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
203 lines
5.8 KiB
TypeScript
203 lines
5.8 KiB
TypeScript
import { execFile } from "node:child_process";
|
|
import { existsSync } from "node:fs";
|
|
import { join } from "node:path";
|
|
import { pathToFileURL } from "node:url";
|
|
import type {
|
|
CaptureResolveRequest,
|
|
CaptureResolveResult,
|
|
CapturesData,
|
|
} from "../../web/lib/knowledge-captures-types.ts";
|
|
import { resolveBridgeRuntimeConfig } from "./bridge-service.ts";
|
|
import {
|
|
buildSubprocessPrefixArgs,
|
|
resolveSubprocessModule,
|
|
} from "./ts-subprocess-flags.ts";
|
|
|
|
const CAPTURES_MAX_BUFFER = 2 * 1024 * 1024;
|
|
const CAPTURES_MODULE_ENV = "SF_CAPTURES_MODULE";
|
|
|
|
function resolveTsLoaderPath(packageRoot: string): string {
|
|
return join(
|
|
packageRoot,
|
|
"src",
|
|
"resources",
|
|
"extensions",
|
|
"sf",
|
|
"tests",
|
|
"resolve-ts.mjs",
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Loads all capture entries via a child process. The child imports the upstream
|
|
* captures module, calls loadAllCaptures() and loadActionableCaptures(), and
|
|
* writes a CapturesData JSON to stdout.
|
|
*/
|
|
export async function collectCapturesData(
|
|
projectCwdOverride?: string,
|
|
): Promise<CapturesData> {
|
|
const config = resolveBridgeRuntimeConfig(undefined, projectCwdOverride);
|
|
const { packageRoot, projectCwd } = config;
|
|
|
|
const resolveTsLoader = resolveTsLoaderPath(packageRoot);
|
|
const moduleResolution = resolveSubprocessModule(
|
|
packageRoot,
|
|
"resources/extensions/sf/captures.ts",
|
|
);
|
|
const capturesModulePath = moduleResolution.modulePath;
|
|
|
|
if (
|
|
!moduleResolution.useCompiledJs &&
|
|
(!existsSync(resolveTsLoader) || !existsSync(capturesModulePath))
|
|
) {
|
|
throw new Error(
|
|
`captures data provider not found; checked=${resolveTsLoader},${capturesModulePath}`,
|
|
);
|
|
}
|
|
if (moduleResolution.useCompiledJs && !existsSync(capturesModulePath)) {
|
|
throw new Error(
|
|
`captures data provider not found; checked=${capturesModulePath}`,
|
|
);
|
|
}
|
|
|
|
const script = [
|
|
'const { pathToFileURL } = await import("node:url");',
|
|
`const mod = await import(pathToFileURL(process.env.${CAPTURES_MODULE_ENV}).href);`,
|
|
`const all = mod.loadAllCaptures(process.env.SF_CAPTURES_BASE);`,
|
|
'const pending = all.filter(c => c.status === "pending");',
|
|
`const actionable = mod.loadActionableCaptures(process.env.SF_CAPTURES_BASE);`,
|
|
"const result = { entries: all, pendingCount: pending.length, actionableCount: actionable.length };",
|
|
"process.stdout.write(JSON.stringify(result));",
|
|
].join(" ");
|
|
|
|
const prefixArgs = buildSubprocessPrefixArgs(
|
|
packageRoot,
|
|
moduleResolution,
|
|
pathToFileURL(resolveTsLoader).href,
|
|
);
|
|
|
|
return await new Promise<CapturesData>((resolveResult, reject) => {
|
|
execFile(
|
|
process.execPath,
|
|
[...prefixArgs, "--eval", script],
|
|
{
|
|
cwd: packageRoot,
|
|
env: {
|
|
...process.env,
|
|
[CAPTURES_MODULE_ENV]: capturesModulePath,
|
|
SF_CAPTURES_BASE: projectCwd,
|
|
},
|
|
maxBuffer: CAPTURES_MAX_BUFFER,
|
|
windowsHide: true,
|
|
},
|
|
(error, stdout, stderr) => {
|
|
if (error) {
|
|
reject(
|
|
new Error(
|
|
`captures data subprocess failed: ${stderr || error.message}`,
|
|
),
|
|
);
|
|
return;
|
|
}
|
|
|
|
try {
|
|
resolveResult(JSON.parse(stdout) as CapturesData);
|
|
} catch (parseError) {
|
|
reject(
|
|
new Error(
|
|
`captures data subprocess returned invalid JSON: ${parseError instanceof Error ? parseError.message : String(parseError)}`,
|
|
),
|
|
);
|
|
}
|
|
},
|
|
);
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Resolves (triages) a single capture by calling markCaptureResolved() in a
|
|
* child process. Returns { ok: true, captureId } on success.
|
|
*/
|
|
export async function resolveCaptureAction(
|
|
request: CaptureResolveRequest,
|
|
projectCwdOverride?: string,
|
|
): Promise<CaptureResolveResult> {
|
|
const config = resolveBridgeRuntimeConfig(undefined, projectCwdOverride);
|
|
const { packageRoot, projectCwd } = config;
|
|
|
|
const resolveTsLoader = resolveTsLoaderPath(packageRoot);
|
|
const moduleResolution = resolveSubprocessModule(
|
|
packageRoot,
|
|
"resources/extensions/sf/captures.ts",
|
|
);
|
|
const capturesModulePath = moduleResolution.modulePath;
|
|
|
|
if (
|
|
!moduleResolution.useCompiledJs &&
|
|
(!existsSync(resolveTsLoader) || !existsSync(capturesModulePath))
|
|
) {
|
|
throw new Error(
|
|
`captures data provider not found; checked=${resolveTsLoader},${capturesModulePath}`,
|
|
);
|
|
}
|
|
if (moduleResolution.useCompiledJs && !existsSync(capturesModulePath)) {
|
|
throw new Error(
|
|
`captures data provider not found; checked=${capturesModulePath}`,
|
|
);
|
|
}
|
|
|
|
const safeId = JSON.stringify(request.captureId);
|
|
const safeClassification = JSON.stringify(request.classification);
|
|
const safeResolution = JSON.stringify(request.resolution);
|
|
const safeRationale = JSON.stringify(request.rationale);
|
|
|
|
const script = [
|
|
'const { pathToFileURL } = await import("node:url");',
|
|
`const mod = await import(pathToFileURL(process.env.${CAPTURES_MODULE_ENV}).href);`,
|
|
`mod.markCaptureResolved(process.env.SF_CAPTURES_BASE, ${safeId}, ${safeClassification}, ${safeResolution}, ${safeRationale});`,
|
|
`process.stdout.write(JSON.stringify({ ok: true, captureId: ${safeId} }));`,
|
|
].join(" ");
|
|
|
|
const prefixArgs = buildSubprocessPrefixArgs(
|
|
packageRoot,
|
|
moduleResolution,
|
|
pathToFileURL(resolveTsLoader).href,
|
|
);
|
|
|
|
return await new Promise<CaptureResolveResult>((resolveResult, reject) => {
|
|
execFile(
|
|
process.execPath,
|
|
[...prefixArgs, "--eval", script],
|
|
{
|
|
cwd: packageRoot,
|
|
env: {
|
|
...process.env,
|
|
[CAPTURES_MODULE_ENV]: capturesModulePath,
|
|
SF_CAPTURES_BASE: projectCwd,
|
|
},
|
|
maxBuffer: CAPTURES_MAX_BUFFER,
|
|
windowsHide: true,
|
|
},
|
|
(error, stdout, stderr) => {
|
|
if (error) {
|
|
reject(
|
|
new Error(
|
|
`capture resolve subprocess failed: ${stderr || error.message}`,
|
|
),
|
|
);
|
|
return;
|
|
}
|
|
|
|
try {
|
|
resolveResult(JSON.parse(stdout) as CaptureResolveResult);
|
|
} catch (parseError) {
|
|
reject(
|
|
new Error(
|
|
`capture resolve subprocess returned invalid JSON: ${parseError instanceof Error ? parseError.message : String(parseError)}`,
|
|
),
|
|
);
|
|
}
|
|
},
|
|
);
|
|
});
|
|
}
|