2022-03-21 00:53:16 +08:00
|
|
|
import url from "url";
|
|
|
|
import path from "path";
|
|
|
|
import fs from "fs-extra";
|
|
|
|
|
2022-12-03 09:43:14 +08:00
|
|
|
import YAML from 'js-yaml';
|
|
|
|
import metadataParser from 'markdown-yaml-metadata-parser';
|
2022-03-21 00:53:16 +08:00
|
|
|
|
|
|
|
import { renderMdx } from "./mdx.ts";
|
|
|
|
|
|
|
|
const PUBLIC_DIR = "public";
|
|
|
|
|
|
|
|
const PEOPLE_DIR = "people";
|
2022-04-09 05:38:52 +08:00
|
|
|
const COMMENTS_DIR = "comments";
|
2022-03-21 00:53:16 +08:00
|
|
|
|
|
|
|
const DIST_DIR = "dist";
|
|
|
|
|
|
|
|
const projectRoot = path.dirname(path.dirname(url.fileURLToPath(import.meta.url)));
|
|
|
|
const peopleDir = path.join(projectRoot, PEOPLE_DIR);
|
|
|
|
const people = fs.readdirSync(peopleDir).map(person => ({
|
|
|
|
dirname: person,
|
|
|
|
srcPath: path.join(peopleDir, person),
|
|
|
|
distPath: path.join(projectRoot, DIST_DIR, PEOPLE_DIR, person)
|
|
|
|
}));
|
|
|
|
|
|
|
|
// Transform `info.json5` to `info.json`.
|
|
|
|
// Extract metadata from `people/${dirname}/info.json5` to `dist/people-list.json`.
|
|
|
|
function buildPeopleInfoAndList() {
|
|
|
|
const PEOPLE_LIST_KEYS = ["id", "name", "profileUrl"] as const;
|
|
|
|
type PeopleMeta = Record<"path" | typeof PEOPLE_LIST_KEYS[number], unknown>;
|
|
|
|
|
2022-12-03 09:43:14 +08:00
|
|
|
// Read internationalized key names
|
|
|
|
const infoKeys = YAML.load(fs.readFileSync('info-i18n.yml'))
|
|
|
|
|
|
|
|
// Compile into multiple languages
|
|
|
|
for (const lang of ['', '.zh_hant']) {
|
|
|
|
|
|
|
|
// Compiled meta of list of people for the front page (contains keys id, name, profileUrl)
|
|
|
|
const peopleList: PeopleMeta[] = [];
|
|
|
|
|
|
|
|
// For each person
|
|
|
|
for (const { dirname, srcPath, distPath } of people) {
|
2022-12-03 09:59:51 +08:00
|
|
|
const infoFile = fs.readFileSync(path.join(srcPath, `info.yml`), "utf-8");
|
|
|
|
const info = YAML.load(infoFile);
|
2022-11-17 14:15:59 +08:00
|
|
|
|
2022-12-03 09:43:14 +08:00
|
|
|
// Read the page.md of that language
|
|
|
|
const markdown = fs.readFileSync(path.join(srcPath, `page${lang}.md`), "utf-8");
|
2022-11-17 14:15:59 +08:00
|
|
|
|
2022-12-03 09:43:14 +08:00
|
|
|
// Get the markdown header
|
|
|
|
const mdMeta = metadataParser(markdown).metadata
|
|
|
|
info.name = mdMeta.name
|
|
|
|
|
2022-12-03 09:59:51 +08:00
|
|
|
// Convert website dict into entries [[k, v], ...]
|
|
|
|
info.websites = Object.entries(info.websites ?? {})
|
|
|
|
|
2022-12-03 09:43:14 +08:00
|
|
|
// Convert info dict to [[key, value], ...]
|
|
|
|
// And add info k-v pairs from markdown to the info object in json5
|
|
|
|
info.info = [...Object.entries(mdMeta.info ?? {}), ...Object.entries(info.info ?? {})]
|
|
|
|
|
|
|
|
// Convert key names to internationalized key names
|
|
|
|
let langKey = indexTrim(lang, ".")
|
|
|
|
if (langKey == '') langKey = "zh_hans"
|
|
|
|
const keys = infoKeys[langKey]['key']
|
|
|
|
info.info = info.info.map(pair => [pair[0] in keys ? keys[pair[0]] : pair[0], pair[1]])
|
2022-11-17 14:15:59 +08:00
|
|
|
|
|
|
|
// Combine comments in people/${dirname}/comments/${cf}.json
|
|
|
|
const commentPath = path.join(srcPath, COMMENTS_DIR)
|
|
|
|
fs.ensureDirSync(commentPath)
|
|
|
|
info.comments = fs.readdirSync(commentPath)
|
2022-04-09 05:38:52 +08:00
|
|
|
.filter(cf => cf.endsWith('.json'))
|
|
|
|
.map(cf => JSON.parse(fs.readFileSync(path.join(commentPath, cf), 'utf-8')))
|
|
|
|
|
2022-11-17 14:15:59 +08:00
|
|
|
// Write info.json
|
|
|
|
fs.ensureDirSync(distPath);
|
|
|
|
fs.writeFileSync(path.join(distPath, `info${lang}.json`), JSON.stringify(info));
|
2022-12-03 09:43:14 +08:00
|
|
|
|
|
|
|
// Create people list meta information
|
|
|
|
const peopleMeta = {
|
|
|
|
path: dirname,
|
|
|
|
...Object.fromEntries(PEOPLE_LIST_KEYS.map(key => [key, info[key]]))
|
|
|
|
} as PeopleMeta;
|
|
|
|
|
|
|
|
// Add meta to people list
|
|
|
|
if (peopleList.filter(it => it.id == peopleMeta.id).length == 0)
|
|
|
|
peopleList.push(peopleMeta);
|
2022-11-17 14:15:59 +08:00
|
|
|
}
|
2022-03-21 00:53:16 +08:00
|
|
|
|
2022-12-03 09:43:14 +08:00
|
|
|
// Write people-list.json
|
|
|
|
fs.writeFileSync(path.join(projectRoot, DIST_DIR, `people-list${lang}.json`), JSON.stringify(peopleList));
|
|
|
|
}
|
2022-03-21 00:53:16 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Render `people/${dirname}/page.md` to `dist/people/${dirname}/page.js`.
|
|
|
|
function buildPeoplePages() {
|
|
|
|
for (const { srcPath, distPath } of people) {
|
2022-11-17 14:15:59 +08:00
|
|
|
for (const lang of ['', '.zh_hant'])
|
|
|
|
{
|
2022-12-03 09:43:14 +08:00
|
|
|
// Read markdown page and remove markdown meta
|
|
|
|
const markdown = metadataParser(fs.readFileSync(path.join(srcPath, `page${lang}.md`), "utf-8")).content;
|
2022-11-17 14:15:59 +08:00
|
|
|
const result = renderMdx(markdown);
|
2022-03-21 00:53:16 +08:00
|
|
|
|
2022-11-17 14:15:59 +08:00
|
|
|
fs.ensureDirSync(distPath);
|
|
|
|
fs.writeFileSync(path.join(distPath, `page${lang}.js`), result);
|
|
|
|
}
|
2022-03-21 00:53:16 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-09 05:38:52 +08:00
|
|
|
// Copy `people/${dirname}/photos` to `dist/people/${dirname}/`.
|
2022-03-21 00:53:16 +08:00
|
|
|
function copyPeopleAssets() {
|
2022-04-09 05:38:52 +08:00
|
|
|
const PEOPLE_ASSETS = ["photos"];
|
2022-03-21 00:53:16 +08:00
|
|
|
|
|
|
|
for (const { srcPath, distPath } of people) {
|
|
|
|
fs.ensureDirSync(distPath);
|
|
|
|
|
|
|
|
for (const assetDirname of PEOPLE_ASSETS) {
|
|
|
|
const assetSrcPath = path.join(srcPath, assetDirname);
|
|
|
|
if (fs.pathExistsSync(assetSrcPath)) {
|
|
|
|
fs.copySync(assetSrcPath, path.join(distPath, assetDirname));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Copy files `public` to dist.
|
|
|
|
function copyPublic() {
|
|
|
|
fs.copySync(path.join(projectRoot, PUBLIC_DIR), path.join(projectRoot, DIST_DIR));
|
|
|
|
}
|
|
|
|
|
|
|
|
buildPeopleInfoAndList();
|
|
|
|
buildPeoplePages();
|
|
|
|
copyPeopleAssets();
|
|
|
|
copyPublic();
|
2022-12-03 09:43:14 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Trim a specific char from a string
|
|
|
|
*
|
|
|
|
* @param str String
|
|
|
|
* @param ch Character (must have len 1)
|
|
|
|
*/
|
|
|
|
function indexTrim(str: string, ch: string) {
|
|
|
|
let start = 0
|
|
|
|
let end = str.length
|
|
|
|
|
|
|
|
while (start < end && str[start] === ch)
|
|
|
|
++start;
|
|
|
|
|
|
|
|
while (end > start && str[end - 1] === ch)
|
|
|
|
--end;
|
|
|
|
|
|
|
|
return (start > 0 || end < str.length) ? str.substring(start, end) : str;
|
|
|
|
}
|