import esbuild, { PluginBuild } from "esbuild"; import linaria from "@linaria/esbuild"; import fs from "fs"; import path from "path"; import postcss from "postcss"; import sass from "sass"; import postcssrc from "postcss-load-config"; // this should give us the current directory where // the project is being built const BASE = process.cwd(); export function getFilesInDirectory( startPath: string, regex?: RegExp, ): string[] { if (!fs.existsSync(startPath)) { return []; } const files = fs.readdirSync(startPath); const result = files .flatMap((file) => { const filename = path.join(startPath, file); const stat = fs.lstatSync(filename); if (stat.isDirectory()) { return getFilesInDirectory(filename, regex); } if (!regex || regex.test(filename)) { return [filename]; } return []; }) .filter((x) => !!x); return result; } let GIT_ROOT = BASE; while (!fs.existsSync(path.join(GIT_ROOT, ".git")) && GIT_ROOT !== "/") { GIT_ROOT = path.join(GIT_ROOT, "../"); } if (GIT_ROOT === "/") { // eslint-disable-next-line no-undef console.log("not found"); // eslint-disable-next-line no-undef process.exit(1); } const GIT_HASH = GIT_ROOT === "/" ? undefined : git_hash(); const buf = fs.readFileSync(path.join(BASE, "package.json")); let _package = JSON.parse(buf.toString("utf-8")); function git_hash() { const rev = fs .readFileSync(path.join(GIT_ROOT, ".git", "HEAD")) .toString() .trim() .split(/.*[: ]/) .slice(-1)[0]; if (rev.indexOf("/") === -1) { return rev; } else { return fs.readFileSync(path.join(GIT_ROOT, ".git", rev)).toString().trim(); } } // FIXME: Put this into some helper library. function copyFilesPlugin(files: Array) { return { name: "copy-files", setup(build: PluginBuild) { const outDir = build.initialOptions.outdir; if (outDir === undefined) throw Error("esbuild build options does not specify outdir"); build.onEnd(() => { for (const file of files) { const name = path.parse(file).base; fs.copyFileSync(file, path.join(outDir, name)); } }); }, }; } const DEFAULT_SASS_FILTER = /\.(s[ac]ss|css)$/; const sassPlugin: esbuild.Plugin = { name: "custom-build-sass", setup(build) { build.onLoad({ filter: DEFAULT_SASS_FILTER }, ({ path: file }) => { const resolveDir = path.dirname(file); const { css: contents } = sass.compile(file, { loadPaths: ["./"] }); return { resolveDir, loader: "css", contents, }; }); }, }; const postCssPlugin: esbuild.Plugin = { name: "custom-build-postcss", setup(build) { build.onLoad({ filter: DEFAULT_SASS_FILTER }, async ({ path: file }) => { const resolveDir = path.dirname(file); const sourceBuffer = fs.readFileSync(file); const source = sourceBuffer.toString("utf-8"); const postCssConfig = await postcssrc(); postCssConfig.options.from = file; const { css: contents } = await postcss(postCssConfig.plugins).process( source, postCssConfig.options, ); return { resolveDir, loader: "css", contents, }; }); }, }; /** * This should be able to load the plugin but but some reason it does not work * * text: "Cannot find module '../plugins/preeval'\n" + * */ function linariaPlugin() { const linariaCssPlugin: esbuild.Plugin = (linaria as any)({ babelOptions: { presets: ["@babel/preset-typescript", "@babel/preset-react", "@linaria"], }, sourceMap: true, }); return linariaCssPlugin; } const defaultEsBuildConfig: esbuild.BuildOptions = { bundle: true, minify: false, loader: { ".svg": "file", ".inline.svg": "text", ".png": "dataurl", ".jpeg": "dataurl", ".ttf": "file", ".woff": "file", ".woff2": "file", ".eot": "file", }, target: ["es6"], format: "esm", platform: "browser", sourcemap: true, jsxFactory: "h", jsxFragment: "Fragment", alias: { react: "preact/compat", "react-dom": "preact/compat", }, define: { __VERSION__: `"${_package.version}"`, __GIT_HASH__: `"${GIT_HASH}"`, }, }; export interface BuildParams { source: { assets: string[]; js: string[]; }; public?: string; destination: string; css: "sass" | "postcss" | "linaria"; linariaPlugin?: () => esbuild.Plugin; } export function computeConfig(params: BuildParams): esbuild.BuildOptions { const plugins: Array = [ copyFilesPlugin(params.source.assets), ]; switch (params.css) { case "sass": { plugins.push(sassPlugin); break; } case "postcss": { plugins.push(postCssPlugin); break; } case "linaria": { if (params.linariaPlugin) { plugins.push(params.linariaPlugin()); } break; } default: { const cssType: never = params.css; throw Error(`not supported: ${cssType}`); } } return { ...defaultEsBuildConfig, entryPoints: params.source.js, publicPath: params.public, outdir: params.destination, plugins, }; } /** * Build sources for prod environment */ export function build(config: BuildParams) { return esbuild.build(computeConfig(config)); } const LIVE_RELOAD_SCRIPT = "./node_modules/@gnu-taler/web-util/lib/live-reload.mjs"; /** * Do startup for development environment */ export function initializeDev( config: BuildParams, ): () => Promise { function buildDevelopment() { const result = computeConfig(config); result.inject = [LIVE_RELOAD_SCRIPT]; return esbuild.build(result); } return buildDevelopment; }