Buckets:
| ; | |
| /* 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, | |
| }; | |
Xet Storage Details
- Size:
- 14.5 kB
- Xet hash:
- 17fd58036a38e584050cef4a8475af9ec9f2b0048b24cb9cad25ba07d2b4246d
·
Xet efficiently stores files, intelligently splitting them into unique chunks and accelerating uploads and downloads. More info.