wallet-core/packages/web-util/src/index.build.ts

243 lines
5.6 KiB
TypeScript
Raw Normal View History

2023-05-05 13:34:21 +02:00
import esbuild, { PluginBuild } from "esbuild";
2023-05-30 19:33:15 +02:00
import linaria from "@linaria/esbuild";
2023-04-24 15:56:45 +02:00
import fs from "fs";
2023-05-05 13:34:21 +02:00
import path from "path";
import postcss from "postcss";
2023-04-24 15:56:45 +02:00
import sass from "sass";
2023-05-05 13:34:21 +02:00
import postcssrc from "postcss-load-config";
2023-04-24 15:56:45 +02:00
// this should give us the current directory where
// the project is being built
const BASE = process.cwd();
2023-05-05 13:34:21 +02:00
export function getFilesInDirectory(
startPath: string,
regex?: RegExp,
): string[] {
2023-04-24 15:56:45 +02:00
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);
}
2023-05-05 13:34:21 +02:00
if (!regex || regex.test(filename)) {
2023-04-24 15:56:45 +02:00
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<string>) {
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)$/;
2023-05-05 13:34:21 +02:00
const sassPlugin: esbuild.Plugin = {
2023-04-24 15:56:45 +02:00
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,
};
});
},
};
2023-05-05 13:34:21 +02:00
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,
};
});
},
};
2023-05-30 19:33:15 +02:00
/**
* This should be able to load the plugin but but some reason it does not work
*
* text: "Cannot find module '../plugins/preeval'\n" +
*
*/
2023-05-05 13:34:21 +02:00
function linariaPlugin() {
2023-05-30 19:33:15 +02:00
const linariaCssPlugin: esbuild.Plugin = (linaria as any)({
babelOptions: {
presets: ["@babel/preset-typescript", "@babel/preset-react", "@linaria"],
},
sourceMap: true,
});
return linariaCssPlugin;
2023-05-05 13:34:21 +02:00
}
const defaultEsBuildConfig: esbuild.BuildOptions = {
2023-04-24 15:56:45 +02:00
bundle: true,
minify: false,
loader: {
".svg": "file",
2023-05-30 19:33:15 +02:00
".inline.svg": "text",
2023-04-24 15:56:45 +02:00
".png": "dataurl",
".jpeg": "dataurl",
".ttf": "file",
".woff": "file",
".woff2": "file",
".eot": "file",
},
target: ["es6"],
format: "esm",
platform: "browser",
sourcemap: true,
jsxFactory: "h",
jsxFragment: "Fragment",
2023-05-30 19:33:15 +02:00
alias: {
react: "preact/compat",
"react-dom": "preact/compat",
},
2023-04-24 15:56:45 +02:00
define: {
__VERSION__: `"${_package.version}"`,
__GIT_HASH__: `"${GIT_HASH}"`,
},
};
2023-05-05 13:34:21 +02:00
export interface BuildParams {
source: {
assets: string[];
js: string[];
};
2023-05-30 19:33:15 +02:00
public?: string;
2023-05-05 13:34:21 +02:00
destination: string;
2023-05-30 19:33:15 +02:00
css: "sass" | "postcss" | "linaria";
linariaPlugin?: () => esbuild.Plugin;
2023-05-05 13:34:21 +02:00
}
2023-05-30 19:33:15 +02:00
export function computeConfig(params: BuildParams): esbuild.BuildOptions {
2023-05-05 13:34:21 +02:00
const plugins: Array<esbuild.Plugin> = [
copyFilesPlugin(params.source.assets),
];
switch (params.css) {
case "sass": {
plugins.push(sassPlugin);
break;
}
case "postcss": {
plugins.push(postCssPlugin);
break;
}
2023-05-30 19:33:15 +02:00
case "linaria": {
if (params.linariaPlugin) {
plugins.push(params.linariaPlugin());
}
break;
}
2023-05-05 13:34:21 +02:00
default: {
const cssType: never = params.css;
throw Error(`not supported: ${cssType}`);
}
}
return {
...defaultEsBuildConfig,
entryPoints: params.source.js,
2023-05-30 19:33:15 +02:00
publicPath: params.public,
2023-05-05 13:34:21 +02:00
outdir: params.destination,
plugins,
};
2023-04-24 15:56:45 +02:00
}
/**
2023-05-05 13:34:21 +02:00
* Build sources for prod environment
2023-04-24 15:56:45 +02:00
*/
2023-05-05 13:34:21 +02:00
export function build(config: BuildParams) {
return esbuild.build(computeConfig(config));
2023-04-24 15:56:45 +02:00
}
2023-05-05 13:34:21 +02:00
const LIVE_RELOAD_SCRIPT =
"./node_modules/@gnu-taler/web-util/lib/live-reload.mjs";
2023-04-24 15:56:45 +02:00
/**
* Do startup for development environment
*/
export function initializeDev(
2023-05-05 13:34:21 +02:00
config: BuildParams,
2023-04-24 15:56:45 +02:00
): () => Promise<esbuild.BuildResult> {
2023-05-05 13:34:21 +02:00
function buildDevelopment() {
const result = computeConfig(config);
result.inject = [LIVE_RELOAD_SCRIPT];
return esbuild.build(result);
}
return buildDevelopment;
2023-04-24 15:56:45 +02:00
}