259 lines
20 KiB
JavaScript
259 lines
20 KiB
JavaScript
/**
|
|
* Wrapper API — replaces gadm_wrapper.py CLI.
|
|
*
|
|
* Provides the same 3 operations: searchRegions, getBoundary, getRegionNames.
|
|
* Includes file-based caching identical to the Python wrapper.
|
|
*/
|
|
import { getNames } from './names.js';
|
|
import { getItems } from './items.js';
|
|
import { existsSync, readFileSync, writeFileSync, mkdirSync } from 'fs';
|
|
import { join, dirname } from 'path';
|
|
import { fileURLToPath } from 'url';
|
|
import { enrichFeatureWithGHS } from './enrich-ghs.js';
|
|
// ---------- cache ----------
|
|
const _filename = fileURLToPath(import.meta.url);
|
|
const _dirname = dirname(_filename);
|
|
const LOCAL_CACHE_DIR = join(_dirname, '../cache/gadm');
|
|
const CACHE_DIR = process.env.GADM_CACHE || LOCAL_CACHE_DIR;
|
|
function getCacheKey(prefix, value) {
|
|
const safeValue = value.replace(/[^a-zA-Z0-9\.\-_]/g, '_').substring(0, 50);
|
|
return `${prefix}_${safeValue}`;
|
|
}
|
|
function cachePath(key) {
|
|
return join(CACHE_DIR, `${key}.json`);
|
|
}
|
|
async function readCache(key, extCache) {
|
|
if (extCache) {
|
|
try {
|
|
const cached = await extCache.get(key);
|
|
if (cached)
|
|
return cached;
|
|
}
|
|
catch { }
|
|
return null; // fallback or ignore
|
|
}
|
|
const path = cachePath(key);
|
|
if (!existsSync(path))
|
|
return null;
|
|
try {
|
|
return JSON.parse(readFileSync(path, 'utf-8'));
|
|
}
|
|
catch {
|
|
return null;
|
|
}
|
|
}
|
|
async function writeCache(key, data, extCache) {
|
|
if (extCache) {
|
|
try {
|
|
await extCache.set(key, data);
|
|
}
|
|
catch { }
|
|
return;
|
|
}
|
|
const path = cachePath(key);
|
|
try {
|
|
mkdirSync(dirname(path), { recursive: true });
|
|
writeFileSync(path, JSON.stringify(data));
|
|
}
|
|
catch (e) {
|
|
console.error('Cache write failed:', e);
|
|
}
|
|
}
|
|
/**
|
|
* Search for admin regions by name.
|
|
* Returns metadata rows or GeoJSON FeatureCollection.
|
|
*/
|
|
export async function searchRegions(opts) {
|
|
const { query, contentLevel, geojson = false, country, cache } = opts;
|
|
const prefix = `search_${contentLevel ?? 'all'}_${geojson ? 'geo' : 'meta'}_${country ?? 'all'}`;
|
|
const key = getCacheKey(prefix, query);
|
|
const cached = await readCache(key, cache);
|
|
if (cached)
|
|
return cached;
|
|
try {
|
|
if (geojson) {
|
|
// GeoJSON mode — use Items
|
|
const itemOpts = { name: [query], cache };
|
|
if (contentLevel != null)
|
|
itemOpts.contentLevel = contentLevel;
|
|
try {
|
|
const gdf = await getItems(itemOpts);
|
|
const output = {
|
|
type: 'FeatureCollection',
|
|
features: gdf.features,
|
|
};
|
|
await writeCache(key, output, cache);
|
|
return output;
|
|
}
|
|
catch (e) {
|
|
// Try first part if comma-separated
|
|
if (query.includes(',')) {
|
|
const first = query.split(',')[0].trim();
|
|
itemOpts.name = [first];
|
|
try {
|
|
const gdf = await getItems(itemOpts);
|
|
const output = {
|
|
type: 'FeatureCollection',
|
|
features: gdf.features,
|
|
};
|
|
await writeCache(key, output, cache);
|
|
return output;
|
|
}
|
|
catch {
|
|
return { data: [] };
|
|
}
|
|
}
|
|
throw e;
|
|
}
|
|
}
|
|
else {
|
|
// Metadata mode — use Names
|
|
const namesOpts = { name: query };
|
|
if (contentLevel != null)
|
|
namesOpts.contentLevel = contentLevel;
|
|
if (country)
|
|
namesOpts.admin = country;
|
|
try {
|
|
const result = await getNames(namesOpts);
|
|
const output = { data: result.rows };
|
|
await writeCache(key, output, cache);
|
|
return output;
|
|
}
|
|
catch {
|
|
// Try first part if comma-separated
|
|
if (query.includes(',')) {
|
|
const first = query.split(',')[0].trim();
|
|
namesOpts.name = first;
|
|
try {
|
|
const result = await getNames(namesOpts);
|
|
const output = { data: result.rows };
|
|
await writeCache(key, output, cache);
|
|
return output;
|
|
}
|
|
catch {
|
|
return { data: [] };
|
|
}
|
|
}
|
|
return { data: [] };
|
|
}
|
|
}
|
|
}
|
|
catch (e) {
|
|
return { error: e.stack || e.message };
|
|
}
|
|
}
|
|
import { getBoundaryFromGpkg } from './gpkg-reader.js';
|
|
/**
|
|
* Get boundary GeoJSON for a specific GADM ID.
|
|
* Optionally enriches the boundary with GHS Population and Built-up Area data!
|
|
*/
|
|
export async function getBoundary(gadmId, contentLevel, cache, enrichOptions) {
|
|
const enrichKeySuffix = enrichOptions ? '_enriched' : '';
|
|
const keySuffix = `${contentLevel ?? 'auto'}_${gadmId}${enrichKeySuffix}`;
|
|
const key = getCacheKey(`boundary`, keySuffix);
|
|
// 1. Check if we already have the EXACT requested state cached
|
|
const cached = await readCache(key, cache);
|
|
if (cached)
|
|
return cached;
|
|
// 2. Fetch the base geometry
|
|
let baseCollection = null;
|
|
// First try the far superior SQLite GeoPackage
|
|
const gpkgRes = await getBoundaryFromGpkg(gadmId, contentLevel, cache);
|
|
if (gpkgRes) {
|
|
baseCollection = gpkgRes;
|
|
}
|
|
else {
|
|
// Fallback exactly as before to Parquet mode
|
|
const baseKey = getCacheKey(`boundary_${contentLevel ?? 'auto'}`, gadmId);
|
|
const baseCached = await readCache(baseKey, cache);
|
|
if (baseCached) {
|
|
baseCollection = baseCached;
|
|
}
|
|
else {
|
|
try {
|
|
const gdf = await getItems({ admin: [gadmId], contentLevel, cache });
|
|
if (gdf.features.length === 0) {
|
|
return { error: 'Region not found' };
|
|
}
|
|
baseCollection = gdf;
|
|
await writeCache(baseKey, baseCollection, cache);
|
|
}
|
|
catch (e) {
|
|
return { error: e.message };
|
|
}
|
|
}
|
|
}
|
|
let collectionToReturn = baseCollection;
|
|
// 3. Apply GeoTIFF enrichment if requested
|
|
if (enrichOptions && baseCollection && baseCollection.features) {
|
|
// Deep clone so we don't mutate an in-memory cached object accidentally
|
|
collectionToReturn = JSON.parse(JSON.stringify(baseCollection));
|
|
for (const feature of collectionToReturn.features) {
|
|
try {
|
|
const enrichResult = await enrichFeatureWithGHS(feature, enrichOptions);
|
|
Object.assign(feature.properties, enrichResult);
|
|
// Set the default population property to the highly accurate GHS number
|
|
if (enrichResult.ghsPopulation !== undefined) {
|
|
feature.properties.population = enrichResult.ghsPopulation;
|
|
}
|
|
}
|
|
catch (e) {
|
|
console.error('GHS Enrichment failed for feature', feature.properties?.name || '', e);
|
|
}
|
|
}
|
|
await writeCache(key, collectionToReturn, cache);
|
|
}
|
|
return collectionToReturn;
|
|
}
|
|
/**
|
|
* Get names for all sub-regions of an admin area, optionally recursing through levels.
|
|
*/
|
|
export async function getRegionNames(opts) {
|
|
const { admin, contentLevel, depth = 1, cache } = opts;
|
|
const key = getCacheKey(`names_${admin}_${contentLevel}_${depth}`, 'names');
|
|
const cached = await readCache(key, cache);
|
|
if (cached)
|
|
return cached;
|
|
try {
|
|
const allResults = [];
|
|
// Determine starting level from admin code
|
|
let startLevel;
|
|
if (contentLevel != null) {
|
|
startLevel = contentLevel;
|
|
}
|
|
else {
|
|
// GID format: XXX.L1.L2_1 — count dots + 1
|
|
startLevel = admin.split('.').length; // "ESP" = 1 dot → level 1 start
|
|
if (!admin.includes('.'))
|
|
startLevel = 1; // country code → start at level 1
|
|
}
|
|
// Handle infinite depth
|
|
const depthStr = String(depth).toLowerCase();
|
|
const limit = (depthStr === 'infinite' || depthStr === 'inf' || depthStr === '-1')
|
|
? 5
|
|
: Number(depth);
|
|
let currentLevel = startLevel;
|
|
for (let i = 0; i < limit; i++) {
|
|
try {
|
|
const result = await getNames({
|
|
admin,
|
|
contentLevel: currentLevel,
|
|
});
|
|
if (result.rows.length === 0)
|
|
break;
|
|
allResults.push(...result.rows);
|
|
}
|
|
catch {
|
|
break; // Level doesn't exist
|
|
}
|
|
currentLevel++;
|
|
}
|
|
const output = { data: allResults };
|
|
await writeCache(key, output, cache);
|
|
return output;
|
|
}
|
|
catch (e) {
|
|
return { error: e.stack || e.message };
|
|
}
|
|
}
|
|
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"wrapper.js","sourceRoot":"","sources":["../src/wrapper.ts"],"names":[],"mappings":"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"}
|