215 lines
15 KiB
JavaScript
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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"}
|