168 lines
15 KiB
JavaScript
168 lines
15 KiB
JavaScript
/**
|
|
* Items — GeoJSON boundary fetcher.
|
|
*
|
|
* Reads geometry from a local GeoPackage file when available,
|
|
* falls back to the GADM CDN. Results include corrected names
|
|
* from the parquet database.
|
|
*/
|
|
import { getNames } from './names.js';
|
|
import { getBoundaryFromGpkg } from './gpkg-reader.js';
|
|
import { readFileSync } from 'fs';
|
|
import { resolve, dirname } from 'path';
|
|
import { fileURLToPath } from 'url';
|
|
const __filename_ = fileURLToPath(import.meta.url);
|
|
const __dirname_ = dirname(__filename_);
|
|
// ---------- constants ----------
|
|
const GADM_URL = 'https://geodata.ucdavis.edu/gadm/gadm4.1/json/gadm41_{}_{}.json';
|
|
/** Continent → ISO3 codes mapping */
|
|
const CONTINENT_MAP = JSON.parse(readFileSync(resolve(__dirname_, '../data/gadm_continent.json'), 'utf-8'));
|
|
// ---------- main function ----------
|
|
/**
|
|
* Fetch GeoJSON boundaries for administrative areas.
|
|
* Tries local GeoPackage first, falls back to GADM CDN.
|
|
*/
|
|
export async function getItems(opts) {
|
|
let { name, admin, contentLevel = -1, cache } = opts;
|
|
// Normalize to arrays
|
|
let names = typeof name === 'string' ? [name] : (name || []);
|
|
let admins = typeof admin === 'string' ? [admin] : (admin || []);
|
|
if (names.length === 0 && admins.length === 0) {
|
|
throw new Error('At least "name" or "admin" need to be set.');
|
|
}
|
|
// Continent expansion
|
|
if (names.length === 1 && names[0].toLowerCase() in CONTINENT_MAP) {
|
|
admins = CONTINENT_MAP[names[0].toLowerCase()];
|
|
names = [];
|
|
}
|
|
// Fetch all items
|
|
const allFeatures = [];
|
|
for (const n of names) {
|
|
const features = await fetchSingleItem(n, '', contentLevel, cache);
|
|
allFeatures.push(...features);
|
|
}
|
|
for (const a of admins) {
|
|
const features = await fetchSingleItem('', a, contentLevel, cache);
|
|
allFeatures.push(...features);
|
|
}
|
|
return {
|
|
type: 'FeatureCollection',
|
|
features: allFeatures,
|
|
};
|
|
}
|
|
// ---------- internal ----------
|
|
async function fetchSingleItem(name, admin, contentLevel, cache) {
|
|
// Step 1: Look up the area to find its level + GID
|
|
const lookupOpts = name ? { name } : { admin };
|
|
const lookup = await getNames(lookupOpts);
|
|
if (lookup.rows.length > 1) {
|
|
const id = name || admin;
|
|
throw new Error(`The requested name ("${id}") is not unique (${lookup.rows.length} results). ` +
|
|
`Use the "admin" parameter instead. ` +
|
|
`To find the GADM code, use: getNames({ name: "${id}" })`);
|
|
}
|
|
const level = lookup.level;
|
|
const gidCol = `GID_${level}`;
|
|
const gadmId = lookup.rows[0][gidCol] || '';
|
|
// Step 2: Resolve content level
|
|
const contentLookup = await getNames({
|
|
...(name ? { name } : { admin }),
|
|
contentLevel,
|
|
});
|
|
const resolvedLevel = contentLookup.level;
|
|
// Step 3: Try local GeoPackage first
|
|
let localResult = await tryLocalGpkg(gadmId, name || admin, level, resolvedLevel, cache);
|
|
// Step 4: Fallback to GADM CDN
|
|
if (!localResult) {
|
|
localResult = await fetchFromCDN(name, admin, gadmId, level, resolvedLevel);
|
|
}
|
|
if (resolvedLevel === level) {
|
|
localResult.forEach(f => f.properties.isOuter = true);
|
|
}
|
|
return localResult;
|
|
}
|
|
/**
|
|
* Read boundaries from local GeoPackage.
|
|
* Returns features with all matching rows at the resolved content level.
|
|
*/
|
|
async function tryLocalGpkg(gadmId, id, level, resolvedLevel, cache) {
|
|
// Query the GeoPackage for all rows belonging to this region
|
|
const gidCol = `GID_${level}`;
|
|
const nameCols = [];
|
|
for (let l = 0; l <= Math.min(resolvedLevel, 5); l++) {
|
|
nameCols.push(`NAME_${l}`, `GID_${l}`);
|
|
}
|
|
const result = await getBoundaryFromGpkg(gadmId, resolvedLevel, cache);
|
|
if (!result || result.features.length === 0)
|
|
return null;
|
|
// Convert BoundaryFeature → GeoJSONFeature
|
|
const features = result.features.map(f => ({
|
|
type: 'Feature',
|
|
properties: f.properties,
|
|
geometry: f.geometry,
|
|
}));
|
|
return features;
|
|
}
|
|
/**
|
|
* Fallback: fetch from GADM CDN + correct names from parquet.
|
|
*/
|
|
async function fetchFromCDN(name, admin, gadmId, level, resolvedLevel) {
|
|
const iso3 = gadmId.substring(0, 3);
|
|
const url = GADM_URL.replace('{}', iso3).replace('{}', String(resolvedLevel));
|
|
let data;
|
|
try {
|
|
const res = await fetch(url);
|
|
if (!res.ok)
|
|
throw new Error(`HTTP ${res.status}`);
|
|
data = await res.json();
|
|
}
|
|
catch (e) {
|
|
throw new Error(`Cannot retrieve data from GADM server. ` +
|
|
`Try opening: ${url}. ` +
|
|
`If it fails, the error is from GADM servers.`);
|
|
}
|
|
// Convert to features
|
|
const rawFeatures = data.features || [];
|
|
let features = rawFeatures.map((f) => ({
|
|
type: 'Feature',
|
|
properties: { ...f.properties, ...(f.properties?.COUNTRY ? { NAME_0: f.properties.COUNTRY } : {}) },
|
|
geometry: f.geometry,
|
|
}));
|
|
// Name correction from parquet
|
|
const isos = [...new Set(features.map(f => f.properties.GID_0).filter(Boolean))];
|
|
if (isos.length > 0) {
|
|
const completeRows = [];
|
|
for (const iso of isos) {
|
|
const nr = await getNames({ admin: iso, contentLevel: resolvedLevel, complete: true });
|
|
completeRows.push(...nr.rows);
|
|
}
|
|
const gidKey = (row) => {
|
|
const parts = [];
|
|
for (let i = 0; i <= resolvedLevel; i++) {
|
|
parts.push(row[`GID_${i}`] || '');
|
|
}
|
|
return parts.join('|');
|
|
};
|
|
const nameLookup = new Map();
|
|
for (const row of completeRows) {
|
|
nameLookup.set(gidKey(row), row);
|
|
}
|
|
for (const feature of features) {
|
|
const key = gidKey(feature.properties);
|
|
const correct = nameLookup.get(key);
|
|
if (correct) {
|
|
for (let i = 0; i <= resolvedLevel; i++) {
|
|
if (correct[`NAME_${i}`]) {
|
|
feature.properties[`NAME_${i}`] = correct[`NAME_${i}`];
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
// Filter to the requested area
|
|
const id = name || admin;
|
|
const colPrefix = name ? 'NAME_' : 'GID_';
|
|
const filterCol = `${colPrefix}${level}`;
|
|
features = features.filter(f => f.properties[filterCol]?.toLowerCase() === id.toLowerCase());
|
|
return features;
|
|
}
|
|
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"items.js","sourceRoot":"","sources":["../src/items.ts"],"names":[],"mappings":"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"}
|