Files
worldmonitor/scripts/seed-etf-flows.mjs
Elie Habib dd127447c0 refactor: consolidate duplicated market data lists into shared JSON configs (#1212)
Adding a new item (crypto, ETF, stablecoin, gulf symbol, etc.) previously
required editing 2-4 files because the same list was hardcoded independently
in seed scripts, RPC handlers, and frontend config. Following the proven
shared/crypto.json pattern, extract 6 new shared JSON configs so each list
has a single source of truth.

New shared configs:
- shared/stablecoins.json (ids + coinpaprika mappings)
- shared/etfs.json (BTC spot ETF tickers + issuers)
- shared/gulf.json (GCC indices, currencies, oil benchmarks)
- shared/sectors.json (sector ETF symbols + names)
- shared/commodities.json (VIX, gold, oil, gas, silver, copper)
- shared/stocks.json (market symbols + yahoo-only set)

All seed scripts, RPC handlers, and frontend config now import from
these shared JSON files instead of maintaining independent copies.
2026-03-07 22:00:55 +04:00

152 lines
4.4 KiB
JavaScript
Executable File

#!/usr/bin/env node
import { createRequire } from 'module';
import { loadEnvFile, CHROME_UA, runSeed } from './_seed-utils.mjs';
const require = createRequire(import.meta.url);
const etfConfig = require('../shared/etfs.json');
loadEnvFile(import.meta.url);
const CANONICAL_KEY = 'market:etf-flows:v1';
const CACHE_TTL = 3600;
const YAHOO_DELAY_MS = 200;
const ETF_LIST = etfConfig.btcSpot;
function sleep(ms) {
return new Promise((r) => setTimeout(r, ms));
}
async function fetchYahooWithRetry(url, label, maxAttempts = 4) {
for (let i = 0; i < maxAttempts; i++) {
const resp = await fetch(url, {
headers: { 'User-Agent': CHROME_UA },
signal: AbortSignal.timeout(10_000),
});
if (resp.status === 429) {
const wait = 5000 * (i + 1);
console.warn(` [Yahoo] ${label} 429 — waiting ${wait / 1000}s (attempt ${i + 1}/${maxAttempts})`);
await sleep(wait);
continue;
}
if (!resp.ok) {
console.warn(` [Yahoo] ${label} HTTP ${resp.status}`);
return null;
}
return resp;
}
console.warn(` [Yahoo] ${label} rate limited after ${maxAttempts} attempts`);
return null;
}
function parseEtfChartData(chart, ticker, issuer) {
const result = chart?.chart?.result?.[0];
if (!result) return null;
const quote = result.indicators?.quote?.[0];
const closes = quote?.close || [];
const volumes = quote?.volume || [];
const validCloses = closes.filter((p) => p != null);
const validVolumes = volumes.filter((v) => v != null);
if (validCloses.length < 2) return null;
const latestPrice = validCloses[validCloses.length - 1];
const prevPrice = validCloses[validCloses.length - 2];
const priceChange = prevPrice ? ((latestPrice - prevPrice) / prevPrice) * 100 : 0;
const latestVolume = validVolumes.length > 0 ? validVolumes[validVolumes.length - 1] : 0;
const avgVolume =
validVolumes.length > 1
? validVolumes.slice(0, -1).reduce((a, b) => a + b, 0) / (validVolumes.length - 1)
: latestVolume;
const volumeRatio = avgVolume > 0 ? latestVolume / avgVolume : 1;
const direction = priceChange > 0.1 ? 'inflow' : priceChange < -0.1 ? 'outflow' : 'neutral';
const estFlowMagnitude = latestVolume * latestPrice * (priceChange > 0 ? 1 : -1) * 0.1;
return {
ticker,
issuer,
price: +latestPrice.toFixed(2),
priceChange: +priceChange.toFixed(2),
volume: latestVolume,
avgVolume: Math.round(avgVolume),
volumeRatio: +volumeRatio.toFixed(2),
direction,
estFlow: Math.round(estFlowMagnitude),
};
}
async function fetchEtfFlows() {
const etfs = [];
let misses = 0;
for (let i = 0; i < ETF_LIST.length; i++) {
const { ticker, issuer } = ETF_LIST[i];
if (i > 0) await sleep(YAHOO_DELAY_MS);
try {
const url = `https://query1.finance.yahoo.com/v8/finance/chart/${ticker}?range=5d&interval=1d`;
const resp = await fetchYahooWithRetry(url, ticker);
if (!resp) {
misses++;
continue;
}
const chart = await resp.json();
const parsed = parseEtfChartData(chart, ticker, issuer);
if (parsed) {
etfs.push(parsed);
console.log(` ${ticker}: $${parsed.price} (${parsed.direction})`);
} else {
misses++;
}
} catch (err) {
console.warn(` [Yahoo] ${ticker} error: ${err.message}`);
misses++;
}
if (misses >= 3 && etfs.length === 0) break;
}
if (etfs.length === 0) {
throw new Error(`All ETF fetches failed (${misses} misses)`);
}
const totalVolume = etfs.reduce((sum, e) => sum + e.volume, 0);
const totalEstFlow = etfs.reduce((sum, e) => sum + e.estFlow, 0);
const inflowCount = etfs.filter((e) => e.direction === 'inflow').length;
const outflowCount = etfs.filter((e) => e.direction === 'outflow').length;
etfs.sort((a, b) => b.volume - a.volume);
return {
timestamp: new Date().toISOString(),
summary: {
etfCount: etfs.length,
totalVolume,
totalEstFlow,
netDirection: totalEstFlow > 0 ? 'NET INFLOW' : totalEstFlow < 0 ? 'NET OUTFLOW' : 'NEUTRAL',
inflowCount,
outflowCount,
},
etfs,
rateLimited: false,
};
}
function validate(data) {
return Array.isArray(data?.etfs) && data.etfs.length >= 1;
}
runSeed('market', 'etf-flows', CANONICAL_KEY, fetchEtfFlows, {
validateFn: validate,
ttlSeconds: CACHE_TTL,
sourceVersion: 'yahoo-chart-5d',
}).catch((err) => {
console.error('FATAL:', err.message || err);
process.exit(1);
});