Files
worldmonitor/api/story.js
Elie Habib c353cf2070 Reduce egress costs, add PWA support, fix Polymarket and Railway relay
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
2026-02-14 19:53:04 +04:00

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, '&amp;').replace(/"/g, '&quot;').replace(/</g, '&lt;').replace(/>/g, '&gt;');
}