mirror of
https://github.com/koala73/worldmonitor.git
synced 2026-04-25 17:14:57 +02:00
* feat(fires): flag possible explosions in satellite thermal detections Adds possibleExplosion field (FRP >80 MW + brightness >380 K) to fire detections, surfacing non-fire thermal signatures that may indicate strikes or explosions. Seeder computes the flag, panel shows inline badge per region and summary alert when explosions are detected. * refactor(fires): extract brightness/frp locals to avoid double-parse
132 lines
4.4 KiB
JavaScript
Executable File
132 lines
4.4 KiB
JavaScript
Executable File
#!/usr/bin/env node
|
||
|
||
import { loadEnvFile, runSeed, CHROME_UA, sleep } from './_seed-utils.mjs';
|
||
|
||
loadEnvFile(import.meta.url);
|
||
|
||
const CANONICAL_KEY = 'wildfire:fires:v1';
|
||
const FIRMS_SOURCES = ['VIIRS_SNPP_NRT', 'VIIRS_NOAA20_NRT', 'VIIRS_NOAA21_NRT'];
|
||
|
||
const MONITORED_REGIONS = {
|
||
'Ukraine': '22,44,40,53',
|
||
'Russia': '20,50,180,82',
|
||
'Iran': '44,25,63,40',
|
||
'Israel/Gaza': '34,29,36,34',
|
||
'Syria': '35,32,42,37',
|
||
'Taiwan': '119,21,123,26',
|
||
'North Korea': '124,37,131,43',
|
||
'Saudi Arabia': '34,16,56,32',
|
||
'Turkey': '26,36,45,42',
|
||
};
|
||
|
||
function mapConfidence(c) {
|
||
switch ((c || '').toLowerCase()) {
|
||
case 'h': return 'FIRE_CONFIDENCE_HIGH';
|
||
case 'n': return 'FIRE_CONFIDENCE_NOMINAL';
|
||
case 'l': return 'FIRE_CONFIDENCE_LOW';
|
||
default: return 'FIRE_CONFIDENCE_UNSPECIFIED';
|
||
}
|
||
}
|
||
|
||
function parseCSV(csv) {
|
||
const lines = csv.trim().split('\n');
|
||
if (lines.length < 2) return [];
|
||
const headers = lines[0].split(',').map(h => h.trim());
|
||
const results = [];
|
||
for (let i = 1; i < lines.length; i++) {
|
||
const vals = lines[i].split(',').map(v => v.trim());
|
||
if (vals.length < headers.length) continue;
|
||
const row = {};
|
||
headers.forEach((h, idx) => { row[h] = vals[idx]; });
|
||
results.push(row);
|
||
}
|
||
return results;
|
||
}
|
||
|
||
function parseDetectedAt(acqDate, acqTime) {
|
||
const padded = (acqTime || '').padStart(4, '0');
|
||
const hours = padded.slice(0, 2);
|
||
const minutes = padded.slice(2);
|
||
return new Date(`${acqDate}T${hours}:${minutes}:00Z`).getTime();
|
||
}
|
||
|
||
async function fetchRegionSource(apiKey, regionName, bbox, source) {
|
||
const url = `https://firms.modaps.eosdis.nasa.gov/api/area/csv/${apiKey}/${source}/${bbox}/1`;
|
||
const res = await fetch(url, {
|
||
headers: { Accept: 'text/csv', 'User-Agent': CHROME_UA },
|
||
signal: AbortSignal.timeout(30_000),
|
||
});
|
||
if (!res.ok) throw new Error(`FIRMS ${res.status} for ${regionName}/${source}`);
|
||
const csv = await res.text();
|
||
return parseCSV(csv);
|
||
}
|
||
|
||
async function fetchAllRegions(apiKey) {
|
||
const entries = Object.entries(MONITORED_REGIONS);
|
||
const seen = new Set();
|
||
const fireDetections = [];
|
||
let fulfilled = 0;
|
||
let failed = 0;
|
||
|
||
for (const source of FIRMS_SOURCES) {
|
||
for (const [regionName, bbox] of entries) {
|
||
try {
|
||
const rows = await fetchRegionSource(apiKey, regionName, bbox, source);
|
||
fulfilled++;
|
||
for (const row of rows) {
|
||
const id = `${row.latitude ?? ''}-${row.longitude ?? ''}-${row.acq_date ?? ''}-${row.acq_time ?? ''}`;
|
||
if (seen.has(id)) continue;
|
||
seen.add(id);
|
||
const detectedAt = parseDetectedAt(row.acq_date || '', row.acq_time || '');
|
||
const brightness = parseFloat(row.bright_ti4 ?? '0') || 0;
|
||
const frp = parseFloat(row.frp ?? '0') || 0;
|
||
fireDetections.push({
|
||
id,
|
||
location: {
|
||
latitude: parseFloat(row.latitude ?? '0') || 0,
|
||
longitude: parseFloat(row.longitude ?? '0') || 0,
|
||
},
|
||
brightness,
|
||
frp,
|
||
confidence: mapConfidence(row.confidence || ''),
|
||
satellite: row.satellite || '',
|
||
detectedAt,
|
||
region: regionName,
|
||
dayNight: row.daynight || '',
|
||
possibleExplosion: frp > 80 && brightness > 380,
|
||
});
|
||
}
|
||
} catch (err) {
|
||
failed++;
|
||
console.error(` [FIRMS] ${source}/${regionName}: ${err.message || err}`);
|
||
}
|
||
await sleep(6_000); // FIRMS free tier: 10 req/min — 6s between calls stays safely under limit
|
||
}
|
||
console.log(` ${source}: ${fireDetections.length} total (${fulfilled} ok, ${failed} failed)`);
|
||
}
|
||
|
||
return { fireDetections, pagination: undefined };
|
||
}
|
||
|
||
async function main() {
|
||
const apiKey = process.env.NASA_FIRMS_API_KEY || process.env.FIRMS_API_KEY || '';
|
||
if (!apiKey) {
|
||
console.log('NASA_FIRMS_API_KEY not set — skipping fire detections seed');
|
||
process.exit(0);
|
||
}
|
||
|
||
console.log(' FIRMS key configured');
|
||
|
||
await runSeed('wildfire', 'fires', CANONICAL_KEY, () => fetchAllRegions(apiKey), {
|
||
validateFn: (data) => Array.isArray(data?.fireDetections) && data.fireDetections.length > 0,
|
||
ttlSeconds: 7200,
|
||
lockTtlMs: 600_000, // 10 min — 27 calls × (6s pace + up to 30s timeout) can exceed 5 min under partial slowness
|
||
sourceVersion: FIRMS_SOURCES.join('+'),
|
||
});
|
||
}
|
||
|
||
main().catch(err => {
|
||
const _cause = err.cause ? ` (cause: ${err.cause.message || err.cause.code || err.cause})` : ''; console.error('FATAL:', (err.message || err) + _cause);
|
||
process.exit(1);
|
||
});
|