gadm-ts/dist/wrapper.js
2026-03-29 03:12:14 +02:00

268 lines
21 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 });
console.log('Writing cache:', path);
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, resolution = 3) {
console.log('getBoundary', gadmId, contentLevel, enrichOptions, resolution, cache);
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) {
console.log('getBoundary cache hit', key);
return cached;
}
// 2. Fetch the base geometry
let baseCollection = null;
// First try the far superior SQLite GeoPackage
const gpkgRes = await getBoundaryFromGpkg(gadmId, contentLevel, cache, resolution);
if (gpkgRes) {
console.log('getBoundary gpkgRes', 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) {
console.log('getBoundary baseCached', baseCached);
baseCollection = baseCached;
}
else {
console.log('getBoundary baseCached miss', baseCached);
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 — skip if already enriched (e.g. from C++ cache)
const alreadyEnriched = baseCollection?.features?.some((f) => f.properties?.ghsPopulation !== undefined);
if (enrichOptions && !alreadyEnriched && baseCollection && baseCollection.features) {
console.log('getBoundary enrichOptions', enrichOptions);
// 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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"}