heissepreise/bundle.js

144 lines
4.8 KiB
JavaScript
Raw Normal View History

2023-06-03 22:00:52 +02:00
const fs = require("fs");
const path = require("path");
2023-06-08 13:48:08 +02:00
const chokidar = require("chokidar");
const esbuild = require("esbuild");
const { exec } = require("child_process");
const { promisify } = require("util");
2023-06-03 22:00:52 +02:00
function deleteDirectory(directory) {
if (fs.existsSync(directory)) {
fs.readdirSync(directory).forEach((file) => {
const filePath = path.join(directory, file);
if (fs.statSync(filePath).isDirectory()) {
deleteDirectory(filePath);
} else {
fs.unlinkSync(filePath);
}
});
fs.rmdirSync(directory);
}
}
function replaceFileContents(string, fileDir) {
const pattern = /%%([^%]+)%%|\/\/\s*include\s*"([^"]+)"/g;
2023-06-03 22:00:52 +02:00
return string.replace(pattern, (_, filename1, filename2) => {
const filename = filename1 || filename2;
2023-06-03 22:00:52 +02:00
const filenamePath = path.join(fileDir, filename);
try {
const data = fs.readFileSync(filenamePath, "utf8");
const replacedData = replaceFileContents(data, path.dirname(filenamePath));
return replacedData;
2023-06-03 22:00:52 +02:00
} catch (error) {
console.error(`Error reading file "${filenamePath}":`, error);
return "";
2023-06-03 22:00:52 +02:00
}
});
}
function processFile(inputFile, outputFile, filter) {
const fileDir = path.dirname(inputFile);
const data = fs.readFileSync(inputFile, "utf8");
if (filter(inputFile, false, data)) {
console.log(`${inputFile} omitted by filter`);
return;
}
console.log(`${inputFile} -> ${outputFile}`);
const replacedData = replaceFileContents(data, fileDir);
2023-06-03 22:00:52 +02:00
fs.writeFileSync(outputFile, replacedData);
}
function generateSite(inputDir, outputDir, deleteOutput, filter) {
2023-06-03 22:00:52 +02:00
if (deleteOutput) {
deleteDirectory(outputDir);
}
if (!fs.existsSync(outputDir)) fs.mkdirSync(outputDir);
const files = fs.readdirSync(inputDir);
files.forEach((file) => {
const filePath = path.join(inputDir, file);
const isDir = fs.statSync(filePath).isDirectory();
2023-06-03 22:00:52 +02:00
if (file.startsWith("_")) return;
if (isDir) {
if (filter(filePath, true, null)) return;
2023-06-03 22:00:52 +02:00
if (path.resolve(filePath) == path.resolve(outputDir)) return;
const subOutputDir = path.join(outputDir, file);
fs.mkdirSync(subOutputDir, { recursive: true });
generateSite(filePath, subOutputDir, deleteOutput, filter);
2023-06-03 22:00:52 +02:00
} else {
const inputFile = filePath;
const outputFile = path.join(outputDir, file);
processFile(inputFile, outputFile, filter);
2023-06-03 22:00:52 +02:00
}
});
}
async function bundleHTML(inputDir, outputDir, deleteDir = true, watch = false, filter) {
generateSite(inputDir, outputDir, deleteDir, filter);
2023-06-08 13:48:08 +02:00
if (!watch) return;
2023-06-03 22:00:52 +02:00
2023-06-08 13:48:08 +02:00
const watcher = chokidar.watch(inputDir, { ignored: /(^|[\/\\])\../ });
let initialScan = true;
watcher.on("ready", () => (initialScan = false));
watcher.on("all", (event, filePath) => {
if (initialScan) return;
if (path.resolve(filePath).startsWith(path.resolve(outputDir))) return;
console.log(`File ${filePath} has been ${event}`);
generateSite(inputDir, outputDir, false, filter);
2023-06-08 13:48:08 +02:00
});
console.log(`Watching directory for changes: ${inputDir}`);
}
async function bundleCSS(inputFile, outputFile, watch = false) {
const execAsync = promisify(exec);
if (!watch) {
2023-06-08 16:51:17 +02:00
await execAsync(`npx tailwindcss -i ${inputFile} -o ${outputFile} --minify`);
console.log("Generated CSS");
} else {
execAsync(`npx tailwindcss -i ${inputFile} -o ${outputFile} --watch`);
}
}
async function bundleJS(inputDir, outputDir, watch) {
2023-06-08 13:48:08 +02:00
let buildContext = await esbuild.context({
entryPoints: {
2023-06-09 02:03:57 +02:00
carts: `${inputDir}/carts.js`,
2023-06-13 01:04:12 +02:00
cart: `${inputDir}/cart.js`,
2023-06-12 10:32:42 +02:00
changes: `${inputDir}/changes.js`,
index: `${inputDir}/index.js`,
2023-06-08 13:48:08 +02:00
},
bundle: true,
sourcemap: true,
outdir: outputDir,
logLevel: "debug",
2023-06-08 17:09:59 +02:00
minify: !watch,
2023-06-08 13:48:08 +02:00
});
if (!watch) {
2023-06-08 13:48:08 +02:00
await buildContext.rebuild();
console.log("Generated JS");
2023-06-08 13:48:08 +02:00
} else {
buildContext.watch();
}
}
async function bundle(inputDir, outputDir, watch) {
const promises = [];
promises.push(bundleCSS(path.join(inputDir, "tailwind.css"), path.join(outputDir, "style.css"), watch));
promises.push(bundleJS(inputDir, outputDir, watch));
promises.push(
bundleHTML(inputDir, outputDir, false, watch, (filePath, isDir, data) => {
if (isDir) return false;
if (filePath.endsWith("tailwind.css")) return true;
if (data.includes(`require("`)) return true;
return false;
})
);
if (!watch) await Promise.all(promises);
2023-06-08 13:48:08 +02:00
}
exports.deleteDirectory = deleteDirectory;
exports.bundle = bundle;