2024-03-01 09:23:35 -05:00
|
|
|
import proc from "node:child_process";
|
|
|
|
import fs from "node:fs/promises";
|
|
|
|
import ph from "node:path";
|
|
|
|
import os from "node:os";
|
|
|
|
import url from "node:url";
|
|
|
|
|
|
|
|
import * as marked from "marked";
|
|
|
|
import SVGSpriter from "svg-sprite";
|
|
|
|
import Watcher from "watcher";
|
|
|
|
import gettext from "gettext-parser";
|
|
|
|
import l from "lodash";
|
|
|
|
import log from "fancy-log";
|
|
|
|
import mustache from "mustache";
|
|
|
|
import pLimit from "p-limit";
|
|
|
|
import ppt from "pretty-time";
|
|
|
|
import wpool from "workerpool";
|
|
|
|
|
|
|
|
function getCoreCount() {
|
|
|
|
return os.cpus().length;
|
|
|
|
}
|
|
|
|
|
|
|
|
export const dirname = url.fileURLToPath(new URL(".", import.meta.url));
|
|
|
|
|
|
|
|
export function startWorker() {
|
|
|
|
return wpool.pool(dirname + "/_worker.js", {
|
2024-05-17 09:46:23 -05:00
|
|
|
maxWorkers: getCoreCount(),
|
2024-03-01 09:23:35 -05:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2024-05-17 09:46:23 -05:00
|
|
|
async function findFiles(basePath, predicate, options = {}) {
|
|
|
|
predicate =
|
|
|
|
predicate ??
|
|
|
|
function () {
|
|
|
|
return true;
|
|
|
|
};
|
2024-03-01 09:23:35 -05:00
|
|
|
|
2024-07-01 03:28:40 -05:00
|
|
|
let files = await fs.readdir(basePath, {
|
|
|
|
recursive: options.recursive ?? false,
|
|
|
|
});
|
2024-03-01 09:23:35 -05:00
|
|
|
files = files.map((path) => ph.join(basePath, path));
|
|
|
|
|
|
|
|
return files;
|
|
|
|
}
|
|
|
|
|
|
|
|
function syncDirs(originPath, destPath) {
|
|
|
|
const command = `rsync -ar --delete ${originPath} ${destPath}`;
|
|
|
|
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
proc.exec(command, (cause, stdout) => {
|
2024-05-17 09:46:23 -05:00
|
|
|
if (cause) {
|
|
|
|
reject(cause);
|
|
|
|
} else {
|
|
|
|
resolve();
|
|
|
|
}
|
2024-03-01 09:23:35 -05:00
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
export function isSassFile(path) {
|
|
|
|
return path.endsWith(".scss");
|
|
|
|
}
|
|
|
|
|
|
|
|
export function isSvgFile(path) {
|
2024-03-26 08:45:03 -05:00
|
|
|
return path.endsWith(".svg");
|
|
|
|
}
|
|
|
|
|
|
|
|
export function isJsFile(path) {
|
|
|
|
return path.endsWith(".js");
|
2024-03-01 09:23:35 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
export async function compileSass(worker, path, options) {
|
|
|
|
path = ph.resolve(path);
|
|
|
|
|
|
|
|
log.info("compile:", path);
|
|
|
|
return worker.exec("compileSass", [path, options]);
|
|
|
|
}
|
|
|
|
|
|
|
|
export async function compileSassAll(worker) {
|
|
|
|
const limitFn = pLimit(4);
|
|
|
|
const sourceDir = "src";
|
|
|
|
|
2024-05-17 09:46:23 -05:00
|
|
|
let files = await fs.readdir(sourceDir, { recursive: true });
|
2024-03-01 09:23:35 -05:00
|
|
|
files = files.filter((path) => path.endsWith(".scss"));
|
|
|
|
files = files.map((path) => ph.join(sourceDir, path));
|
|
|
|
|
|
|
|
const procs = [
|
|
|
|
compileSass(worker, "resources/styles/main-default.scss", {}),
|
2024-05-17 09:46:23 -05:00
|
|
|
compileSass(worker, "resources/styles/debug.scss", {}),
|
2024-03-01 09:23:35 -05:00
|
|
|
];
|
|
|
|
|
|
|
|
for (let path of files) {
|
2024-05-17 09:46:23 -05:00
|
|
|
const proc = limitFn(() => compileSass(worker, path, { modules: true }));
|
2024-03-01 09:23:35 -05:00
|
|
|
procs.push(proc);
|
|
|
|
}
|
|
|
|
|
|
|
|
const result = await Promise.all(procs);
|
|
|
|
|
2024-05-17 09:46:23 -05:00
|
|
|
return result.reduce(
|
|
|
|
(acc, item, index) => {
|
|
|
|
acc.index[item.outputPath] = item.css;
|
|
|
|
acc.items.push(item.outputPath);
|
|
|
|
return acc;
|
|
|
|
},
|
|
|
|
{ index: {}, items: [] },
|
|
|
|
);
|
2024-03-01 09:23:35 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
function compare(a, b) {
|
|
|
|
if (a < b) {
|
|
|
|
return -1;
|
|
|
|
} else if (a > b) {
|
|
|
|
return 1;
|
|
|
|
} else {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export function concatSass(data) {
|
2024-05-17 09:46:23 -05:00
|
|
|
const output = [];
|
2024-03-01 09:23:35 -05:00
|
|
|
|
|
|
|
for (let path of data.items) {
|
|
|
|
output.push(data.index[path]);
|
|
|
|
}
|
|
|
|
|
|
|
|
return output.join("\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
export async function watch(baseDir, predicate, callback) {
|
|
|
|
predicate = predicate ?? (() => true);
|
|
|
|
|
|
|
|
const watcher = new Watcher(baseDir, {
|
|
|
|
persistent: true,
|
2024-05-17 09:46:23 -05:00
|
|
|
recursive: true,
|
2024-03-01 09:23:35 -05:00
|
|
|
});
|
|
|
|
|
|
|
|
watcher.on("change", (path) => {
|
|
|
|
if (predicate(path)) {
|
|
|
|
callback(path);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
async function readShadowManifest() {
|
|
|
|
try {
|
2024-05-17 09:46:23 -05:00
|
|
|
const manifestPath = "resources/public/js/manifest.json";
|
2024-03-01 09:23:35 -05:00
|
|
|
let content = await fs.readFile(manifestPath, { encoding: "utf8" });
|
|
|
|
content = JSON.parse(content);
|
|
|
|
|
|
|
|
const index = {
|
|
|
|
config: "js/config.js?ts=" + Date.now(),
|
|
|
|
polyfills: "js/polyfills.js?ts=" + Date.now(),
|
|
|
|
};
|
|
|
|
|
|
|
|
for (let item of content) {
|
|
|
|
index[item.name] = "js/" + item["output-name"];
|
|
|
|
}
|
|
|
|
|
|
|
|
return index;
|
|
|
|
} catch (cause) {
|
|
|
|
return {
|
|
|
|
config: "js/config.js",
|
|
|
|
polyfills: "js/polyfills.js",
|
|
|
|
main: "js/main.js",
|
|
|
|
shared: "js/shared.js",
|
|
|
|
worker: "js/worker.js",
|
|
|
|
rasterizer: "js/rasterizer.js",
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-05-17 09:46:23 -05:00
|
|
|
async function renderTemplate(path, context = {}, partials = {}) {
|
|
|
|
const content = await fs.readFile(path, { encoding: "utf-8" });
|
2024-03-01 09:23:35 -05:00
|
|
|
|
|
|
|
const ts = Math.floor(new Date());
|
|
|
|
|
|
|
|
context = Object.assign({}, context, {
|
|
|
|
ts: ts,
|
2024-05-17 09:46:23 -05:00
|
|
|
isDebug: process.env.NODE_ENV !== "production",
|
2024-03-01 09:23:35 -05:00
|
|
|
});
|
|
|
|
|
|
|
|
return mustache.render(content, context, partials);
|
|
|
|
}
|
|
|
|
|
|
|
|
const renderer = {
|
|
|
|
link(href, title, text) {
|
|
|
|
return `<a href="${href}" target="_blank">${text}</a>`;
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
marked.use({ renderer });
|
|
|
|
|
|
|
|
async function readTranslations() {
|
|
|
|
const langs = [
|
|
|
|
"ar",
|
|
|
|
"ca",
|
|
|
|
"de",
|
|
|
|
"el",
|
|
|
|
"en",
|
|
|
|
"eu",
|
|
|
|
"it",
|
|
|
|
"es",
|
|
|
|
"fa",
|
|
|
|
"fr",
|
|
|
|
"he",
|
|
|
|
"nb_NO",
|
|
|
|
"pl",
|
|
|
|
"pt_BR",
|
|
|
|
"ro",
|
|
|
|
"id",
|
|
|
|
"ru",
|
|
|
|
"tr",
|
|
|
|
"zh_CN",
|
|
|
|
"zh_Hant",
|
|
|
|
"hr",
|
|
|
|
"gl",
|
|
|
|
"pt_PT",
|
|
|
|
"cs",
|
|
|
|
"fo",
|
|
|
|
"ko",
|
|
|
|
"lv",
|
|
|
|
"nl",
|
|
|
|
// this happens when file does not matches correct
|
|
|
|
// iso code for the language.
|
|
|
|
["ja_jp", "jpn_JP"],
|
|
|
|
["uk", "ukr_UA"],
|
2024-05-17 09:46:23 -05:00
|
|
|
"ha",
|
2024-03-01 09:23:35 -05:00
|
|
|
];
|
|
|
|
const result = {};
|
|
|
|
|
|
|
|
for (let lang of langs) {
|
|
|
|
let filename = `${lang}.po`;
|
|
|
|
if (l.isArray(lang)) {
|
|
|
|
filename = `${lang[1]}.po`;
|
|
|
|
lang = lang[0];
|
|
|
|
}
|
|
|
|
|
2024-07-01 03:28:40 -05:00
|
|
|
const content = await fs.readFile(`./translations/${filename}`, {
|
|
|
|
encoding: "utf-8",
|
|
|
|
});
|
2024-03-01 09:23:35 -05:00
|
|
|
|
|
|
|
lang = lang.toLowerCase();
|
|
|
|
|
|
|
|
const data = gettext.po.parse(content, "utf-8");
|
|
|
|
const trdata = data.translations[""];
|
|
|
|
|
|
|
|
for (let key of Object.keys(trdata)) {
|
|
|
|
if (key === "") continue;
|
|
|
|
const comments = trdata[key].comments || {};
|
|
|
|
|
|
|
|
if (l.isNil(result[key])) {
|
|
|
|
result[key] = {};
|
|
|
|
}
|
|
|
|
|
|
|
|
const isMarkdown = l.includes(comments.flag, "markdown");
|
|
|
|
|
|
|
|
const msgs = trdata[key].msgstr;
|
|
|
|
if (msgs.length === 1) {
|
|
|
|
let message = msgs[0];
|
|
|
|
if (isMarkdown) {
|
|
|
|
message = marked.parseInline(message);
|
|
|
|
}
|
|
|
|
|
|
|
|
result[key][lang] = message;
|
|
|
|
} else {
|
|
|
|
result[key][lang] = msgs.map((item) => {
|
|
|
|
if (isMarkdown) {
|
|
|
|
return marked.parseInline(item);
|
|
|
|
} else {
|
|
|
|
return item;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return JSON.stringify(result);
|
|
|
|
}
|
|
|
|
|
|
|
|
async function generateSvgSprite(files, prefix) {
|
|
|
|
const spriter = new SVGSpriter({
|
|
|
|
mode: {
|
2024-05-17 09:46:23 -05:00
|
|
|
symbol: { inline: true },
|
|
|
|
},
|
2024-03-01 09:23:35 -05:00
|
|
|
});
|
|
|
|
|
|
|
|
for (let path of files) {
|
2024-05-17 09:46:23 -05:00
|
|
|
const name = `${prefix}${ph.basename(path)}`;
|
|
|
|
const content = await fs.readFile(path, { encoding: "utf-8" });
|
2024-03-01 09:23:35 -05:00
|
|
|
spriter.add(name, name, content);
|
|
|
|
}
|
|
|
|
|
|
|
|
const { result } = await spriter.compileAsync();
|
|
|
|
const resource = result.symbol.sprite;
|
|
|
|
return resource.contents;
|
|
|
|
}
|
|
|
|
|
|
|
|
async function generateSvgSprites() {
|
2024-07-01 03:28:40 -05:00
|
|
|
await fs.mkdir("resources/public/images/sprites/symbol/", {
|
|
|
|
recursive: true,
|
|
|
|
});
|
2024-03-01 09:23:35 -05:00
|
|
|
|
|
|
|
const icons = await findFiles("resources/images/icons/", isSvgFile);
|
|
|
|
const iconsSprite = await generateSvgSprite(icons, "icon-");
|
2024-07-01 03:28:40 -05:00
|
|
|
await fs.writeFile(
|
|
|
|
"resources/public/images/sprites/symbol/icons.svg",
|
|
|
|
iconsSprite,
|
|
|
|
);
|
2024-03-01 09:23:35 -05:00
|
|
|
|
|
|
|
const cursors = await findFiles("resources/images/cursors/", isSvgFile);
|
|
|
|
const cursorsSprite = await generateSvgSprite(icons, "cursor-");
|
2024-07-01 03:28:40 -05:00
|
|
|
await fs.writeFile(
|
|
|
|
"resources/public/images/sprites/symbol/cursors.svg",
|
|
|
|
cursorsSprite,
|
|
|
|
);
|
2024-07-04 02:17:05 -05:00
|
|
|
|
|
|
|
const assets = await findFiles("resources/images/assets/", isSvgFile);
|
|
|
|
const assetsSprite = await generateSvgSprite(assets, "asset-");
|
|
|
|
await fs.writeFile(
|
|
|
|
"resources/public/images/sprites/assets.svg",
|
|
|
|
assetsSprite,
|
|
|
|
);
|
2024-03-01 09:23:35 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
async function generateTemplates() {
|
2024-05-17 09:48:03 -05:00
|
|
|
const isDebug = process.env.NODE_ENV !== "production";
|
2024-03-01 09:23:35 -05:00
|
|
|
await fs.mkdir("./resources/public/", { recursive: true });
|
|
|
|
|
|
|
|
const translations = await readTranslations();
|
|
|
|
const manifest = await readShadowManifest();
|
|
|
|
let content;
|
|
|
|
|
2024-07-01 03:28:40 -05:00
|
|
|
const iconsSprite = await fs.readFile(
|
|
|
|
"resources/public/images/sprites/symbol/icons.svg",
|
|
|
|
"utf8",
|
|
|
|
);
|
|
|
|
const cursorsSprite = await fs.readFile(
|
|
|
|
"resources/public/images/sprites/symbol/cursors.svg",
|
|
|
|
"utf8",
|
|
|
|
);
|
2024-07-04 02:17:05 -05:00
|
|
|
const assetsSprite = await fs.readFile(
|
|
|
|
"resources/public/images/sprites/assets.svg",
|
|
|
|
"utf-8",
|
|
|
|
);
|
2024-03-01 09:23:35 -05:00
|
|
|
const partials = {
|
|
|
|
"../public/images/sprites/symbol/icons.svg": iconsSprite,
|
|
|
|
"../public/images/sprites/symbol/cursors.svg": cursorsSprite,
|
2024-07-04 02:17:05 -05:00
|
|
|
"../public/images/sprites/assets.svg": assetsSprite,
|
2024-03-01 09:23:35 -05:00
|
|
|
};
|
|
|
|
|
2024-05-17 09:46:23 -05:00
|
|
|
const pluginRuntimeUri =
|
2024-07-01 03:28:40 -05:00
|
|
|
process.env.PENPOT_PLUGIN_DEV === "true"
|
|
|
|
? "http://localhost:4200"
|
|
|
|
: "./plugins-runtime";
|
2024-04-18 09:39:04 -05:00
|
|
|
|
2024-05-17 09:46:23 -05:00
|
|
|
content = await renderTemplate(
|
|
|
|
"resources/templates/index.mustache",
|
|
|
|
{
|
|
|
|
manifest: manifest,
|
|
|
|
translations: JSON.stringify(translations),
|
|
|
|
pluginRuntimeUri,
|
2024-05-17 09:48:03 -05:00
|
|
|
isDebug,
|
2024-05-17 09:46:23 -05:00
|
|
|
},
|
|
|
|
partials,
|
|
|
|
);
|
2024-03-01 09:23:35 -05:00
|
|
|
|
|
|
|
await fs.writeFile("./resources/public/index.html", content);
|
|
|
|
|
2024-07-01 08:52:15 -05:00
|
|
|
content = await renderTemplate(
|
|
|
|
"resources/templates/preview-body.mustache",
|
|
|
|
{
|
|
|
|
manifest: manifest,
|
|
|
|
},
|
|
|
|
partials,
|
|
|
|
);
|
2024-03-01 09:23:35 -05:00
|
|
|
await fs.writeFile("./.storybook/preview-body.html", content);
|
|
|
|
|
|
|
|
content = await renderTemplate("resources/templates/render.mustache", {
|
|
|
|
manifest: manifest,
|
|
|
|
translations: JSON.stringify(translations),
|
|
|
|
});
|
|
|
|
|
|
|
|
await fs.writeFile("./resources/public/render.html", content);
|
|
|
|
|
|
|
|
content = await renderTemplate("resources/templates/rasterizer.mustache", {
|
|
|
|
manifest: manifest,
|
|
|
|
translations: JSON.stringify(translations),
|
|
|
|
});
|
|
|
|
|
|
|
|
await fs.writeFile("./resources/public/rasterizer.html", content);
|
|
|
|
}
|
|
|
|
|
|
|
|
export async function compileStyles() {
|
|
|
|
const worker = startWorker();
|
|
|
|
const start = process.hrtime();
|
|
|
|
|
2024-05-17 09:46:23 -05:00
|
|
|
log.info("init: compile styles");
|
2024-03-01 09:23:35 -05:00
|
|
|
let result = await compileSassAll(worker);
|
|
|
|
result = concatSass(result);
|
|
|
|
|
|
|
|
await fs.mkdir("./resources/public/css", { recursive: true });
|
|
|
|
await fs.writeFile("./resources/public/css/main.css", result);
|
|
|
|
|
|
|
|
const end = process.hrtime(start);
|
|
|
|
log.info("done: compile styles", `(${ppt(end)})`);
|
|
|
|
worker.terminate();
|
|
|
|
}
|
|
|
|
|
|
|
|
export async function compileSvgSprites() {
|
|
|
|
const start = process.hrtime();
|
2024-05-17 09:46:23 -05:00
|
|
|
log.info("init: compile svgsprite");
|
2024-03-01 09:23:35 -05:00
|
|
|
await generateSvgSprites();
|
|
|
|
const end = process.hrtime(start);
|
|
|
|
log.info("done: compile svgsprite", `(${ppt(end)})`);
|
|
|
|
}
|
|
|
|
|
|
|
|
export async function compileTemplates() {
|
|
|
|
const start = process.hrtime();
|
2024-05-17 09:46:23 -05:00
|
|
|
log.info("init: compile templates");
|
2024-03-01 09:23:35 -05:00
|
|
|
await generateTemplates();
|
|
|
|
const end = process.hrtime(start);
|
|
|
|
log.info("done: compile templates", `(${ppt(end)})`);
|
|
|
|
}
|
|
|
|
|
|
|
|
export async function compilePolyfills() {
|
|
|
|
const start = process.hrtime();
|
2024-05-17 09:46:23 -05:00
|
|
|
log.info("init: compile polyfills");
|
2024-03-01 09:23:35 -05:00
|
|
|
|
2024-03-26 08:45:03 -05:00
|
|
|
const files = await findFiles("resources/polyfills/", isJsFile);
|
2024-03-01 09:23:35 -05:00
|
|
|
let result = [];
|
|
|
|
for (let path of files) {
|
2024-05-17 09:46:23 -05:00
|
|
|
const content = await fs.readFile(path, { encoding: "utf-8" });
|
2024-03-01 09:23:35 -05:00
|
|
|
result.push(content);
|
|
|
|
}
|
|
|
|
|
|
|
|
await fs.mkdir("./resources/public/js", { recursive: true });
|
|
|
|
fs.writeFile("resources/public/js/polyfills.js", result.join("\n"));
|
|
|
|
|
|
|
|
const end = process.hrtime(start);
|
|
|
|
log.info("done: compile polyfills", `(${ppt(end)})`);
|
|
|
|
}
|
|
|
|
|
|
|
|
export async function copyAssets() {
|
|
|
|
const start = process.hrtime();
|
2024-05-17 09:46:23 -05:00
|
|
|
log.info("init: copy assets");
|
2024-03-01 09:23:35 -05:00
|
|
|
|
|
|
|
await syncDirs("resources/images/", "resources/public/images/");
|
|
|
|
await syncDirs("resources/fonts/", "resources/public/fonts/");
|
2024-07-01 03:28:40 -05:00
|
|
|
await syncDirs(
|
|
|
|
"resources/plugins-runtime/",
|
|
|
|
"resources/public/plugins-runtime/",
|
|
|
|
);
|
2024-03-01 09:23:35 -05:00
|
|
|
|
|
|
|
const end = process.hrtime(start);
|
|
|
|
log.info("done: copy assets", `(${ppt(end)})`);
|
|
|
|
}
|