363 lines
		
	
	
		
			9.9 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			363 lines
		
	
	
		
			9.9 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| "use strict";
 | |
| 
 | |
| /** @typedef {import("./index.js").Input} Input */
 | |
| 
 | |
| /** @typedef {import("source-map").RawSourceMap} RawSourceMap */
 | |
| 
 | |
| /** @typedef {import("source-map").SourceMapGenerator} SourceMapGenerator */
 | |
| 
 | |
| /** @typedef {import("./index.js").MinimizedResult} MinimizedResult */
 | |
| 
 | |
| /** @typedef {import("./index.js").CustomOptions} CustomOptions */
 | |
| 
 | |
| /** @typedef {import("postcss").ProcessOptions} ProcessOptions */
 | |
| 
 | |
| /** @typedef {import("postcss").Postcss} Postcss */
 | |
| const notSettled = Symbol(`not-settled`);
 | |
| /**
 | |
|  * @template T
 | |
|  * @typedef {() => Promise<T>} Task
 | |
|  */
 | |
| 
 | |
| /**
 | |
|  * Run tasks with limited concurency.
 | |
|  * @template T
 | |
|  * @param {number} limit - Limit of tasks that run at once.
 | |
|  * @param {Task<T>[]} tasks - List of tasks to run.
 | |
|  * @returns {Promise<T[]>} A promise that fulfills to an array of the results
 | |
|  */
 | |
| 
 | |
| function throttleAll(limit, tasks) {
 | |
|   if (!Number.isInteger(limit) || limit < 1) {
 | |
|     throw new TypeError(`Expected \`limit\` to be a finite number > 0, got \`${limit}\` (${typeof limit})`);
 | |
|   }
 | |
| 
 | |
|   if (!Array.isArray(tasks) || !tasks.every(task => typeof task === `function`)) {
 | |
|     throw new TypeError(`Expected \`tasks\` to be a list of functions returning a promise`);
 | |
|   }
 | |
| 
 | |
|   return new Promise((resolve, reject) => {
 | |
|     const result = Array(tasks.length).fill(notSettled);
 | |
|     const entries = tasks.entries();
 | |
| 
 | |
|     const next = () => {
 | |
|       const {
 | |
|         done,
 | |
|         value
 | |
|       } = entries.next();
 | |
| 
 | |
|       if (done) {
 | |
|         const isLast = !result.includes(notSettled);
 | |
|         if (isLast) resolve(result);
 | |
|         return;
 | |
|       }
 | |
| 
 | |
|       const [index, task] = value;
 | |
|       /**
 | |
|        * @param {T} x
 | |
|        */
 | |
| 
 | |
|       const onFulfilled = x => {
 | |
|         result[index] = x;
 | |
|         next();
 | |
|       };
 | |
| 
 | |
|       task().then(onFulfilled, reject);
 | |
|     };
 | |
| 
 | |
|     Array(limit).fill(0).forEach(next);
 | |
|   });
 | |
| }
 | |
| /* istanbul ignore next */
 | |
| 
 | |
| /**
 | |
|  * @param {Input} input
 | |
|  * @param {RawSourceMap | undefined} sourceMap
 | |
|  * @param {CustomOptions} minimizerOptions
 | |
|  * @return {Promise<MinimizedResult>}
 | |
|  */
 | |
| 
 | |
| 
 | |
| async function cssnanoMinify(input, sourceMap, minimizerOptions = {
 | |
|   preset: "default"
 | |
| }) {
 | |
|   /**
 | |
|    * @template T
 | |
|    * @param {string} module
 | |
|    * @returns {Promise<T>}
 | |
|    */
 | |
|   const load = async module => {
 | |
|     let exports;
 | |
| 
 | |
|     try {
 | |
|       // eslint-disable-next-line import/no-dynamic-require, global-require
 | |
|       exports = require(module);
 | |
|       return exports;
 | |
|     } catch (requireError) {
 | |
|       let importESM;
 | |
| 
 | |
|       try {
 | |
|         // eslint-disable-next-line no-new-func
 | |
|         importESM = new Function("id", "return import(id);");
 | |
|       } catch (e) {
 | |
|         importESM = null;
 | |
|       }
 | |
| 
 | |
|       if (
 | |
|       /** @type {Error & {code: string}} */
 | |
|       requireError.code === "ERR_REQUIRE_ESM" && importESM) {
 | |
|         exports = await importESM(module);
 | |
|         return exports.default;
 | |
|       }
 | |
| 
 | |
|       throw requireError;
 | |
|     }
 | |
|   };
 | |
| 
 | |
|   const [[name, code]] = Object.entries(input);
 | |
|   /** @type {ProcessOptions} */
 | |
| 
 | |
|   const postcssOptions = {
 | |
|     from: name,
 | |
|     ...minimizerOptions.processorOptions
 | |
|   };
 | |
| 
 | |
|   if (typeof postcssOptions.parser === "string") {
 | |
|     try {
 | |
|       postcssOptions.parser = await load(postcssOptions.parser);
 | |
|     } catch (error) {
 | |
|       throw new Error(`Loading PostCSS "${postcssOptions.parser}" parser failed: ${
 | |
|       /** @type {Error} */
 | |
|       error.message}\n\n(@${name})`);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (typeof postcssOptions.stringifier === "string") {
 | |
|     try {
 | |
|       postcssOptions.stringifier = await load(postcssOptions.stringifier);
 | |
|     } catch (error) {
 | |
|       throw new Error(`Loading PostCSS "${postcssOptions.stringifier}" stringifier failed: ${
 | |
|       /** @type {Error} */
 | |
|       error.message}\n\n(@${name})`);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (typeof postcssOptions.syntax === "string") {
 | |
|     try {
 | |
|       postcssOptions.syntax = await load(postcssOptions.syntax);
 | |
|     } catch (error) {
 | |
|       throw new Error(`Loading PostCSS "${postcssOptions.syntax}" syntax failed: ${
 | |
|       /** @type {Error} */
 | |
|       error.message}\n\n(@${name})`);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (sourceMap) {
 | |
|     postcssOptions.map = {
 | |
|       annotation: false
 | |
|     };
 | |
|   }
 | |
|   /** @type {Postcss} */
 | |
|   // eslint-disable-next-line global-require
 | |
| 
 | |
| 
 | |
|   const postcss = require("postcss").default; // @ts-ignore
 | |
|   // eslint-disable-next-line global-require
 | |
| 
 | |
| 
 | |
|   const cssnano = require("cssnano"); // @ts-ignore
 | |
|   // Types are broken
 | |
| 
 | |
| 
 | |
|   const result = await postcss([cssnano(minimizerOptions)]).process(code, postcssOptions);
 | |
|   return {
 | |
|     code: result.css,
 | |
|     map: result.map ? result.map.toJSON() : // eslint-disable-next-line no-undefined
 | |
|     undefined,
 | |
|     warnings: result.warnings().map(String)
 | |
|   };
 | |
| }
 | |
| /* istanbul ignore next */
 | |
| 
 | |
| /**
 | |
|  * @param {Input} input
 | |
|  * @param {RawSourceMap | undefined} sourceMap
 | |
|  * @param {CustomOptions} minimizerOptions
 | |
|  * @return {Promise<MinimizedResult>}
 | |
|  */
 | |
| 
 | |
| 
 | |
| async function cssoMinify(input, sourceMap, minimizerOptions) {
 | |
|   // eslint-disable-next-line global-require,import/no-extraneous-dependencies
 | |
|   const csso = require("csso");
 | |
| 
 | |
|   const [[filename, code]] = Object.entries(input);
 | |
|   const result = csso.minify(code, {
 | |
|     filename,
 | |
|     sourceMap: Boolean(sourceMap),
 | |
|     ...minimizerOptions
 | |
|   });
 | |
|   return {
 | |
|     code: result.css,
 | |
|     map: result.map ?
 | |
|     /** @type {SourceMapGenerator & { toJSON(): RawSourceMap }} */
 | |
|     result.map.toJSON() : // eslint-disable-next-line no-undefined
 | |
|     undefined
 | |
|   };
 | |
| }
 | |
| /* istanbul ignore next */
 | |
| 
 | |
| /**
 | |
|  * @param {Input} input
 | |
|  * @param {RawSourceMap | undefined} sourceMap
 | |
|  * @param {CustomOptions} minimizerOptions
 | |
|  * @return {Promise<MinimizedResult>}
 | |
|  */
 | |
| 
 | |
| 
 | |
| async function cleanCssMinify(input, sourceMap, minimizerOptions) {
 | |
|   // eslint-disable-next-line global-require,import/no-extraneous-dependencies
 | |
|   const CleanCSS = require("clean-css");
 | |
| 
 | |
|   const [[name, code]] = Object.entries(input);
 | |
|   const result = await new CleanCSS({
 | |
|     sourceMap: Boolean(sourceMap),
 | |
|     ...minimizerOptions,
 | |
|     returnPromise: true
 | |
|   }).minify({
 | |
|     [name]: {
 | |
|       styles: code
 | |
|     }
 | |
|   });
 | |
|   const generatedSourceMap = result.sourceMap &&
 | |
|   /** @type {SourceMapGenerator & { toJSON(): RawSourceMap }} */
 | |
|   result.sourceMap.toJSON(); // workaround for source maps on windows
 | |
| 
 | |
|   if (generatedSourceMap) {
 | |
|     // eslint-disable-next-line global-require
 | |
|     const isWindowsPathSep = require("path").sep === "\\";
 | |
|     generatedSourceMap.sources = generatedSourceMap.sources.map(
 | |
|     /**
 | |
|      * @param {string} item
 | |
|      * @returns {string}
 | |
|      */
 | |
|     item => isWindowsPathSep ? item.replace(/\\/g, "/") : item);
 | |
|   }
 | |
| 
 | |
|   return {
 | |
|     code: result.styles,
 | |
|     map: generatedSourceMap,
 | |
|     warnings: result.warnings
 | |
|   };
 | |
| }
 | |
| /* istanbul ignore next */
 | |
| 
 | |
| /**
 | |
|  * @param {Input} input
 | |
|  * @param {RawSourceMap | undefined} sourceMap
 | |
|  * @param {CustomOptions} minimizerOptions
 | |
|  * @return {Promise<MinimizedResult>}
 | |
|  */
 | |
| 
 | |
| 
 | |
| async function esbuildMinify(input, sourceMap, minimizerOptions) {
 | |
|   /**
 | |
|    * @param {import("esbuild").TransformOptions} [esbuildOptions={}]
 | |
|    * @returns {import("esbuild").TransformOptions}
 | |
|    */
 | |
|   const buildEsbuildOptions = (esbuildOptions = {}) => {
 | |
|     // Need deep copy objects to avoid https://github.com/terser/terser/issues/366
 | |
|     return {
 | |
|       loader: "css",
 | |
|       minify: true,
 | |
|       legalComments: "inline",
 | |
|       ...esbuildOptions,
 | |
|       sourcemap: false
 | |
|     };
 | |
|   }; // eslint-disable-next-line import/no-extraneous-dependencies, global-require
 | |
| 
 | |
| 
 | |
|   const esbuild = require("esbuild"); // Copy `esbuild` options
 | |
| 
 | |
| 
 | |
|   const esbuildOptions = buildEsbuildOptions(minimizerOptions); // Let `esbuild` generate a SourceMap
 | |
| 
 | |
|   if (sourceMap) {
 | |
|     esbuildOptions.sourcemap = true;
 | |
|     esbuildOptions.sourcesContent = false;
 | |
|   }
 | |
| 
 | |
|   const [[filename, code]] = Object.entries(input);
 | |
|   esbuildOptions.sourcefile = filename;
 | |
|   const result = await esbuild.transform(code, esbuildOptions);
 | |
|   return {
 | |
|     code: result.code,
 | |
|     // eslint-disable-next-line no-undefined
 | |
|     map: result.map ? JSON.parse(result.map) : undefined,
 | |
|     warnings: result.warnings.length > 0 ? result.warnings.map(item => {
 | |
|       return {
 | |
|         source: item.location && item.location.file,
 | |
|         // eslint-disable-next-line no-undefined
 | |
|         line: item.location && item.location.line ? item.location.line : undefined,
 | |
|         // eslint-disable-next-line no-undefined
 | |
|         column: item.location && item.location.column ? item.location.column : undefined,
 | |
|         plugin: item.pluginName,
 | |
|         message: `${item.text}${item.detail ? `\nDetails:\n${item.detail}` : ""}${item.notes.length > 0 ? `\n\nNotes:\n${item.notes.map(note => `${note.location ? `[${note.location.file}:${note.location.line}:${note.location.column}] ` : ""}${note.text}${note.location ? `\nSuggestion: ${note.location.suggestion}` : ""}${note.location ? `\nLine text:\n${note.location.lineText}\n` : ""}`).join("\n")}` : ""}`
 | |
|       };
 | |
|     }) : []
 | |
|   };
 | |
| }
 | |
| /* istanbul ignore next */
 | |
| 
 | |
| /**
 | |
|  * @param {Input} input
 | |
|  * @param {RawSourceMap | undefined} sourceMap
 | |
|  * @param {CustomOptions} minimizerOptions
 | |
|  * @return {Promise<MinimizedResult>}
 | |
|  */
 | |
| 
 | |
| 
 | |
| async function parcelCssMinify(input, sourceMap, minimizerOptions) {
 | |
|   const [[filename, code]] = Object.entries(input);
 | |
|   /**
 | |
|    * @param {Partial<import("@parcel/css").TransformOptions>} [parcelCssOptions={}]
 | |
|    * @returns {import("@parcel/css").TransformOptions}
 | |
|    */
 | |
| 
 | |
|   const buildParcelCssOptions = (parcelCssOptions = {}) => {
 | |
|     // Need deep copy objects to avoid https://github.com/terser/terser/issues/366
 | |
|     return {
 | |
|       minify: true,
 | |
|       ...parcelCssOptions,
 | |
|       sourceMap: false,
 | |
|       filename,
 | |
|       code: Buffer.from(code)
 | |
|     };
 | |
|   }; // eslint-disable-next-line import/no-extraneous-dependencies, global-require
 | |
| 
 | |
| 
 | |
|   const parcelCss = require("@parcel/css"); // Copy `esbuild` options
 | |
| 
 | |
| 
 | |
|   const parcelCssOptions = buildParcelCssOptions(minimizerOptions); // Let `esbuild` generate a SourceMap
 | |
| 
 | |
|   if (sourceMap) {
 | |
|     parcelCssOptions.sourceMap = true;
 | |
|   }
 | |
| 
 | |
|   const result = await parcelCss.transform(parcelCssOptions);
 | |
|   return {
 | |
|     code: result.code.toString(),
 | |
|     // eslint-disable-next-line no-undefined
 | |
|     map: result.map ? JSON.parse(result.map.toString()) : undefined
 | |
|   };
 | |
| }
 | |
| 
 | |
| module.exports = {
 | |
|   throttleAll,
 | |
|   cssnanoMinify,
 | |
|   cssoMinify,
 | |
|   cleanCssMinify,
 | |
|   esbuildMinify,
 | |
|   parcelCssMinify
 | |
| }; |