mirror of
https://github.com/koala73/worldmonitor.git
synced 2026-04-25 17:14:57 +02:00
- Tighten CORS regex to block worldmonitorEVIL.vercel.app spoofing - Move sidecar /api/local-env-update behind token auth + add key allowlist - Add postMessage origin/source validation in LiveNewsPanel - Replace postMessage wildcard '*' targetOrigin with specific origin - Add isDisallowedOrigin() check to 25 API endpoints missing it - Migrate gdelt-geo & EIA from custom CORS to shared _cors.js - Add CORS to firms-fires, stock-index, youtube/live endpoints - Tighten youtube/embed.js ALLOWED_ORIGINS regex - Remove 'unsafe-inline' from CSP script-src - Add iframe sandbox attribute to YouTube embed - Validate meta-tags URL query params with regex allowlist
155 lines
5.1 KiB
JavaScript
155 lines
5.1 KiB
JavaScript
export const config = { runtime: 'edge' };
|
|
|
|
import { getCachedJson, hashString, setCachedJson } from './_upstash-cache.js';
|
|
import { recordCacheTelemetry } from './_cache-telemetry.js';
|
|
import { getCorsHeaders, isDisallowedOrigin } from './_cors.js';
|
|
|
|
const ALLOWED_CURRENCIES = ['usd', 'eur', 'gbp', 'jpy', 'cny', 'btc', 'eth'];
|
|
const MAX_COIN_IDS = 20;
|
|
const COIN_ID_PATTERN = /^[a-z0-9-]+$/;
|
|
|
|
const CACHE_TTL_SECONDS = 120; // 2 minutes
|
|
const CACHE_TTL_MS = CACHE_TTL_SECONDS * 1000;
|
|
const RESPONSE_CACHE_CONTROL = 'public, max-age=120, s-maxage=120, stale-while-revalidate=60';
|
|
const CACHE_VERSION = 'v2';
|
|
|
|
// In-memory fallback cache for the current instance.
|
|
let fallbackCache = { key: '', payload: null, timestamp: 0 };
|
|
|
|
function validateCoinIds(idsParam) {
|
|
if (!idsParam) return 'bitcoin,ethereum,solana';
|
|
|
|
const ids = idsParam.split(',')
|
|
.map(id => id.trim().toLowerCase())
|
|
.filter(id => COIN_ID_PATTERN.test(id) && id.length <= 50)
|
|
.slice(0, MAX_COIN_IDS);
|
|
|
|
return ids.length > 0 ? ids.join(',') : 'bitcoin,ethereum,solana';
|
|
}
|
|
|
|
function validateCurrency(val) {
|
|
const currency = (val || 'usd').toLowerCase();
|
|
return ALLOWED_CURRENCIES.includes(currency) ? currency : 'usd';
|
|
}
|
|
|
|
function validateBoolean(val, defaultVal) {
|
|
if (val === 'true' || val === 'false') return val;
|
|
return defaultVal;
|
|
}
|
|
|
|
function getHeaders(cors, xCache, cacheControl = RESPONSE_CACHE_CONTROL) {
|
|
return {
|
|
'Content-Type': 'application/json',
|
|
...cors,
|
|
'Cache-Control': cacheControl,
|
|
'X-Cache': xCache,
|
|
};
|
|
}
|
|
|
|
function isValidPayload(payload) {
|
|
return Boolean(
|
|
payload &&
|
|
typeof payload === 'object' &&
|
|
typeof payload.body === 'string' &&
|
|
Number.isFinite(payload.status)
|
|
);
|
|
}
|
|
|
|
export default async function handler(req) {
|
|
const cors = getCorsHeaders(req);
|
|
if (isDisallowedOrigin(req)) {
|
|
return new Response(JSON.stringify({ error: 'Origin not allowed' }), { status: 403, headers: cors });
|
|
}
|
|
const url = new URL(req.url);
|
|
|
|
const ids = validateCoinIds(url.searchParams.get('ids'));
|
|
const vsCurrencies = validateCurrency(url.searchParams.get('vs_currencies'));
|
|
const include24hrChange = validateBoolean(url.searchParams.get('include_24hr_change'), 'true');
|
|
|
|
const now = Date.now();
|
|
const cacheKey = `${ids}:${vsCurrencies}:${include24hrChange}`;
|
|
const redisKey = `coingecko:${CACHE_VERSION}:${hashString(cacheKey)}`;
|
|
|
|
const redisCached = await getCachedJson(redisKey);
|
|
if (isValidPayload(redisCached)) {
|
|
recordCacheTelemetry('/api/coingecko', 'REDIS-HIT');
|
|
return new Response(redisCached.body, {
|
|
status: redisCached.status,
|
|
headers: getHeaders(cors, 'REDIS-HIT'),
|
|
});
|
|
}
|
|
|
|
if (
|
|
isValidPayload(fallbackCache.payload) &&
|
|
fallbackCache.key === cacheKey &&
|
|
now - fallbackCache.timestamp < CACHE_TTL_MS
|
|
) {
|
|
recordCacheTelemetry('/api/coingecko', 'MEMORY-HIT');
|
|
return new Response(fallbackCache.payload.body, {
|
|
status: fallbackCache.payload.status,
|
|
headers: getHeaders(cors, 'MEMORY-HIT'),
|
|
});
|
|
}
|
|
|
|
const endpoint = url.searchParams.get('endpoint');
|
|
|
|
try {
|
|
let geckoUrl;
|
|
if (endpoint === 'markets') {
|
|
geckoUrl = `https://api.coingecko.com/api/v3/coins/markets?vs_currency=${vsCurrencies}&ids=${ids}&order=market_cap_desc&sparkline=true&price_change_percentage=24h`;
|
|
} else {
|
|
geckoUrl = `https://api.coingecko.com/api/v3/simple/price?ids=${ids}&vs_currencies=${vsCurrencies}&include_24hr_change=${include24hrChange}`;
|
|
}
|
|
const response = await fetch(geckoUrl, {
|
|
headers: {
|
|
'Accept': 'application/json',
|
|
},
|
|
});
|
|
|
|
// If rate limited, return cached data if available
|
|
if (
|
|
response.status === 429 &&
|
|
isValidPayload(fallbackCache.payload) &&
|
|
fallbackCache.key === cacheKey
|
|
) {
|
|
recordCacheTelemetry('/api/coingecko', 'STALE');
|
|
return new Response(fallbackCache.payload.body, {
|
|
status: fallbackCache.payload.status,
|
|
headers: getHeaders(cors, 'STALE'),
|
|
});
|
|
}
|
|
|
|
const data = await response.text();
|
|
|
|
// Cache successful responses
|
|
if (response.ok) {
|
|
const payload = { body: data, status: response.status };
|
|
fallbackCache = { key: cacheKey, payload, timestamp: Date.now() };
|
|
void setCachedJson(redisKey, payload, CACHE_TTL_SECONDS);
|
|
recordCacheTelemetry('/api/coingecko', 'MISS');
|
|
} else {
|
|
recordCacheTelemetry('/api/coingecko', 'UPSTREAM-ERROR');
|
|
}
|
|
|
|
return new Response(data, {
|
|
status: response.status,
|
|
headers: getHeaders(cors, 'MISS'),
|
|
});
|
|
} catch (error) {
|
|
// Return cached data on error if available
|
|
if (isValidPayload(fallbackCache.payload) && fallbackCache.key === cacheKey) {
|
|
recordCacheTelemetry('/api/coingecko', 'ERROR-FALLBACK');
|
|
return new Response(fallbackCache.payload.body, {
|
|
status: fallbackCache.payload.status,
|
|
headers: getHeaders(cors, 'ERROR-FALLBACK', 'public, max-age=120'),
|
|
});
|
|
}
|
|
|
|
recordCacheTelemetry('/api/coingecko', 'ERROR');
|
|
return new Response(JSON.stringify({ error: 'Failed to fetch data' }), {
|
|
status: 500,
|
|
headers: { 'Content-Type': 'application/json', ...cors },
|
|
});
|
|
}
|
|
}
|