open-design/apps/daemon/tests/diagnostics-export.test.ts
Amy 1c2a1c4459
Add launch review regression coverage and stabilize daemon tests (#3207)
* Add launch review E2E regression coverage

* Harden daemon launch review regressions

* Stabilize daemon runtime tests

* fix(tests): restore e2e preflight typing

Generated-By: looper 0.8.1 (runner=fixer, agent=codex)

* fix(tests): make fake plugin runtime ESM-safe

Generated-By: looper 0.8.1 (runner=fixer, agent=codex)

* Stabilize e2e fake agent and regression tests

* fix(tests): repair fake agent cjs runtime

Generated-By: looper 0.8.1 (runner=fixer, agent=codex)

* fix(review): harden plugin authoring checks

Generated-By: looper 0.9.2 (runner=fixer, agent=codex)

* fix(tests): bind plugin authoring run to seeded conversation

Generated-By: looper 0.9.2 (runner=fixer, agent=codex)
2026-05-29 02:39:33 +00:00

172 lines
7.3 KiB
TypeScript

import { mkdir, rm, writeFile } from 'node:fs/promises';
import { tmpdir } from 'node:os';
import { dirname, join } from 'node:path';
import { randomUUID } from 'node:crypto';
import { describe, expect, it } from 'vitest';
import JSZip from 'jszip';
import {
APP_KEYS,
SIDECAR_MODES,
SIDECAR_SOURCES,
type SidecarStamp,
} from '@open-design/sidecar-proto';
import type { SidecarRuntimeContext } from '@open-design/sidecar';
import {
STANDALONE_LAUNCH_WARNING,
createDiagnosticsExportHandler,
} from '../src/diagnostics-export.js';
interface MockResponse {
status(code: number): MockResponse;
setHeader(name: string, value: string): MockResponse;
end(payload: Buffer): void;
json(payload: unknown): void;
capturedStatus?: number;
capturedPayload?: Buffer;
capturedJson?: unknown;
}
function mockResponse(): MockResponse {
const res: MockResponse = {
status(code) { res.capturedStatus = code; return res; },
setHeader() { return res; },
end(payload) { res.capturedPayload = payload; },
json(payload) { res.capturedJson = payload; },
};
return res;
}
interface DiagnosticsManifestFile {
name: string;
}
describe('diagnostics export handler — non-sidecar launch', () => {
// Reviewer-requested regression spec: `runDaemonCliStartup()` calls
// `startDaemonRuntime()` without a runtime context, so plain `od` users
// hit the diagnostics handler with `options.runtime == null`. The bundle
// must still produce a valid zip AND surface a manifest warning that
// file-based logs were not captured, so the operator can tell the
// diff between "no logs because plain launch" and "no logs because
// something genuinely broke."
it('emits a standalone-launch warning when runtime is null', async () => {
const handler = createDiagnosticsExportHandler({ runtime: null, projectRoot: '/tmp/test-project' });
const res = mockResponse();
// Express RequestHandler signature wants three args; the handler only
// reads `res`, so casting through `unknown` keeps the test focused.
await handler({} as never, res as never, () => undefined);
expect(res.capturedStatus).toBe(200);
expect(res.capturedPayload).toBeInstanceOf(Buffer);
const zip = await JSZip.loadAsync(res.capturedPayload!);
const manifestRaw = await zip.file('summary/manifest.json')!.async('string');
const manifest = JSON.parse(manifestRaw) as {
warnings: string[];
files: DiagnosticsManifestFile[];
};
expect(manifest.warnings).toContain(STANDALONE_LAUNCH_WARNING);
// Standalone launches intentionally omit sidecar-managed daemon/web/desktop
// log files, but real developer machines may still contribute matching
// macOS crash reports from /Library/Logs/DiagnosticReports. Keep the test
// focused on the contract that no sidecar log files are bundled.
expect(
manifest.files.filter((file) => file.name.startsWith('logs/')),
).toEqual([]);
});
});
describe('diagnostics export handler — packaged (runtime) layout', () => {
// Regression for the namespaceRoot off-by-one that left every packaged
// bundle without daemon/web logs (the agent-run flow lives in the daemon
// log). In packaged builds the orchestrator launches each child with
// `base = <namespaceRoot>/runtime` while the logs live a level up at
// `<namespaceRoot>/logs`. The old `resolveNamespaceRoot(base, namespace)`
// resolved the daemon log to `<namespaceRoot>/runtime/<namespace>/logs/...`
// → ENOENT, so the bundle silently captured nothing.
it('captures the daemon log from the real <namespaceRoot>/logs tree', async () => {
const root = join(tmpdir(), `od-diag-${randomUUID()}`);
const namespaceRoot = join(root, 'namespaces', 'release-stable');
const daemonLogPath = join(namespaceRoot, 'logs', APP_KEYS.DAEMON, 'latest.log');
const marker = 'DAEMON-LOG-MARKER critique runId=rc100-poster';
try {
await mkdir(dirname(daemonLogPath), { recursive: true });
await writeFile(daemonLogPath, `${marker}\n`, 'utf8');
const runtime: SidecarRuntimeContext<SidecarStamp> = {
app: APP_KEYS.DAEMON,
// packaged launches children with base == <namespaceRoot>/runtime
base: join(namespaceRoot, 'runtime'),
ipc: '/tmp/od-diag-test-daemon.sock',
mode: SIDECAR_MODES.RUNTIME,
namespace: 'release-stable',
source: SIDECAR_SOURCES.PACKAGED,
};
const handler = createDiagnosticsExportHandler({ runtime, projectRoot: '/tmp/test-project' });
const res = mockResponse();
await handler({} as never, res as never, () => undefined);
expect(res.capturedStatus).toBe(200);
const zip = await JSZip.loadAsync(res.capturedPayload!);
// The log must be present with its real contents, not a missing-file
// placeholder.
const daemonEntry = zip.file('logs/daemon/latest.log');
expect(daemonEntry).not.toBeNull();
expect(await daemonEntry!.async('string')).toContain(marker);
const manifest = JSON.parse(await zip.file('summary/manifest.json')!.async('string')) as {
files: { name: string; bytes: number; error?: string }[];
};
const daemonFile = manifest.files.find((f) => f.name === 'logs/daemon/latest.log');
expect(daemonFile?.error).toBeUndefined();
expect(daemonFile?.bytes ?? 0).toBeGreaterThan(0);
} finally {
await rm(root, { recursive: true, force: true });
}
});
it('reports missing packaged log files under logical log paths without duplicating runtime segments', async () => {
const root = join(tmpdir(), `od-diag-missing-${randomUUID()}`);
const namespaceRoot = join(root, 'namespaces', 'release-beta');
const daemonLogPath = join(namespaceRoot, 'logs', APP_KEYS.DAEMON, 'latest.log');
try {
await mkdir(dirname(daemonLogPath), { recursive: true });
await writeFile(daemonLogPath, 'daemon ok\n', 'utf8');
const runtime: SidecarRuntimeContext<SidecarStamp> = {
app: APP_KEYS.DAEMON,
base: join(namespaceRoot, 'runtime'),
ipc: '/tmp/od-diag-missing.sock',
mode: SIDECAR_MODES.RUNTIME,
namespace: 'release-beta',
source: SIDECAR_SOURCES.PACKAGED,
};
const handler = createDiagnosticsExportHandler({ runtime, projectRoot: '/tmp/test-project' });
const res = mockResponse();
await handler({} as never, res as never, () => undefined);
expect(res.capturedStatus).toBe(200);
const zip = await JSZip.loadAsync(res.capturedPayload!);
const manifest = JSON.parse(await zip.file('summary/manifest.json')!.async('string')) as {
files: Array<{ name: string; bytes?: number; error?: string }>;
};
const fileNames = manifest.files.map((file) => file.name);
expect(fileNames).toContain('logs/daemon/latest.log');
expect(fileNames).toContain('logs/web/latest.log');
expect(fileNames).toContain('logs/desktop/latest.log');
expect(fileNames.some((name) => name.includes('runtime/release-beta/logs'))).toBe(false);
const webLog = manifest.files.find((file) => file.name === 'logs/web/latest.log');
const desktopLog = manifest.files.find((file) => file.name === 'logs/desktop/latest.log');
expect(webLog?.error).toBeTruthy();
expect(desktopLog?.error).toBeTruthy();
} finally {
await rm(root, { recursive: true, force: true });
}
});
});