You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
178 lines
4.1 KiB
178 lines
4.1 KiB
4 years ago
|
"use strict";
|
||
|
|
||
|
const {
|
||
|
minify: terserMinify
|
||
|
} = require('terser');
|
||
|
|
||
|
const buildTerserOptions = ({
|
||
|
ecma,
|
||
|
warnings,
|
||
|
parse = {},
|
||
|
compress = {},
|
||
|
mangle,
|
||
|
module,
|
||
|
output,
|
||
|
toplevel,
|
||
|
nameCache,
|
||
|
ie8,
|
||
|
|
||
|
/* eslint-disable camelcase */
|
||
|
keep_classnames,
|
||
|
keep_fnames,
|
||
|
|
||
|
/* eslint-enable camelcase */
|
||
|
safari10
|
||
|
} = {}) => ({
|
||
|
ecma,
|
||
|
warnings,
|
||
|
parse: { ...parse
|
||
|
},
|
||
|
compress: typeof compress === 'boolean' ? compress : { ...compress
|
||
|
},
|
||
|
// eslint-disable-next-line no-nested-ternary
|
||
|
mangle: mangle == null ? true : typeof mangle === 'boolean' ? mangle : { ...mangle
|
||
|
},
|
||
|
output: {
|
||
|
beautify: false,
|
||
|
...output
|
||
|
},
|
||
|
module,
|
||
|
// Ignoring sourceMap from options
|
||
|
sourceMap: null,
|
||
|
toplevel,
|
||
|
nameCache,
|
||
|
ie8,
|
||
|
keep_classnames,
|
||
|
keep_fnames,
|
||
|
safari10
|
||
|
});
|
||
|
|
||
|
function isObject(value) {
|
||
|
const type = typeof value;
|
||
|
return value != null && (type === 'object' || type === 'function');
|
||
|
}
|
||
|
|
||
|
const buildComments = (options, terserOptions, extractedComments) => {
|
||
|
const condition = {};
|
||
|
const commentsOpts = terserOptions.output.comments;
|
||
|
const {
|
||
|
extractComments
|
||
|
} = options;
|
||
|
condition.preserve = typeof commentsOpts !== 'undefined' ? commentsOpts : false;
|
||
|
|
||
|
if (typeof extractComments === 'boolean' && extractComments) {
|
||
|
condition.extract = 'some';
|
||
|
} else if (typeof extractComments === 'string' || extractComments instanceof RegExp) {
|
||
|
condition.extract = extractComments;
|
||
|
} else if (typeof extractComments === 'function') {
|
||
|
condition.extract = extractComments;
|
||
|
} else if (isObject(extractComments)) {
|
||
|
condition.extract = typeof extractComments.condition === 'boolean' && extractComments.condition ? 'some' : typeof extractComments.condition !== 'undefined' ? extractComments.condition : 'some';
|
||
|
} else {
|
||
|
// No extract
|
||
|
// Preserve using "commentsOpts" or "some"
|
||
|
condition.preserve = typeof commentsOpts !== 'undefined' ? commentsOpts : 'some';
|
||
|
condition.extract = false;
|
||
|
} // Ensure that both conditions are functions
|
||
|
|
||
|
|
||
|
['preserve', 'extract'].forEach(key => {
|
||
|
let regexStr;
|
||
|
let regex;
|
||
|
|
||
|
switch (typeof condition[key]) {
|
||
|
case 'boolean':
|
||
|
condition[key] = condition[key] ? () => true : () => false;
|
||
|
break;
|
||
|
|
||
|
case 'function':
|
||
|
break;
|
||
|
|
||
|
case 'string':
|
||
|
if (condition[key] === 'all') {
|
||
|
condition[key] = () => true;
|
||
|
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
if (condition[key] === 'some') {
|
||
|
condition[key] = (astNode, comment) => {
|
||
|
return (comment.type === 'comment2' || comment.type === 'comment1') && /@preserve|@lic|@cc_on|^\**!/i.test(comment.value);
|
||
|
};
|
||
|
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
regexStr = condition[key];
|
||
|
|
||
|
condition[key] = (astNode, comment) => {
|
||
|
return new RegExp(regexStr).test(comment.value);
|
||
|
};
|
||
|
|
||
|
break;
|
||
|
|
||
|
default:
|
||
|
regex = condition[key];
|
||
|
|
||
|
condition[key] = (astNode, comment) => regex.test(comment.value);
|
||
|
|
||
|
}
|
||
|
}); // Redefine the comments function to extract and preserve
|
||
|
// comments according to the two conditions
|
||
|
|
||
|
return (astNode, comment) => {
|
||
|
if (condition.extract(astNode, comment)) {
|
||
|
const commentText = comment.type === 'comment2' ? `/*${comment.value}*/` : `//${comment.value}`; // Don't include duplicate comments
|
||
|
|
||
|
if (!extractedComments.includes(commentText)) {
|
||
|
extractedComments.push(commentText);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return condition.preserve(astNode, comment);
|
||
|
};
|
||
|
};
|
||
|
|
||
|
const minify = options => {
|
||
|
const {
|
||
|
file,
|
||
|
input,
|
||
|
inputSourceMap,
|
||
|
minify: minifyFn
|
||
|
} = options;
|
||
|
|
||
|
if (minifyFn) {
|
||
|
return minifyFn({
|
||
|
[file]: input
|
||
|
}, inputSourceMap);
|
||
|
} // Copy terser options
|
||
|
|
||
|
|
||
|
const terserOptions = buildTerserOptions(options.terserOptions); // Let terser generate a SourceMap
|
||
|
|
||
|
if (inputSourceMap) {
|
||
|
terserOptions.sourceMap = {
|
||
|
asObject: true
|
||
|
};
|
||
|
}
|
||
|
|
||
|
const extractedComments = [];
|
||
|
terserOptions.output.comments = buildComments(options, terserOptions, extractedComments);
|
||
|
const {
|
||
|
error,
|
||
|
map,
|
||
|
code,
|
||
|
warnings
|
||
|
} = terserMinify({
|
||
|
[file]: input
|
||
|
}, terserOptions);
|
||
|
return {
|
||
|
error,
|
||
|
map,
|
||
|
code,
|
||
|
warnings,
|
||
|
extractedComments
|
||
|
};
|
||
|
};
|
||
|
|
||
|
module.exports = minify;
|