thefoldwithin-earth/tools/generate-index.mjs

82 lines
2.4 KiB
JavaScript
Raw Normal View History

2025-11-08 11:23:21 -06:00
#!/usr/bin/env node
2025-11-08 11:29:10 -06:00
import { promises as fs } from "fs";
2025-11-08 09:05:04 -06:00
import path from "path";
2025-11-08 15:47:51 -06:00
import pdf from "pdf-parse";
2025-11-08 14:34:15 -06:00
2025-11-08 15:24:49 -06:00
const ROOT = "public";
2025-11-08 11:23:21 -06:00
const OUT = path.join(ROOT, "index.json");
2025-11-08 11:29:10 -06:00
const STATIC_TOPLEVEL = new Set(["about", "contact", "legal"]);
2025-11-08 11:23:21 -06:00
const MAX_BYTES = 64 * 1024;
2025-11-08 10:37:54 -06:00
2025-11-08 11:29:10 -06:00
function dateFromName(name) {
const m = name.match(/^(\d{4}-\d{2}-\d{2})/);
return m ? new Date(m[0]).getTime() : null;
2025-11-08 10:37:54 -06:00
}
2025-11-08 15:24:49 -06:00
2025-11-08 11:29:10 -06:00
async function readHead(abs) {
const fh = await fs.open(abs, "r");
const buf = Buffer.alloc(MAX_BYTES);
const { bytesRead } = await fh.read(buf, 0, MAX_BYTES, 0);
2025-11-08 10:37:54 -06:00
await fh.close();
2025-11-08 11:29:10 -06:00
return buf.slice(0, bytesRead).toString("utf8");
2025-11-08 10:37:54 -06:00
}
2025-11-08 15:24:49 -06:00
2025-11-08 11:29:10 -06:00
function parseTitle(raw, ext) {
2025-11-08 15:24:49 -06:00
if (ext === ".md") return raw.match(/^\s*#\s+(.+?)\s*$/m)?.[1].trim();
if (ext === ".html") return raw.match(/<title[^>]*>([^<]+)<\/title>/i)?.[1].trim();
2025-11-08 10:37:54 -06:00
return null;
}
2025-11-08 15:34:32 -06:00
async function collectFiles(relBase = "", flat = []) {
2025-11-08 11:30:41 -06:00
const abs = path.join(ROOT, relBase);
const entries = await fs.readdir(abs, { withFileTypes: true });
2025-11-08 14:34:15 -06:00
2025-11-08 11:30:41 -06:00
for (const e of entries) {
2025-11-08 14:40:38 -06:00
if (e.name.startsWith(".")) continue;
2025-11-08 11:30:41 -06:00
const rel = path.posix.join(relBase, e.name);
const absPath = path.join(ROOT, rel);
if (e.isDirectory()) {
const top = rel.split("/")[0];
if (STATIC_TOPLEVEL.has(top)) continue;
2025-11-08 15:34:32 -06:00
await collectFiles(rel, flat);
2025-11-08 11:30:41 -06:00
continue;
}
2025-11-08 14:34:15 -06:00
2025-11-08 11:30:41 -06:00
const ext = path.posix.extname(e.name).toLowerCase();
2025-11-08 15:47:51 -06:00
if (![".md", ".html", ".pdf"].includes(ext)) continue;
2025-11-08 11:30:41 -06:00
const st = await fs.stat(absPath);
2025-11-08 15:47:51 -06:00
let title;
if (ext === ".pdf") {
const buffer = await fs.readFile(absPath);
const pdfData = await pdf(buffer);
title = pdfData.info.Title || e.name.replace(/\.pdf$/, "").trim();
} else {
const raw = await readHead(absPath);
title = parseTitle(raw, ext) || e.name.replace(new RegExp(`\\${ext}$`), "").trim();
}
2025-11-08 11:30:41 -06:00
const mtime = dateFromName(e.name) ?? st.mtimeMs;
2025-11-08 15:24:49 -06:00
2025-11-08 15:34:32 -06:00
flat.push({
2025-11-08 11:30:41 -06:00
type: "file",
name: e.name,
title,
path: rel,
ext,
pinned: rel.startsWith("pinned/"),
2025-11-08 14:40:38 -06:00
mtime
2025-11-08 11:30:41 -06:00
});
}
2025-11-08 15:34:32 -06:00
return flat;
2025-11-08 11:30:41 -06:00
}
(async () => {
try {
2025-11-08 15:34:32 -06:00
const flat = await collectFiles();
2025-11-08 14:40:38 -06:00
const sections = [...new Set(flat.map(f => f.path.split("/")[0]))];
2025-11-08 15:34:32 -06:00
await fs.writeFile(OUT, JSON.stringify({ flat, sections }, null, 2));
2025-11-08 15:24:49 -06:00
console.log(`index.json built with ${flat.length} files across ${sections.length} sections.`);
2025-11-08 11:30:41 -06:00
} catch (e) {
2025-11-08 15:24:49 -06:00
console.error("Build failed:", e);
2025-11-08 11:30:41 -06:00
process.exit(1);
}
})();