gadm-ts/dist/tree.js
2026-03-23 17:35:02 +01:00

215 lines
15 KiB
JavaScript

/**
* GADMTree — hierarchical tree built from the flat parquet database.
*
* Walk the tree from a country or province down to the deepest GADM level.
* Cache is optional and directory is supplied by the caller.
*/
import { loadDatabase } from './database.js';
import { createHash } from 'crypto';
import { existsSync, readFileSync, writeFileSync, mkdirSync } from 'fs';
import { join, dirname } from 'path';
// ────────── cache helpers ──────────
function cacheKey(opts) {
const raw = `tree_${opts.name || ''}_${opts.admin || ''}`;
return createHash('md5').update(raw).digest('hex');
}
function readCache(dir, key) {
const fp = join(dir, `tree_${key}.json`);
if (!existsSync(fp))
return null;
try {
return JSON.parse(readFileSync(fp, 'utf-8'));
}
catch {
return null;
}
}
function writeCache(dir, key, tree) {
try {
const fp = join(dir, `tree_${key}.json`);
mkdirSync(dirname(fp), { recursive: true });
writeFileSync(fp, JSON.stringify(tree));
}
catch (e) {
console.error('[GADMTree] cache write failed:', e);
}
}
// ────────── core ──────────
/**
* Build a hierarchical GADMTree for a region.
*
* ```ts
* const tree = await buildTree({ name: 'Spain', cacheDir: './cache/gadm' });
* // tree.root.children → 18 comunidades
* // tree.root.children[5].children → 4 provinces of Cataluña
* ```
*/
export async function buildTree(opts) {
if (!opts.name && !opts.admin) {
throw new Error('buildTree requires either "name" or "admin".');
}
if (opts.name && opts.admin) {
throw new Error('"name" and "admin" cannot both be set.');
}
// ── check cache ──
const key = cacheKey(opts);
if (opts.cacheDir) {
const cached = readCache(opts.cacheDir, key);
if (cached)
return cached;
}
const db = await loadDatabase();
// ── find root level ──
const isName = !!opts.name;
const id = (opts.name || opts.admin);
const colPrefix = isName ? 'NAME_' : 'GID_';
const rootLevel = findLevel(db, colPrefix, id);
if (rootLevel === -1) {
throw new Error(`"${id}" not found in GADM database.`);
}
// ── filter rows that belong to this root ──
const col = `${colPrefix}${rootLevel}`;
const rows = db.filter(r => eq(r[col], id));
// ── find max available level ──
let maxLevel = rootLevel;
for (let l = 5; l > rootLevel; l--) {
if (rows.some(r => r[`GID_${l}`] && r[`GID_${l}`] !== '')) {
maxLevel = l;
break;
}
}
// ── build root node ──
const rootName = rows[0]?.[`NAME_${rootLevel}`] || id;
const rootGid = rows[0]?.[`GID_${rootLevel}`] || id;
const root = { name: rootName, gid: rootGid, level: rootLevel, children: [] };
// ── recursively attach children ──
let nodeCount = 1; // root counts
nodeCount += attachChildren(root, rows, rootLevel + 1, maxLevel);
const tree = { root, maxLevel, nodeCount };
// ── persist cache ──
if (opts.cacheDir) {
writeCache(opts.cacheDir, key, tree);
}
return tree;
}
// ────────── iterators ──────────
/**
* Depth-first walk of the tree. Yields every node top-down.
*
* ```ts
* for (const node of walkDFS(tree.root)) {
* console.log(' '.repeat(node.level) + node.name);
* }
* ```
*/
export function* walkDFS(node) {
yield node;
for (const child of node.children) {
yield* walkDFS(child);
}
}
/**
* Breadth-first walk. Yields nodes level by level.
*/
export function* walkBFS(node) {
const queue = [node];
while (queue.length > 0) {
const current = queue.shift();
yield current;
for (const child of current.children) {
queue.push(child);
}
}
}
/**
* Yield only nodes at a specific admin level.
*
* ```ts
* const municipalities = [...walkLevel(tree.root, 4)];
* ```
*/
export function* walkLevel(node, level) {
if (node.level === level) {
yield node;
return; // don't go deeper
}
for (const child of node.children) {
yield* walkLevel(child, level);
}
}
/**
* Yield only leaf nodes (deepest level, no children).
*/
export function* leaves(node) {
if (node.children.length === 0) {
yield node;
}
else {
for (const child of node.children) {
yield* leaves(child);
}
}
}
/**
* Find a node by name or GID (case-insensitive). Returns first match via DFS.
*/
export function findNode(root, query) {
const q = query.toLowerCase();
for (const node of walkDFS(root)) {
if (node.name.toLowerCase() === q || node.gid.toLowerCase() === q) {
return node;
}
}
return undefined;
}
// ────────── internal helpers ──────────
function eq(a, b) {
if (!a)
return false;
return a.toLowerCase() === b.toLowerCase();
}
function findLevel(db, colPrefix, id) {
const low = id.toLowerCase();
for (let l = 0; l < 6; l++) {
const col = `${colPrefix}${l}`;
if (db.some(r => r[col]?.toLowerCase() === low))
return l;
}
return -1;
}
/**
* Group `rows` by unique (NAME_level, GID_level), create child GADMNodes,
* then recurse for each child's subset of rows.
* Returns the count of nodes added.
*/
function attachChildren(parent, rows, level, maxLevel) {
if (level > maxLevel)
return 0;
const nameCol = `NAME_${level}`;
const gidCol = `GID_${level}`;
// Deduplicate by GID at this level
const groups = new Map();
for (const r of rows) {
const gid = r[gidCol];
const name = r[nameCol];
if (!gid || gid === '')
continue;
if (!groups.has(gid)) {
groups.set(gid, { name: name || gid, gid, rows: [] });
}
groups.get(gid).rows.push(r);
}
if (groups.size === 0)
return 0;
let count = 0;
// Sort children alphabetically by name
const sorted = [...groups.values()].sort((a, b) => a.name.localeCompare(b.name));
for (const g of sorted) {
const node = { name: g.name, gid: g.gid, level, children: [] };
parent.children.push(node);
count++;
count += attachChildren(node, g.rows, level + 1, maxLevel);
}
return count;
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tree.js","sourceRoot":"","sources":["../src/tree.ts"],"names":[],"mappings":"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"}