silverbullet/plugos/compile.ts

184 lines
4.9 KiB
TypeScript
Raw Normal View History

import { denoPlugins, esbuild, path, YAML } from "./deps.ts";
import { bundleAssets } from "./asset_bundle/builder.ts";
2022-10-13 21:16:18 +08:00
import { Manifest } from "./types.ts";
import { version } from "../version.ts";
2023-10-03 21:27:56 +08:00
// const workerRuntimeUrl = new URL("./worker_runtime.ts", import.meta.url);
const workerRuntimeUrl =
`https://deno.land/x/silverbullet@${version}/plugos/worker_runtime.ts`;
export type CompileOptions = {
debug?: boolean;
runtimeUrl?: string;
importMap?: string;
2022-10-13 21:16:18 +08:00
// Reload plug import cache
reload?: boolean;
// Print info on bundle size
info?: boolean;
};
export async function compileManifest(
manifestPath: string,
destPath: string,
options: CompileOptions = {},
): Promise<string> {
const rootPath = path.dirname(manifestPath);
const manifest = YAML.parse(
await Deno.readTextFile(manifestPath),
) as Manifest<any>;
if (!manifest.name) {
throw new Error(`Missing 'name' in ${manifestPath}`);
2022-10-13 21:16:18 +08:00
}
// Assets
const assetsBundle = await bundleAssets(
path.resolve(rootPath),
manifest.assets as string[] || [],
);
manifest.assets = assetsBundle.toJSON();
const jsFile = `
import { setupMessageListener } from "${
options.runtimeUrl || workerRuntimeUrl
}";
// Imports
${
Object.entries(manifest.functions).map(([funcName, def]) => {
if (!def.path) {
return "";
2022-10-13 21:16:18 +08:00
}
let [filePath, jsFunctionName] = def.path.split(":");
// Resolve path
filePath = path.join(rootPath, filePath);
2022-10-13 21:16:18 +08:00
return `import {${jsFunctionName} as ${funcName}} from "file://${
2023-01-02 03:28:25 +08:00
// Replacaing \ with / for Windows
2023-01-23 01:53:14 +08:00
path.resolve(filePath).replaceAll(
"\\",
"\\\\",
)}";\n`;
}).join("")
}
// Function mapping
export const functionMapping = {
${
Object.entries(manifest.functions).map(([funcName, def]) => {
if (!def.path) {
return "";
}
return ` ${funcName}: ${funcName},\n`;
}).join("")
}
};
const manifest = ${JSON.stringify(manifest, null, 2)};
setupMessageListener(functionMapping, manifest);
`;
// console.log("Code:", jsFile);
const inFile = await Deno.makeTempFile({ suffix: ".js" });
const outFile = `${destPath}/${manifest.name}.plug.js`;
await Deno.writeTextFile(inFile, jsFile);
const result = await esbuild.build({
entryPoints: [path.basename(inFile)],
bundle: true,
format: "iife",
globalName: "mod",
platform: "browser",
sourcemap: options.debug ? "linked" : false,
minify: !options.debug,
outfile: outFile,
metafile: options.info,
treeShaking: true,
plugins: [
{
name: "json",
setup: (build) =>
build.onLoad({ filter: /\.json$/ }, () => ({ loader: "json" })),
},
...denoPlugins({
// TODO do this differently
importMapURL: options.importMap ||
new URL("../import_map.json", import.meta.url).toString(),
loader: "native",
}),
],
absWorkingDir: path.resolve(path.dirname(inFile)),
});
if (options.info) {
const text = await esbuild.analyzeMetafile(result.metafile!);
console.log("Bundle info for", manifestPath, text);
}
let jsCode = await Deno.readTextFile(outFile);
jsCode = patchDenoLibJS(jsCode);
await Deno.writeTextFile(outFile, jsCode);
console.log(`Plug ${manifest.name} written to ${outFile}.`);
return outFile;
}
export async function compileManifests(
manifestFiles: string[],
dist: string,
watch: boolean,
options: CompileOptions = {},
) {
let building = false;
dist = path.resolve(dist);
async function buildAll() {
if (building) {
return;
}
console.log("Building", manifestFiles);
building = true;
Deno.mkdirSync(dist, { recursive: true });
const startTime = Date.now();
// Build all plugs in parallel
await Promise.all(manifestFiles.map(async (plugManifestPath) => {
const manifestPath = plugManifestPath as string;
try {
await compileManifest(
manifestPath,
dist,
options,
);
} catch (e) {
console.error(`Error building ${manifestPath}:`, e);
}
}));
console.log(`Done building plugs in ${Date.now() - startTime}ms`);
building = false;
}
await buildAll();
if (watch) {
console.log("Watching for changes...");
const watcher = Deno.watchFs(manifestFiles.map((p) => path.dirname(p)));
for await (const event of watcher) {
if (event.paths.length > 0) {
2023-08-15 13:55:36 +08:00
if (
event.paths[0].endsWith(".json") || event.paths[0].startsWith(dist)
) {
continue;
}
}
console.log("Change detected, rebuilding...");
await buildAll();
}
}
}
export function patchDenoLibJS(code: string): string {
// The Deno std lib has one occurence of a regex that Webkit JS doesn't (yet parse), we'll strip it because it's likely never invoked anyway, YOLO
return code.replaceAll("/(?<=\\n)/", "/()/");
}