/** * 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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"}