"use strict"; /* eslint-env browser, es6, node */ import { defaults, map_from_object, map_to_object, HOP, } from "./utils/index.js"; import { AST_Toplevel, AST_Node, walk, AST_Scope } from "./ast.js"; import { parse } from "./parse.js"; import { OutputStream } from "./output.js"; import { Compressor } from "./compress/index.js"; import { base54 } from "./scope.js"; import { SourceMap } from "./sourcemap.js"; import { mangle_properties, mangle_private_properties, reserve_quoted_keys, find_annotated_props, } from "./propmangle.js"; // to/from base64 functions // Prefer built-in Buffer, if available, then use hack // https://developer.mozilla.org/en-US/docs/Glossary/Base64#The_Unicode_Problem var to_ascii = typeof Buffer !== "undefined" ? (b64) => Buffer.from(b64, "base64").toString() : (b64) => decodeURIComponent(escape(atob(b64))); var to_base64 = typeof Buffer !== "undefined" ? (str) => Buffer.from(str).toString("base64") : (str) => btoa(unescape(encodeURIComponent(str))); function read_source_map(code) { var match = /(?:^|[^.])\/\/# sourceMappingURL=data:application\/json(;[\w=-]*)?;base64,([+/0-9A-Za-z]*=*)\s*$/.exec(code); if (!match) { console.warn("inline source map not found"); return null; } return to_ascii(match[2]); } function set_shorthand(name, options, keys) { if (options[name]) { keys.forEach(function(key) { if (options[key]) { if (typeof options[key] != "object") options[key] = {}; if (!(name in options[key])) options[key][name] = options[name]; } }); } } function init_cache(cache) { if (!cache) return; if (!("props" in cache)) { cache.props = new Map(); } else if (!(cache.props instanceof Map)) { cache.props = map_from_object(cache.props); } } function cache_to_json(cache) { return { props: map_to_object(cache.props) }; } function log_input(files, options, fs, debug_folder) { if (!(fs && fs.writeFileSync && fs.mkdirSync)) { return; } try { fs.mkdirSync(debug_folder); } catch (e) { if (e.code !== "EEXIST") throw e; } const log_path = `${debug_folder}/terser-debug-${(Math.random() * 9999999) | 0}.log`; options = options || {}; const options_str = JSON.stringify(options, (_key, thing) => { if (typeof thing === "function") return "[Function " + thing.toString() + "]"; if (thing instanceof RegExp) return "[RegExp " + thing.toString() + "]"; return thing; }, 4); const files_str = (file) => { if (typeof file === "object" && options.parse && options.parse.spidermonkey) { return JSON.stringify(file, null, 2); } else if (typeof file === "object") { return Object.keys(file) .map((key) => key + ": " + files_str(file[key])) .join("\n\n"); } else if (typeof file === "string") { return "```\n" + file + "\n```"; } else { return file; // What do? } }; fs.writeFileSync(log_path, "Options: \n" + options_str + "\n\nInput files:\n\n" + files_str(files) + "\n"); } function* minify_sync_or_async(files, options, _fs_module) { if ( _fs_module && typeof process === "object" && process.env && typeof process.env.TERSER_DEBUG_DIR === "string" ) { log_input(files, options, _fs_module, process.env.TERSER_DEBUG_DIR); } options = defaults(options, { compress: {}, ecma: undefined, enclose: false, ie8: false, keep_classnames: undefined, keep_fnames: false, mangle: {}, module: false, nameCache: null, output: null, format: null, parse: {}, rename: undefined, safari10: false, sourceMap: false, spidermonkey: false, timings: false, toplevel: false, warnings: false, wrap: false, }, true); var timings = options.timings && { start: Date.now() }; if (options.keep_classnames === undefined) { options.keep_classnames = options.keep_fnames; } if (options.rename === undefined) { options.rename = options.compress && options.mangle; } if (options.output && options.format) { throw new Error("Please only specify either output or format option, preferrably format."); } options.format = options.format || options.output || {}; set_shorthand("ecma", options, [ "parse", "compress", "format" ]); set_shorthand("ie8", options, [ "compress", "mangle", "format" ]); set_shorthand("keep_classnames", options, [ "compress", "mangle" ]); set_shorthand("keep_fnames", options, [ "compress", "mangle" ]); set_shorthand("module", options, [ "parse", "compress", "mangle" ]); set_shorthand("safari10", options, [ "mangle", "format" ]); set_shorthand("toplevel", options, [ "compress", "mangle" ]); set_shorthand("warnings", options, [ "compress" ]); // legacy var quoted_props; if (options.mangle) { options.mangle = defaults(options.mangle, { cache: options.nameCache && (options.nameCache.vars || {}), eval: false, ie8: false, keep_classnames: false, keep_fnames: false, module: false, nth_identifier: base54, properties: false, reserved: [], safari10: false, toplevel: false, }, true); if (options.mangle.properties) { if (typeof options.mangle.properties != "object") { options.mangle.properties = {}; } if (options.mangle.properties.keep_quoted) { quoted_props = options.mangle.properties.reserved; if (!Array.isArray(quoted_props)) quoted_props = []; options.mangle.properties.reserved = quoted_props; } if (options.nameCache && !("cache" in options.mangle.properties)) { options.mangle.properties.cache = options.nameCache.props || {}; } } init_cache(options.mangle.cache); init_cache(options.mangle.properties.cache); } if (options.sourceMap) { options.sourceMap = defaults(options.sourceMap, { asObject: false, content: null, filename: null, includeSources: false, root: null, url: null, }, true); } // -- Parse phase -- if (timings) timings.parse = Date.now(); var toplevel; if (files instanceof AST_Toplevel) { toplevel = files; } else { if (typeof files == "string" || (options.parse.spidermonkey && !Array.isArray(files))) { files = [ files ]; } options.parse = options.parse || {}; options.parse.toplevel = null; if (options.parse.spidermonkey) { options.parse.toplevel = AST_Node.from_mozilla_ast(Object.keys(files).reduce(function(toplevel, name) { if (!toplevel) return files[name]; toplevel.body = toplevel.body.concat(files[name].body); return toplevel; }, null)); } else { delete options.parse.spidermonkey; for (var name in files) if (HOP(files, name)) { options.parse.filename = name; options.parse.toplevel = parse(files[name], options.parse); if (options.sourceMap && options.sourceMap.content == "inline") { if (Object.keys(files).length > 1) throw new Error("inline source map only works with singular input"); options.sourceMap.content = read_source_map(files[name]); } } } if (options.parse.toplevel === null) { throw new Error("no source file given"); } toplevel = options.parse.toplevel; } if (quoted_props && options.mangle.properties.keep_quoted !== "strict") { reserve_quoted_keys(toplevel, quoted_props); } var annotated_props; if (options.mangle && options.mangle.properties) { annotated_props = find_annotated_props(toplevel); } if (options.wrap) { toplevel = toplevel.wrap_commonjs(options.wrap); } if (options.enclose) { toplevel = toplevel.wrap_enclose(options.enclose); } if (timings) timings.rename = Date.now(); // disable rename on harmony due to expand_names bug in for-of loops // https://github.com/mishoo/UglifyJS2/issues/2794 if (0 && options.rename) { toplevel.figure_out_scope(options.mangle); toplevel.expand_names(options.mangle); } // -- Compress phase -- if (timings) timings.compress = Date.now(); if (options.compress) { toplevel = new Compressor(options.compress, { mangle_options: options.mangle }).compress(toplevel); } // -- Mangle phase -- if (timings) timings.scope = Date.now(); if (options.mangle) toplevel.figure_out_scope(options.mangle); if (timings) timings.mangle = Date.now(); if (options.mangle) { toplevel.compute_char_frequency(options.mangle); toplevel.mangle_names(options.mangle); toplevel = mangle_private_properties(toplevel, options.mangle); } if (timings) timings.properties = Date.now(); if (options.mangle && options.mangle.properties) { toplevel = mangle_properties(toplevel, options.mangle.properties, annotated_props); } // Format phase if (timings) timings.format = Date.now(); var result = {}; if (options.format.ast) { result.ast = toplevel; } if (options.format.spidermonkey) { result.ast = toplevel.to_mozilla_ast(); } let format_options; if (!HOP(options.format, "code") || options.format.code) { // Make a shallow copy so that we can modify without mutating the user's input. format_options = {...options.format}; if (!format_options.ast) { // Destroy stuff to save RAM. (unless the deprecated `ast` option is on) format_options._destroy_ast = true; walk(toplevel, node => { if (node instanceof AST_Scope) { node.variables = undefined; node.enclosed = undefined; node.parent_scope = undefined; } if (node.block_scope) { node.block_scope.variables = undefined; node.block_scope.enclosed = undefined; node.block_scope.parent_scope = undefined; } }); } if (options.sourceMap) { if (options.sourceMap.includeSources && files instanceof AST_Toplevel) { throw new Error("original source content unavailable"); } format_options.source_map = yield* SourceMap({ file: options.sourceMap.filename, orig: options.sourceMap.content, root: options.sourceMap.root, files: options.sourceMap.includeSources ? files : null, }); } delete format_options.ast; delete format_options.code; delete format_options.spidermonkey; var stream = OutputStream(format_options); toplevel.print(stream); result.code = stream.get(); if (options.sourceMap) { Object.defineProperty(result, "map", { configurable: true, enumerable: true, get() { const map = format_options.source_map.getEncoded(); return (result.map = options.sourceMap.asObject ? map : JSON.stringify(map)); }, set(value) { Object.defineProperty(result, "map", { value, writable: true, }); } }); result.decoded_map = format_options.source_map.getDecoded(); if (options.sourceMap.url == "inline") { var sourceMap = typeof result.map === "object" ? JSON.stringify(result.map) : result.map; result.code += "\n//# sourceMappingURL=data:application/json;charset=utf-8;base64," + to_base64(sourceMap); } else if (options.sourceMap.url) { result.code += "\n//# sourceMappingURL=" + options.sourceMap.url; } } } if (options.nameCache && options.mangle) { if (options.mangle.cache) options.nameCache.vars = cache_to_json(options.mangle.cache); if (options.mangle.properties && options.mangle.properties.cache) { options.nameCache.props = cache_to_json(options.mangle.properties.cache); } } if (format_options && format_options.source_map) { format_options.source_map.destroy(); } if (timings) { timings.end = Date.now(); result.timings = { parse: 1e-3 * (timings.rename - timings.parse), rename: 1e-3 * (timings.compress - timings.rename), compress: 1e-3 * (timings.scope - timings.compress), scope: 1e-3 * (timings.mangle - timings.scope), mangle: 1e-3 * (timings.properties - timings.mangle), properties: 1e-3 * (timings.format - timings.properties), format: 1e-3 * (timings.end - timings.format), total: 1e-3 * (timings.end - timings.start) }; } return result; } async function minify(files, options, _fs_module) { const gen = minify_sync_or_async(files, options, _fs_module); let yielded; let val; do { val = gen.next(await yielded); yielded = val.value; } while (!val.done); return val.value; } function minify_sync(files, options, _fs_module) { const gen = minify_sync_or_async(files, options, _fs_module); let yielded; let val; do { if (yielded && typeof yielded.then === "function") { throw new Error("minify_sync cannot be used with the legacy source-map module"); } val = gen.next(yielded); yielded = val.value; } while (!val.done); return val.value; } export { minify, minify_sync, to_ascii, };