import { copy } from "@std/fs"; import sass from "denosass"; import { bundleFolder } from "./lib/asset_bundle/builder.ts"; import { parseArgs } from "@std/cli/parse-args"; import { patchDenoLibJS } from "./cmd/compile.ts"; import { denoPlugins } from "@luca/esbuild-deno-loader"; import * as esbuild from "esbuild"; export async function bundleAll( watch: boolean, ): Promise { let building = false; await buildCopyBundleAssets(); let timer; if (watch) { const watcher = Deno.watchFs(["web", "dist_plug_bundle"]); for await (const _event of watcher) { if (timer) { clearTimeout(timer); } timer = setTimeout(() => { console.log("Change detected, rebuilding..."); if (building) { return; } building = true; buildCopyBundleAssets().finally(() => { building = false; }); }, 1000); } } } export async function copyAssets(dist: string) { await Deno.mkdir(dist, { recursive: true }); await copy("web/fonts", `${dist}`, { overwrite: true }); await copy("web/index.html", `${dist}/index.html`, { overwrite: true, }); await copy("web/auth.html", `${dist}/auth.html`, { overwrite: true, }); await copy("web/images/favicon.png", `${dist}/favicon.png`, { overwrite: true, }); await copy("web/images/logo.png", `${dist}/logo.png`, { overwrite: true, }); await copy("web/images/logo-dock.png", `${dist}/logo-dock.png`, { overwrite: true, }); await copy("web/manifest.json", `${dist}/manifest.json`, { overwrite: true, }); const compiler = sass( Deno.readTextFileSync("web/styles/main.scss"), { load_paths: ["web/styles"], }, ); await Deno.writeTextFile( `${dist}/main.css`, compiler.to_string("expanded") as string, ); // HACK: Patch the JS by removing an invalid regex let bundleJs = await Deno.readTextFile(`${dist}/client.js`); bundleJs = patchDenoLibJS(bundleJs); await Deno.writeTextFile(`${dist}/client.js`, bundleJs); } async function buildCopyBundleAssets() { await Deno.mkdir("dist_client_bundle", { recursive: true }); await Deno.mkdir("dist_plug_bundle", { recursive: true }); await bundleFolder( "dist_plug_bundle", "dist/plug_asset_bundle.json", ); console.log("Now ESBuilding the client and service workers..."); const result = await esbuild.build({ entryPoints: [ { in: "web/boot.ts", out: ".client/client", }, { in: "web/service_worker.ts", out: "service_worker", }, ], outdir: "dist_client_bundle", absWorkingDir: Deno.cwd(), bundle: true, treeShaking: true, sourcemap: "linked", minify: true, jsxFactory: "h", // metafile: true, jsx: "automatic", jsxFragment: "Fragment", jsxImportSource: "https://esm.sh/preact@10.23.1", plugins: denoPlugins({ configPath: new URL("./deno.json", import.meta.url).pathname, }), }); if (result.metafile) { const text = await esbuild.analyzeMetafile(result.metafile!); console.log("Bundle info", text); } // Patch the service_worker {{CACHE_NAME}} let swCode = await Deno.readTextFile("dist_client_bundle/service_worker.js"); swCode = swCode.replaceAll("{{CACHE_NAME}}", `cache-${Date.now()}`); await Deno.writeTextFile("dist_client_bundle/service_worker.js", swCode); await copyAssets("dist_client_bundle/.client"); await bundleFolder("dist_client_bundle", "dist/client_asset_bundle.json"); console.log("Built!"); } if (import.meta.main) { const args = parseArgs(Deno.args, { boolean: ["watch"], alias: { w: "watch" }, default: { watch: false, }, }); await bundleAll(args.watch); if (!args.watch) { esbuild.stop(); } }