mirror of
https://github.com/koala73/worldmonitor.git
synced 2026-05-13 18:46:21 +02:00
Egress optimization: - Add s-maxage + stale-while-revalidate to all API endpoints for Vercel CDN caching - Add vercel.json with immutable caching for hashed assets - Add gzip compression to sidecar responses >1KB - Add gzip to Railway RSS responses (4 paths previously uncompressed) - Increase polling intervals: markets/crypto 60s→120s, ETF/macro/stablecoins 60s→180s - Remove hardcoded Railway URL from theater-posture.js (now env-var only) PWA / Service Worker: - Add vite-plugin-pwa with autoUpdate strategy - Cache map tiles (CacheFirst), fonts (StaleWhileRevalidate), static assets - NetworkOnly for all /api/* routes (real-time data must be fresh) - Manual SW registration (web only, skip Tauri) - Add offline fallback page - Replace manual manifest with plugin-generated manifest Polymarket fix: - Route dev proxy through production Vercel (bypasses JA3 blocking) - Add 4th fallback tier: production URL as absolute fallback Desktop/Sidecar: - Dual-backend cache (_upstash-cache.js): Redis cloud + in-memory+file desktop - Settings window OK/Cancel redesign - Runtime config and secret injection improvements
85 lines
3.4 KiB
JavaScript
85 lines
3.4 KiB
JavaScript
/**
|
|
* Story Page for Social Crawlers
|
|
* Returns HTML with proper og:image and twitter:card meta tags.
|
|
* Twitter/Facebook/LinkedIn crawlers hit this, real users get redirected to the SPA.
|
|
*/
|
|
|
|
const COUNTRY_NAMES = {
|
|
UA: 'Ukraine', RU: 'Russia', CN: 'China', US: 'United States',
|
|
IR: 'Iran', IL: 'Israel', TW: 'Taiwan', KP: 'North Korea',
|
|
SA: 'Saudi Arabia', TR: 'Turkey', PL: 'Poland', DE: 'Germany',
|
|
FR: 'France', GB: 'United Kingdom', IN: 'India', PK: 'Pakistan',
|
|
SY: 'Syria', YE: 'Yemen', MM: 'Myanmar', VE: 'Venezuela',
|
|
};
|
|
|
|
const BOT_UA = /twitterbot|facebookexternalhit|linkedinbot|slackbot|telegrambot|whatsapp|discordbot|redditbot|googlebot/i;
|
|
|
|
export default function handler(req, res) {
|
|
const url = new URL(req.url, `https://${req.headers.host}`);
|
|
const countryCode = (url.searchParams.get('c') || '').toUpperCase();
|
|
const type = url.searchParams.get('t') || 'ciianalysis';
|
|
const ts = url.searchParams.get('ts') || '';
|
|
const score = url.searchParams.get('s') || '';
|
|
const level = url.searchParams.get('l') || '';
|
|
|
|
const ua = req.headers['user-agent'] || '';
|
|
const isBot = BOT_UA.test(ua);
|
|
|
|
const baseUrl = `https://${req.headers.host}`;
|
|
const spaUrl = `${baseUrl}/?c=${countryCode}&t=${type}${ts ? `&ts=${ts}` : ''}`;
|
|
|
|
// Real users → redirect to SPA
|
|
if (!isBot) {
|
|
res.writeHead(302, { Location: spaUrl });
|
|
res.end();
|
|
return;
|
|
}
|
|
|
|
// Bots → serve meta tags
|
|
const countryName = COUNTRY_NAMES[countryCode] || countryCode || 'Global';
|
|
const title = `${countryName} Intelligence Brief | World Monitor`;
|
|
const description = `Real-time instability analysis for ${countryName}. Country Instability Index, military posture, threat classification, and prediction markets. Free, open-source geopolitical intelligence.`;
|
|
const imageParams = `c=${countryCode}&t=${type}${score ? `&s=${score}` : ''}${level ? `&l=${level}` : ''}`;
|
|
const imageUrl = `${baseUrl}/api/og-story?${imageParams}`;
|
|
const storyUrl = `${baseUrl}/api/story?c=${countryCode}&t=${type}${ts ? `&ts=${ts}` : ''}`;
|
|
|
|
const html = `<!DOCTYPE html>
|
|
<html lang="en">
|
|
<head>
|
|
<meta charset="utf-8"/>
|
|
<title>${esc(title)}</title>
|
|
<meta name="description" content="${esc(description)}"/>
|
|
|
|
<meta property="og:type" content="article"/>
|
|
<meta property="og:title" content="${esc(title)}"/>
|
|
<meta property="og:description" content="${esc(description)}"/>
|
|
<meta property="og:image" content="${esc(imageUrl)}"/>
|
|
<meta property="og:image:width" content="1200"/>
|
|
<meta property="og:image:height" content="630"/>
|
|
<meta property="og:url" content="${esc(storyUrl)}"/>
|
|
<meta property="og:site_name" content="World Monitor"/>
|
|
|
|
<meta name="twitter:card" content="summary_large_image"/>
|
|
<meta name="twitter:site" content="@WorldMonitorApp"/>
|
|
<meta name="twitter:title" content="${esc(title)}"/>
|
|
<meta name="twitter:description" content="${esc(description)}"/>
|
|
<meta name="twitter:image" content="${esc(imageUrl)}"/>
|
|
|
|
<link rel="canonical" href="${esc(storyUrl)}"/>
|
|
</head>
|
|
<body>
|
|
<h1>${esc(title)}</h1>
|
|
<p>${esc(description)}</p>
|
|
<p><a href="${esc(spaUrl)}">View live analysis</a></p>
|
|
</body>
|
|
</html>`;
|
|
|
|
res.setHeader('Content-Type', 'text/html; charset=utf-8');
|
|
res.setHeader('Cache-Control', 'public, max-age=300, s-maxage=300, stale-while-revalidate=60');
|
|
res.status(200).send(html);
|
|
}
|
|
|
|
function esc(str) {
|
|
return str.replace(/&/g, '&').replace(/"/g, '"').replace(/</g, '<').replace(/>/g, '>');
|
|
}
|