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
69 lines
2.2 KiB
JavaScript
69 lines
2.2 KiB
JavaScript
import { getCorsHeaders, isDisallowedOrigin } from './_cors.js';
|
|
export const config = { runtime: 'edge' };
|
|
|
|
const MAX_RECORDS = 20;
|
|
const DEFAULT_RECORDS = 10;
|
|
|
|
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 query = url.searchParams.get('query');
|
|
const maxrecords = Math.min(
|
|
parseInt(url.searchParams.get('maxrecords') || DEFAULT_RECORDS, 10),
|
|
MAX_RECORDS
|
|
);
|
|
const timespan = url.searchParams.get('timespan') || '72h';
|
|
|
|
if (!query || query.length < 2) {
|
|
return new Response(JSON.stringify({ error: 'Query parameter required' }), {
|
|
status: 400,
|
|
headers: { 'Content-Type': 'application/json' },
|
|
});
|
|
}
|
|
|
|
try {
|
|
const gdeltUrl = new URL('https://api.gdeltproject.org/api/v2/doc/doc');
|
|
gdeltUrl.searchParams.set('query', query);
|
|
gdeltUrl.searchParams.set('mode', 'artlist');
|
|
gdeltUrl.searchParams.set('maxrecords', maxrecords.toString());
|
|
gdeltUrl.searchParams.set('format', 'json');
|
|
gdeltUrl.searchParams.set('sort', 'date');
|
|
gdeltUrl.searchParams.set('timespan', timespan);
|
|
|
|
const response = await fetch(gdeltUrl.toString());
|
|
|
|
if (!response.ok) {
|
|
throw new Error(`GDELT returned ${response.status}`);
|
|
}
|
|
|
|
const data = await response.json();
|
|
|
|
const articles = (data.articles || []).map(article => ({
|
|
title: article.title,
|
|
url: article.url,
|
|
source: article.domain || article.source?.domain,
|
|
date: article.seendate,
|
|
image: article.socialimage,
|
|
language: article.language,
|
|
tone: article.tone,
|
|
}));
|
|
|
|
return new Response(JSON.stringify({ articles, query }), {
|
|
status: 200,
|
|
headers: {
|
|
'Content-Type': 'application/json',
|
|
...cors,
|
|
'Cache-Control': 'public, max-age=300, s-maxage=300, stale-while-revalidate=60',
|
|
},
|
|
});
|
|
} catch (error) {
|
|
return new Response(JSON.stringify({ error: error.message, articles: [] }), {
|
|
status: 500,
|
|
headers: { 'Content-Type': 'application/json' },
|
|
});
|
|
}
|
|
}
|