From 92a82d1ae378c590bd0b99b67a2cc8bea579b62c Mon Sep 17 00:00:00 2001 From: lingbopro Date: Sat, 4 Jan 2025 21:32:42 +0800 Subject: [PATCH] build: faster & smoother build script With one less rollup to run, and tasks can be run asynchronously, builds are now faster :) We don't need rollup-plugin-string now :D // Enable source map in tsconfig too --- dev/scripts/lib/build.mjs | 137 ++++++++++++++++++++------------------ package.json | 1 - pnpm-lock.yaml | 22 ------ tsconfig.json | 4 +- 4 files changed, 74 insertions(+), 90 deletions(-) diff --git a/dev/scripts/lib/build.mjs b/dev/scripts/lib/build.mjs index d6e1aae..bd9320d 100644 --- a/dev/scripts/lib/build.mjs +++ b/dev/scripts/lib/build.mjs @@ -3,7 +3,6 @@ import fs from 'fs'; import { fileURLToPath } from 'node:url'; import { rollup } from 'rollup'; import { babel } from '@rollup/plugin-babel'; -import { string } from 'rollup-plugin-string'; import terser from '@rollup/plugin-terser'; import path from 'path'; import { debug, log, logError, logSuccess } from './utils.mjs'; @@ -18,11 +17,7 @@ const basicOutputConfig = { sourcemap: true, }; -const basicPlugins = [ - string({ - include: ['**/*.css', '**/*.html'], - }), -]; +const basicPlugins = []; const minifiedBundlePlugins = [ babel({ babelHelpers: 'bundled', @@ -33,53 +28,74 @@ const minifiedBundlePlugins = [ export async function main(options) { log('starting bundle...'); - log('copying files...'); - await fs.promises.cp( - path.join(root, 'src'), - path.join(root, '.__compile_cache__'), - { - recursive: true, - }, - ); - log('compiling TypeScript...'); - // This may not seem like standard usage, but it can at least reduce the waiting time by 3s - const tscProcess = child_process.spawn( - 'node', - [path.join(root, 'node_modules', 'typescript', 'lib', 'tsc.js')], - { cwd: root }, - ); - await new Promise((resolve) => tscProcess.once('exit', resolve)); - logSuccess('success compiled TypeScript'); - log('generating bundles...'); - const globedFiles = await new Promise((resolve) => - fs.glob( - path.resolve(root, '.__compile_cache__', '**', '*.js'), - (err, matches) => { - if (err) { - throw err; - } - resolve(matches); - }, - ), - ); - debug({ globedFiles }); - const generateBundlesResult = await rollup({ - // input: path.resolve(__dirname, 'exports__compile_cache.js'), - input: globedFiles, - plugins: [...basicPlugins], - }); - const generateBundlesOutput = await generateBundlesResult.write({ - dir: path.resolve(root, 'dist'), - format: 'esm', - preserveModules: true, - ...basicOutputConfig, - }); - debug({ generateBundlesResult, generateBundlesOutput }); - logSuccess('success generated bundles'); - log('cleaning up temp dir...'); - await fs.promises.rm(path.join(root, '.__compile_cache__'), { - recursive: true, - }); + log('removing dist folder...'); + if (fs.existsSync(path.resolve(root, 'dist'))) { + await fs.promises.rm(path.resolve(root, 'dist'), { recursive: true }); + } + await fs.promises.mkdir(path.resolve(root, 'dist'), { recursive: true }); + log('compiling TypeScript & generating imports...'); + const compileTypescript = async () => { + if (options.includes('--no-tsc')) { + return; + } + /* + // This may not seem like standard usage, but it can at least reduce the waiting time by 3s + const tscProcess = child_process.spawn( + 'node', + [path.resolve(root, 'node_modules', 'typescript', 'lib', 'tsc.js')], + { cwd: root }, + ); + */ + // Well, the speed of standard usage (npx) is not bad... + const tscProcess = child_process.exec('npx tsc', { cwd: root }); + await new Promise((resolve) => tscProcess.once('exit', resolve)); + logSuccess('success compiled TypeScript'); + }; + const generateImports = async () => { + if (options.includes('--no-gen-imports')) { + return; + } + let globedFiles = []; + const globFiles = async (pattern) => + new Promise((resolve) => + fs.glob(path.resolve(pattern), (err, matches) => { + if (err) throw err; + globedFiles.push(...matches); + resolve(matches); + }), + ); + // Glob all HTML & CSS files + await Promise.all([ + globFiles(path.resolve(root, 'src', '**', '*.html')), + globFiles(path.resolve(root, 'src', '**', '*.css')), + ]); + debug({ globedFiles }); + const writeTasks = []; + for (const file of globedFiles) { + const filePath = path.resolve(file); + // Replace src -> dist + const distFilePath = + path.resolve( + path.resolve(root, 'dist'), + path.relative(path.resolve(root, 'src'), file), + ) + '.js'; + const fileContent = await fs.promises.readFile(filePath, 'utf-8'); + const distFileContent = `export default ${JSON.stringify(fileContent)};`; + // Ensure the directory exists before writing the file + if (!fs.existsSync(path.dirname(distFilePath))) { + await fs.promises.mkdir(path.dirname(distFilePath), { + recursive: true, + }); + } + // Write the file + writeTasks.push( + fs.promises.writeFile(distFilePath, distFileContent, 'utf-8'), + ); + } + await Promise.all(writeTasks); + logSuccess('success generated imports'); + }; + await Promise.all([compileTypescript(), generateImports()]); if (!options.includes('--no-min-bundle')) { log('generating minified bundle...'); const generateMinifiedBundleResult = await rollup({ @@ -109,23 +125,14 @@ export async function execute(options) { logSuccess(`done in ${(new Date() - startTime) / 1000}s`); } -export async function cleanup(options) { - if ( - fs.existsSync(path.join(root, '.__compile_cache__')) && - !options.includes('--no-finally-clean') - ) { - log('cleaning up temp dir...'); - await fs.promises.rm(path.join(root, '.__compile_cache__'), { - recursive: true, - }); - } -} +export async function cleanup(options) {} export const docs = ` Usage: build [options] Options: - --no-finally-clean Do not clean up temp directory after build + --no-gen-imports Do not generate imports (for HTML/CSS) + --no-tsc Do not compile TypeScript --no-min-bundle Do not generate minified bundle Examples: diff --git a/package.json b/package.json index f1b2ac6..5e0cdc0 100644 --- a/package.json +++ b/package.json @@ -43,7 +43,6 @@ "husky": "^9.1.7", "prettier": "^3.4.2", "rollup": "^4.27.4", - "rollup-plugin-string": "^3.0.0", "typescript": "^5.7.2" } } diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml index e850c8f..5d02f78 100644 --- a/pnpm-lock.yaml +++ b/pnpm-lock.yaml @@ -35,9 +35,6 @@ importers: rollup: specifier: ^4.27.4 version: 4.27.4 - rollup-plugin-string: - specifier: ^3.0.0 - version: 3.0.0 typescript: specifier: ^5.7.2 version: 5.7.2 @@ -917,9 +914,6 @@ packages: resolution: {integrity: sha512-WUj2qlxaQtO4g6Pq5c29GTcWGDyd8itL8zTlipgECz3JesAiiOKotd8JU6otB3PACgG6xkJUyVhboMS+bje/jA==} engines: {node: '>=6'} - estree-walker@0.6.1: - resolution: {integrity: sha512-SqmZANLWS0mnatqbSfRP5g8OXZC12Fgg1IwNtLsyHDzJizORW4khDfjPqJZsemPWBB2uqykUah5YpQ6epsqC/w==} - estree-walker@2.0.2: resolution: {integrity: sha512-Rfkk/Mp/DL7JVje3u18FxFujQlTNR2q6QfMSMB7AvCBx91NGj/ba3kCfza0f6dVDbw7YlRf/nDrn7pQrCCyQ/w==} @@ -1171,12 +1165,6 @@ packages: resolution: {integrity: sha512-oKWePCxqpd6FlLvGV1VU0x7bkPmmCNolxzjMf4NczoDnQcIWrAF+cPtZn5i6n+RfD2d9i0tzpKnG6Yk168yIyw==} hasBin: true - rollup-plugin-string@3.0.0: - resolution: {integrity: sha512-vqyzgn9QefAgeKi+Y4A7jETeIAU1zQmS6VotH6bzm/zmUQEnYkpIGRaOBPY41oiWYV4JyBoGAaBjYMYuv+6wVw==} - - rollup-pluginutils@2.8.2: - resolution: {integrity: sha512-EEp9NhnUkwY8aif6bxgovPHMoMoNr2FulJziTndpt5H9RdwC47GSGuII9XxpSdzVGM0GWrNPHV6ie1LTNJPaLQ==} - rollup@4.27.4: resolution: {integrity: sha512-RLKxqHEMjh/RGLsDxAEsaLO3mWgyoU6x9w6n1ikAzet4B3gI2/3yP6PWY2p9QzRTh6MfEIXB3MwsOY0Iv3vNrw==} engines: {node: '>=18.0.0', npm: '>=8.0.0'} @@ -2322,8 +2310,6 @@ snapshots: escalade@3.2.0: {} - estree-walker@0.6.1: {} - estree-walker@2.0.2: {} esutils@2.0.3: {} @@ -2518,14 +2504,6 @@ snapshots: path-parse: 1.0.7 supports-preserve-symlinks-flag: 1.0.0 - rollup-plugin-string@3.0.0: - dependencies: - rollup-pluginutils: 2.8.2 - - rollup-pluginutils@2.8.2: - dependencies: - estree-walker: 0.6.1 - rollup@4.27.4: dependencies: '@types/estree': 1.0.6 diff --git a/tsconfig.json b/tsconfig.json index 0852dbf..9fabcc0 100644 --- a/tsconfig.json +++ b/tsconfig.json @@ -10,11 +10,11 @@ "strict": true, "allowJs": true, "checkJs": true, - "outDir": "./.__compile_cache__", + "outDir": "./dist", "rootDir": "./src", "noImplicitAny": false, "declaration": true, - "sourceMap": false, + "sourceMap": true, "moduleResolution": "Bundler", "esModuleInterop": true, "forceConsistentCasingInFileNames": true,