fix: avoid double-counting fetch failures
This commit is contained in:
@@ -42,6 +42,7 @@ export async function safeFetch(url, opts = {}) {
|
||||
let lastError;
|
||||
for (let i = 0; i <= retries; i++) {
|
||||
const started = Date.now();
|
||||
let metricRecorded = false;
|
||||
try {
|
||||
const controller = new AbortController();
|
||||
const timer = setTimeout(() => controller.abort(), timeout);
|
||||
@@ -51,22 +52,29 @@ export async function safeFetch(url, opts = {}) {
|
||||
});
|
||||
clearTimeout(timer);
|
||||
const status = res.status;
|
||||
if (!res.ok) {
|
||||
const body = await res.text().catch(() => '');
|
||||
recordFetchMetric({ url, source, ok: false, status, bytes: body.length, durationMs: Date.now() - started, error: `HTTP ${res.status}` });
|
||||
throw new Error(`HTTP ${res.status}: ${body.slice(0, 200)}`);
|
||||
}
|
||||
const text = await res.text();
|
||||
recordFetchMetric({ url, source, ok: true, status, bytes: text.length, durationMs: Date.now() - started });
|
||||
if (!res.ok) {
|
||||
const error = `HTTP ${res.status}`;
|
||||
recordFetchMetric({ url, source, ok: false, status, bytes: text.length, durationMs: Date.now() - started, error });
|
||||
metricRecorded = true;
|
||||
throw new Error(`${error}: ${text.slice(0, 200)}`);
|
||||
}
|
||||
const trimmed = text.trim();
|
||||
const contentType = res.headers.get('content-type') || '';
|
||||
if (contentType.includes('text/html') || trimmed.startsWith('<!DOCTYPE html') || trimmed.startsWith('<html')) {
|
||||
throw new Error(`Expected JSON but received HTML from ${new URL(url).host}`);
|
||||
const error = `Expected JSON but received HTML from ${new URL(url).host}`;
|
||||
recordFetchMetric({ url, source, ok: false, status, bytes: text.length, durationMs: Date.now() - started, error });
|
||||
metricRecorded = true;
|
||||
throw new Error(error);
|
||||
}
|
||||
recordFetchMetric({ url, source, ok: true, status, bytes: text.length, durationMs: Date.now() - started });
|
||||
metricRecorded = true;
|
||||
try { return JSON.parse(text); } catch { return { rawText: text.slice(0, 500) }; }
|
||||
} catch (e) {
|
||||
lastError = e;
|
||||
recordFetchMetric({ url, source, ok: false, status: null, bytes: 0, durationMs: Date.now() - started, error: e.message });
|
||||
if (!metricRecorded) {
|
||||
recordFetchMetric({ url, source, ok: false, status: null, bytes: 0, durationMs: Date.now() - started, error: e.message });
|
||||
}
|
||||
// GDELT needs 5s between requests, others are fine with shorter delays
|
||||
if (i < retries) await new Promise(r => setTimeout(r, 2000 * (i + 1)));
|
||||
}
|
||||
@@ -79,6 +87,7 @@ export async function safeFetchText(url, opts = {}) {
|
||||
let lastError;
|
||||
for (let i = 0; i <= retries; i++) {
|
||||
const started = Date.now();
|
||||
let metricRecorded = false;
|
||||
try {
|
||||
const controller = new AbortController();
|
||||
const timer = setTimeout(() => controller.abort(), timeout);
|
||||
@@ -89,11 +98,14 @@ export async function safeFetchText(url, opts = {}) {
|
||||
clearTimeout(timer);
|
||||
const text = await res.text();
|
||||
recordFetchMetric({ url, source, ok: res.ok, status: res.status, bytes: text.length, durationMs: Date.now() - started, error: res.ok ? null : `HTTP ${res.status}` });
|
||||
metricRecorded = true;
|
||||
if (!res.ok) throw new Error(`HTTP ${res.status}: ${text.slice(0, 200)}`);
|
||||
return { text, status: res.status, bytes: text.length };
|
||||
} catch (e) {
|
||||
lastError = e;
|
||||
recordFetchMetric({ url, source, ok: false, status: null, bytes: 0, durationMs: Date.now() - started, error: e.message });
|
||||
if (!metricRecorded) {
|
||||
recordFetchMetric({ url, source, ok: false, status: null, bytes: 0, durationMs: Date.now() - started, error: e.message });
|
||||
}
|
||||
if (i < retries) await new Promise(r => setTimeout(r, 2000 * (i + 1)));
|
||||
}
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user