Merge remote-tracking branch 'origin/codex/production-intelligence-terminal' into codex/issue-2-stale-alerts
This commit is contained in:
@@ -1,10 +1,12 @@
|
||||
import test from 'node:test';
|
||||
import assert from 'node:assert/strict';
|
||||
import { readFileSync } from 'node:fs';
|
||||
import { safeFetch, safeFetchText, getFetchMetrics } from '../apis/utils/fetch.mjs';
|
||||
import { formatStaleAlert, shouldSendStaleAlert } from '../lib/stale-alerts.mjs';
|
||||
|
||||
test('safeFetch reports HTML as degraded JSON response', async () => {
|
||||
const originalFetch = globalThis.fetch;
|
||||
const source = 'unit-html-once';
|
||||
globalThis.fetch = async () => ({
|
||||
ok: true,
|
||||
status: 200,
|
||||
@@ -12,9 +14,72 @@ test('safeFetch reports HTML as degraded JSON response', async () => {
|
||||
text: async () => '<html>not json</html>',
|
||||
});
|
||||
try {
|
||||
const data = await safeFetch('https://example.test/json', { retries: 0, source: 'unit' });
|
||||
const data = await safeFetch('https://example.test/json', { retries: 0, source });
|
||||
assert.match(data.error, /Expected JSON/);
|
||||
assert.ok(getFetchMetrics().bySource.unit.requests >= 1);
|
||||
const bucket = getFetchMetrics().bySource[source];
|
||||
assert.equal(bucket.requests, 1);
|
||||
assert.equal(bucket.ok, 0);
|
||||
assert.equal(bucket.failed, 1);
|
||||
assert.equal(bucket.lastStatus, 200);
|
||||
} finally {
|
||||
globalThis.fetch = originalFetch;
|
||||
}
|
||||
});
|
||||
|
||||
test('safeFetch records HTTP failure once with status and bytes', async () => {
|
||||
const originalFetch = globalThis.fetch;
|
||||
const source = 'unit-http-failure-once';
|
||||
globalThis.fetch = async () => ({
|
||||
ok: false,
|
||||
status: 503,
|
||||
headers: { get: () => 'application/json' },
|
||||
text: async () => 'service unavailable',
|
||||
});
|
||||
try {
|
||||
const data = await safeFetch('https://example.test/fail', { retries: 0, source });
|
||||
assert.match(data.error, /HTTP 503/);
|
||||
const bucket = getFetchMetrics().bySource[source];
|
||||
assert.equal(bucket.requests, 1);
|
||||
assert.equal(bucket.ok, 0);
|
||||
assert.equal(bucket.failed, 1);
|
||||
assert.equal(bucket.lastStatus, 503);
|
||||
assert.equal(bucket.bytes, 'service unavailable'.length);
|
||||
assert.match(bucket.lastError, /HTTP 503/);
|
||||
} finally {
|
||||
globalThis.fetch = originalFetch;
|
||||
}
|
||||
});
|
||||
|
||||
test('safeFetch retry metrics count one record per attempt', async () => {
|
||||
const originalFetch = globalThis.fetch;
|
||||
const source = 'unit-retry-attempts';
|
||||
let calls = 0;
|
||||
globalThis.fetch = async () => {
|
||||
calls += 1;
|
||||
if (calls === 1) {
|
||||
return {
|
||||
ok: false,
|
||||
status: 502,
|
||||
headers: { get: () => 'application/json' },
|
||||
text: async () => 'bad gateway',
|
||||
};
|
||||
}
|
||||
return {
|
||||
ok: true,
|
||||
status: 200,
|
||||
headers: { get: () => 'application/json' },
|
||||
text: async () => '{"ok":true}',
|
||||
};
|
||||
};
|
||||
try {
|
||||
const data = await safeFetch('https://example.test/retry', { retries: 1, source });
|
||||
assert.equal(data.ok, true);
|
||||
assert.equal(calls, 2);
|
||||
const bucket = getFetchMetrics().bySource[source];
|
||||
assert.equal(bucket.requests, 2);
|
||||
assert.equal(bucket.ok, 1);
|
||||
assert.equal(bucket.failed, 1);
|
||||
assert.equal(bucket.lastStatus, 200);
|
||||
} finally {
|
||||
globalThis.fetch = originalFetch;
|
||||
}
|
||||
@@ -92,3 +157,18 @@ test('stale alert message includes operator context and affected sources', () =>
|
||||
assert.match(message, /GDELT: degraded \(timeout\)/);
|
||||
assert.match(message, /Dashboard: https:\/\/terminal\.example\.test/);
|
||||
});
|
||||
|
||||
test('scenario watchlist feature is wired into sweep, briefing, and dashboard', () => {
|
||||
const scenarios = readFileSync(new URL('../lib/scenarios.mjs', import.meta.url), 'utf8');
|
||||
const server = readFileSync(new URL('../server.mjs', import.meta.url), 'utf8');
|
||||
const html = readFileSync(new URL('../dashboard/public/jarvis.html', import.meta.url), 'utf8');
|
||||
const readme = readFileSync(new URL('../README.md', import.meta.url), 'utf8');
|
||||
assert.match(scenarios, /DEFAULT_SCENARIOS/);
|
||||
assert.match(scenarios, /runsDir, 'scenarios\.json'/);
|
||||
assert.match(scenarios, /scenario-state\.json/);
|
||||
assert.match(scenarios, /watching.*building.*confirmed/s);
|
||||
assert.match(server, /evaluateScenarios\(synthesized, delta, RUNS_DIR\)/);
|
||||
assert.match(server, /\*Scenario Watchlist\*/);
|
||||
assert.match(html, /Scenario Watchlist/);
|
||||
assert.match(readme, /runs\/scenarios\.json/);
|
||||
});
|
||||
|
||||
Reference in New Issue
Block a user