1 | import { minify, _default_options } from "../main.js";
|
---|
2 | import { parse } from "./parse.js";
|
---|
3 | import {
|
---|
4 | AST_Assign,
|
---|
5 | AST_Array,
|
---|
6 | AST_Constant,
|
---|
7 | AST_Node,
|
---|
8 | AST_PropAccess,
|
---|
9 | AST_RegExp,
|
---|
10 | AST_Sequence,
|
---|
11 | AST_Symbol,
|
---|
12 | AST_Token,
|
---|
13 | walk
|
---|
14 | } from "./ast.js";
|
---|
15 | import { OutputStream } from "./output.js";
|
---|
16 |
|
---|
17 | export async function run_cli({ program, packageJson, fs, path }) {
|
---|
18 | const skip_keys = new Set([ "cname", "parent_scope", "scope", "uses_eval", "uses_with" ]);
|
---|
19 | var files = {};
|
---|
20 | var options = {
|
---|
21 | compress: false,
|
---|
22 | mangle: false
|
---|
23 | };
|
---|
24 | const default_options = await _default_options();
|
---|
25 | program.version(packageJson.name + " " + packageJson.version);
|
---|
26 | program.parseArgv = program.parse;
|
---|
27 | program.parse = undefined;
|
---|
28 |
|
---|
29 | if (process.argv.includes("ast")) program.helpInformation = describe_ast;
|
---|
30 | else if (process.argv.includes("options")) program.helpInformation = function() {
|
---|
31 | var text = [];
|
---|
32 | for (var option in default_options) {
|
---|
33 | text.push("--" + (option === "sourceMap" ? "source-map" : option) + " options:");
|
---|
34 | text.push(format_object(default_options[option]));
|
---|
35 | text.push("");
|
---|
36 | }
|
---|
37 | return text.join("\n");
|
---|
38 | };
|
---|
39 |
|
---|
40 | program.option("-p, --parse <options>", "Specify parser options.", parse_js());
|
---|
41 | program.option("-c, --compress [options]", "Enable compressor/specify compressor options.", parse_js());
|
---|
42 | program.option("-m, --mangle [options]", "Mangle names/specify mangler options.", parse_js());
|
---|
43 | program.option("--mangle-props [options]", "Mangle properties/specify mangler options.", parse_js());
|
---|
44 | program.option("-f, --format [options]", "Format options.", parse_js());
|
---|
45 | program.option("-b, --beautify [options]", "Alias for --format.", parse_js());
|
---|
46 | program.option("-o, --output <file>", "Output file (default STDOUT).");
|
---|
47 | program.option("--comments [filter]", "Preserve copyright comments in the output.");
|
---|
48 | program.option("--config-file <file>", "Read minify() options from JSON file.");
|
---|
49 | program.option("-d, --define <expr>[=value]", "Global definitions.", parse_js("define"));
|
---|
50 | program.option("--ecma <version>", "Specify ECMAScript release: 5, 2015, 2016 or 2017...");
|
---|
51 | program.option("-e, --enclose [arg[,...][:value[,...]]]", "Embed output in a big function with configurable arguments and values.");
|
---|
52 | program.option("--ie8", "Support non-standard Internet Explorer 8.");
|
---|
53 | program.option("--keep-classnames", "Do not mangle/drop class names.");
|
---|
54 | program.option("--keep-fnames", "Do not mangle/drop function names. Useful for code relying on Function.prototype.name.");
|
---|
55 | program.option("--module", "Input is an ES6 module");
|
---|
56 | program.option("--name-cache <file>", "File to hold mangled name mappings.");
|
---|
57 | program.option("--rename", "Force symbol expansion.");
|
---|
58 | program.option("--no-rename", "Disable symbol expansion.");
|
---|
59 | program.option("--safari10", "Support non-standard Safari 10.");
|
---|
60 | program.option("--source-map [options]", "Enable source map/specify source map options.", parse_js());
|
---|
61 | program.option("--timings", "Display operations run time on STDERR.");
|
---|
62 | program.option("--toplevel", "Compress and/or mangle variables in toplevel scope.");
|
---|
63 | program.option("--wrap <name>", "Embed everything as a function with “exports” corresponding to “name” globally.");
|
---|
64 | program.arguments("[files...]").parseArgv(process.argv);
|
---|
65 | if (program.configFile) {
|
---|
66 | options = JSON.parse(read_file(program.configFile));
|
---|
67 | }
|
---|
68 | if (!program.output && program.sourceMap && program.sourceMap.url != "inline") {
|
---|
69 | fatal("ERROR: cannot write source map to STDOUT");
|
---|
70 | }
|
---|
71 |
|
---|
72 | [
|
---|
73 | "compress",
|
---|
74 | "enclose",
|
---|
75 | "ie8",
|
---|
76 | "mangle",
|
---|
77 | "module",
|
---|
78 | "safari10",
|
---|
79 | "sourceMap",
|
---|
80 | "toplevel",
|
---|
81 | "wrap"
|
---|
82 | ].forEach(function(name) {
|
---|
83 | if (name in program) {
|
---|
84 | options[name] = program[name];
|
---|
85 | }
|
---|
86 | });
|
---|
87 |
|
---|
88 | if ("ecma" in program) {
|
---|
89 | if (program.ecma != (program.ecma | 0)) fatal("ERROR: ecma must be an integer");
|
---|
90 | const ecma = program.ecma | 0;
|
---|
91 | if (ecma > 5 && ecma < 2015)
|
---|
92 | options.ecma = ecma + 2009;
|
---|
93 | else
|
---|
94 | options.ecma = ecma;
|
---|
95 | }
|
---|
96 | if (program.format || program.beautify) {
|
---|
97 | const chosenOption = program.format || program.beautify;
|
---|
98 | options.format = typeof chosenOption === "object" ? chosenOption : {};
|
---|
99 | }
|
---|
100 | if (program.comments) {
|
---|
101 | if (typeof options.format != "object") options.format = {};
|
---|
102 | options.format.comments = typeof program.comments == "string" ? (program.comments == "false" ? false : program.comments) : "some";
|
---|
103 | }
|
---|
104 | if (program.define) {
|
---|
105 | if (typeof options.compress != "object") options.compress = {};
|
---|
106 | if (typeof options.compress.global_defs != "object") options.compress.global_defs = {};
|
---|
107 | for (var expr in program.define) {
|
---|
108 | options.compress.global_defs[expr] = program.define[expr];
|
---|
109 | }
|
---|
110 | }
|
---|
111 | if (program.keepClassnames) {
|
---|
112 | options.keep_classnames = true;
|
---|
113 | }
|
---|
114 | if (program.keepFnames) {
|
---|
115 | options.keep_fnames = true;
|
---|
116 | }
|
---|
117 | if (program.mangleProps) {
|
---|
118 | if (program.mangleProps.domprops) {
|
---|
119 | delete program.mangleProps.domprops;
|
---|
120 | } else {
|
---|
121 | if (typeof program.mangleProps != "object") program.mangleProps = {};
|
---|
122 | if (!Array.isArray(program.mangleProps.reserved)) program.mangleProps.reserved = [];
|
---|
123 | }
|
---|
124 | if (typeof options.mangle != "object") options.mangle = {};
|
---|
125 | options.mangle.properties = program.mangleProps;
|
---|
126 | }
|
---|
127 | if (program.nameCache) {
|
---|
128 | options.nameCache = JSON.parse(read_file(program.nameCache, "{}"));
|
---|
129 | }
|
---|
130 | if (program.output == "ast") {
|
---|
131 | options.format = {
|
---|
132 | ast: true,
|
---|
133 | code: false
|
---|
134 | };
|
---|
135 | }
|
---|
136 | if (program.parse) {
|
---|
137 | if (!program.parse.acorn && !program.parse.spidermonkey) {
|
---|
138 | options.parse = program.parse;
|
---|
139 | } else if (program.sourceMap && program.sourceMap.content == "inline") {
|
---|
140 | fatal("ERROR: inline source map only works with built-in parser");
|
---|
141 | }
|
---|
142 | }
|
---|
143 | if (~program.rawArgs.indexOf("--rename")) {
|
---|
144 | options.rename = true;
|
---|
145 | } else if (!program.rename) {
|
---|
146 | options.rename = false;
|
---|
147 | }
|
---|
148 |
|
---|
149 | let convert_path = name => name;
|
---|
150 | if (typeof program.sourceMap == "object" && "base" in program.sourceMap) {
|
---|
151 | convert_path = function() {
|
---|
152 | var base = program.sourceMap.base;
|
---|
153 | delete options.sourceMap.base;
|
---|
154 | return function(name) {
|
---|
155 | return path.relative(base, name);
|
---|
156 | };
|
---|
157 | }();
|
---|
158 | }
|
---|
159 |
|
---|
160 | let filesList;
|
---|
161 | if (options.files && options.files.length) {
|
---|
162 | filesList = options.files;
|
---|
163 |
|
---|
164 | delete options.files;
|
---|
165 | } else if (program.args.length) {
|
---|
166 | filesList = program.args;
|
---|
167 | }
|
---|
168 |
|
---|
169 | if (filesList) {
|
---|
170 | simple_glob(filesList).forEach(function(name) {
|
---|
171 | files[convert_path(name)] = read_file(name);
|
---|
172 | });
|
---|
173 | } else {
|
---|
174 | await new Promise((resolve) => {
|
---|
175 | var chunks = [];
|
---|
176 | process.stdin.setEncoding("utf8");
|
---|
177 | process.stdin.on("data", function(chunk) {
|
---|
178 | chunks.push(chunk);
|
---|
179 | }).on("end", function() {
|
---|
180 | files = [ chunks.join("") ];
|
---|
181 | resolve();
|
---|
182 | });
|
---|
183 | process.stdin.resume();
|
---|
184 | });
|
---|
185 | }
|
---|
186 |
|
---|
187 | await run_cli();
|
---|
188 |
|
---|
189 | function convert_ast(fn) {
|
---|
190 | return AST_Node.from_mozilla_ast(Object.keys(files).reduce(fn, null));
|
---|
191 | }
|
---|
192 |
|
---|
193 | async function run_cli() {
|
---|
194 | var content = program.sourceMap && program.sourceMap.content;
|
---|
195 | if (content && content !== "inline") {
|
---|
196 | options.sourceMap.content = read_file(content, content);
|
---|
197 | }
|
---|
198 | if (program.timings) options.timings = true;
|
---|
199 |
|
---|
200 | try {
|
---|
201 | if (program.parse) {
|
---|
202 | if (program.parse.acorn) {
|
---|
203 | files = convert_ast(function(toplevel, name) {
|
---|
204 | return require("acorn").parse(files[name], {
|
---|
205 | ecmaVersion: 2018,
|
---|
206 | locations: true,
|
---|
207 | program: toplevel,
|
---|
208 | sourceFile: name,
|
---|
209 | sourceType: options.module || program.parse.module ? "module" : "script"
|
---|
210 | });
|
---|
211 | });
|
---|
212 | } else if (program.parse.spidermonkey) {
|
---|
213 | files = convert_ast(function(toplevel, name) {
|
---|
214 | var obj = JSON.parse(files[name]);
|
---|
215 | if (!toplevel) return obj;
|
---|
216 | toplevel.body = toplevel.body.concat(obj.body);
|
---|
217 | return toplevel;
|
---|
218 | });
|
---|
219 | }
|
---|
220 | }
|
---|
221 | } catch (ex) {
|
---|
222 | fatal(ex);
|
---|
223 | }
|
---|
224 |
|
---|
225 | let result;
|
---|
226 | try {
|
---|
227 | result = await minify(files, options, fs);
|
---|
228 | } catch (ex) {
|
---|
229 | if (ex.name == "SyntaxError") {
|
---|
230 | print_error("Parse error at " + ex.filename + ":" + ex.line + "," + ex.col);
|
---|
231 | var col = ex.col;
|
---|
232 | var lines = files[ex.filename].split(/\r?\n/);
|
---|
233 | var line = lines[ex.line - 1];
|
---|
234 | if (!line && !col) {
|
---|
235 | line = lines[ex.line - 2];
|
---|
236 | col = line.length;
|
---|
237 | }
|
---|
238 | if (line) {
|
---|
239 | var limit = 70;
|
---|
240 | if (col > limit) {
|
---|
241 | line = line.slice(col - limit);
|
---|
242 | col = limit;
|
---|
243 | }
|
---|
244 | print_error(line.slice(0, 80));
|
---|
245 | print_error(line.slice(0, col).replace(/\S/g, " ") + "^");
|
---|
246 | }
|
---|
247 | }
|
---|
248 | if (ex.defs) {
|
---|
249 | print_error("Supported options:");
|
---|
250 | print_error(format_object(ex.defs));
|
---|
251 | }
|
---|
252 | fatal(ex);
|
---|
253 | return;
|
---|
254 | }
|
---|
255 |
|
---|
256 | if (program.output == "ast") {
|
---|
257 | if (!options.compress && !options.mangle) {
|
---|
258 | result.ast.figure_out_scope({});
|
---|
259 | }
|
---|
260 | console.log(JSON.stringify(result.ast, function(key, value) {
|
---|
261 | if (value) switch (key) {
|
---|
262 | case "thedef":
|
---|
263 | return symdef(value);
|
---|
264 | case "enclosed":
|
---|
265 | return value.length ? value.map(symdef) : undefined;
|
---|
266 | case "variables":
|
---|
267 | case "globals":
|
---|
268 | return value.size ? collect_from_map(value, symdef) : undefined;
|
---|
269 | }
|
---|
270 | if (skip_keys.has(key)) return;
|
---|
271 | if (value instanceof AST_Token) return;
|
---|
272 | if (value instanceof Map) return;
|
---|
273 | if (value instanceof AST_Node) {
|
---|
274 | var result = {
|
---|
275 | _class: "AST_" + value.TYPE
|
---|
276 | };
|
---|
277 | if (value.block_scope) {
|
---|
278 | result.variables = value.block_scope.variables;
|
---|
279 | result.enclosed = value.block_scope.enclosed;
|
---|
280 | }
|
---|
281 | value.CTOR.PROPS.forEach(function(prop) {
|
---|
282 | if (prop !== "block_scope") {
|
---|
283 | result[prop] = value[prop];
|
---|
284 | }
|
---|
285 | });
|
---|
286 | return result;
|
---|
287 | }
|
---|
288 | return value;
|
---|
289 | }, 2));
|
---|
290 | } else if (program.output == "spidermonkey") {
|
---|
291 | try {
|
---|
292 | const minified = await minify(
|
---|
293 | result.code,
|
---|
294 | {
|
---|
295 | compress: false,
|
---|
296 | mangle: false,
|
---|
297 | format: {
|
---|
298 | ast: true,
|
---|
299 | code: false
|
---|
300 | }
|
---|
301 | },
|
---|
302 | fs
|
---|
303 | );
|
---|
304 | console.log(JSON.stringify(minified.ast.to_mozilla_ast(), null, 2));
|
---|
305 | } catch (ex) {
|
---|
306 | fatal(ex);
|
---|
307 | return;
|
---|
308 | }
|
---|
309 | } else if (program.output) {
|
---|
310 | fs.mkdirSync(path.dirname(program.output), { recursive: true });
|
---|
311 | fs.writeFileSync(program.output, result.code);
|
---|
312 | if (options.sourceMap && options.sourceMap.url !== "inline" && result.map) {
|
---|
313 | fs.writeFileSync(program.output + ".map", result.map);
|
---|
314 | }
|
---|
315 | } else {
|
---|
316 | console.log(result.code);
|
---|
317 | }
|
---|
318 | if (program.nameCache) {
|
---|
319 | fs.writeFileSync(program.nameCache, JSON.stringify(options.nameCache));
|
---|
320 | }
|
---|
321 | if (result.timings) for (var phase in result.timings) {
|
---|
322 | print_error("- " + phase + ": " + result.timings[phase].toFixed(3) + "s");
|
---|
323 | }
|
---|
324 | }
|
---|
325 |
|
---|
326 | function fatal(message) {
|
---|
327 | if (message instanceof Error) message = message.stack.replace(/^\S*?Error:/, "ERROR:");
|
---|
328 | print_error(message);
|
---|
329 | process.exit(1);
|
---|
330 | }
|
---|
331 |
|
---|
332 | // A file glob function that only supports "*" and "?" wildcards in the basename.
|
---|
333 | // Example: "foo/bar/*baz??.*.js"
|
---|
334 | // Argument `glob` may be a string or an array of strings.
|
---|
335 | // Returns an array of strings. Garbage in, garbage out.
|
---|
336 | function simple_glob(glob) {
|
---|
337 | if (Array.isArray(glob)) {
|
---|
338 | return [].concat.apply([], glob.map(simple_glob));
|
---|
339 | }
|
---|
340 | if (glob && glob.match(/[*?]/)) {
|
---|
341 | var dir = path.dirname(glob);
|
---|
342 | try {
|
---|
343 | var entries = fs.readdirSync(dir);
|
---|
344 | } catch (ex) {}
|
---|
345 | if (entries) {
|
---|
346 | var pattern = "^" + path.basename(glob)
|
---|
347 | .replace(/[.+^$[\]\\(){}]/g, "\\$&")
|
---|
348 | .replace(/\*/g, "[^/\\\\]*")
|
---|
349 | .replace(/\?/g, "[^/\\\\]") + "$";
|
---|
350 | var mod = process.platform === "win32" ? "i" : "";
|
---|
351 | var rx = new RegExp(pattern, mod);
|
---|
352 | var results = entries.filter(function(name) {
|
---|
353 | return rx.test(name);
|
---|
354 | }).map(function(name) {
|
---|
355 | return path.join(dir, name);
|
---|
356 | });
|
---|
357 | if (results.length) return results;
|
---|
358 | }
|
---|
359 | }
|
---|
360 | return [ glob ];
|
---|
361 | }
|
---|
362 |
|
---|
363 | function read_file(path, default_value) {
|
---|
364 | try {
|
---|
365 | return fs.readFileSync(path, "utf8");
|
---|
366 | } catch (ex) {
|
---|
367 | if ((ex.code == "ENOENT" || ex.code == "ENAMETOOLONG") && default_value != null) return default_value;
|
---|
368 | fatal(ex);
|
---|
369 | }
|
---|
370 | }
|
---|
371 |
|
---|
372 | function parse_js(flag) {
|
---|
373 | return function(value, options) {
|
---|
374 | options = options || {};
|
---|
375 | try {
|
---|
376 | walk(parse(value, { expression: true }), node => {
|
---|
377 | if (node instanceof AST_Assign) {
|
---|
378 | var name = node.left.print_to_string();
|
---|
379 | var value = node.right;
|
---|
380 | if (flag) {
|
---|
381 | options[name] = value;
|
---|
382 | } else if (value instanceof AST_Array) {
|
---|
383 | options[name] = value.elements.map(to_string);
|
---|
384 | } else if (value instanceof AST_RegExp) {
|
---|
385 | value = value.value;
|
---|
386 | options[name] = new RegExp(value.source, value.flags);
|
---|
387 | } else {
|
---|
388 | options[name] = to_string(value);
|
---|
389 | }
|
---|
390 | return true;
|
---|
391 | }
|
---|
392 | if (node instanceof AST_Symbol || node instanceof AST_PropAccess) {
|
---|
393 | var name = node.print_to_string();
|
---|
394 | options[name] = true;
|
---|
395 | return true;
|
---|
396 | }
|
---|
397 | if (!(node instanceof AST_Sequence)) throw node;
|
---|
398 |
|
---|
399 | function to_string(value) {
|
---|
400 | return value instanceof AST_Constant ? value.getValue() : value.print_to_string({
|
---|
401 | quote_keys: true
|
---|
402 | });
|
---|
403 | }
|
---|
404 | });
|
---|
405 | } catch(ex) {
|
---|
406 | if (flag) {
|
---|
407 | fatal("Error parsing arguments for '" + flag + "': " + value);
|
---|
408 | } else {
|
---|
409 | options[value] = null;
|
---|
410 | }
|
---|
411 | }
|
---|
412 | return options;
|
---|
413 | };
|
---|
414 | }
|
---|
415 |
|
---|
416 | function symdef(def) {
|
---|
417 | var ret = (1e6 + def.id) + " " + def.name;
|
---|
418 | if (def.mangled_name) ret += " " + def.mangled_name;
|
---|
419 | return ret;
|
---|
420 | }
|
---|
421 |
|
---|
422 | function collect_from_map(map, callback) {
|
---|
423 | var result = [];
|
---|
424 | map.forEach(function (def) {
|
---|
425 | result.push(callback(def));
|
---|
426 | });
|
---|
427 | return result;
|
---|
428 | }
|
---|
429 |
|
---|
430 | function format_object(obj) {
|
---|
431 | var lines = [];
|
---|
432 | var padding = "";
|
---|
433 | Object.keys(obj).map(function(name) {
|
---|
434 | if (padding.length < name.length) padding = Array(name.length + 1).join(" ");
|
---|
435 | return [ name, JSON.stringify(obj[name]) ];
|
---|
436 | }).forEach(function(tokens) {
|
---|
437 | lines.push(" " + tokens[0] + padding.slice(tokens[0].length - 2) + tokens[1]);
|
---|
438 | });
|
---|
439 | return lines.join("\n");
|
---|
440 | }
|
---|
441 |
|
---|
442 | function print_error(msg) {
|
---|
443 | process.stderr.write(msg);
|
---|
444 | process.stderr.write("\n");
|
---|
445 | }
|
---|
446 |
|
---|
447 | function describe_ast() {
|
---|
448 | var out = OutputStream({ beautify: true });
|
---|
449 | function doitem(ctor) {
|
---|
450 | out.print("AST_" + ctor.TYPE);
|
---|
451 | const props = ctor.SELF_PROPS.filter(prop => !/^\$/.test(prop));
|
---|
452 |
|
---|
453 | if (props.length > 0) {
|
---|
454 | out.space();
|
---|
455 | out.with_parens(function() {
|
---|
456 | props.forEach(function(prop, i) {
|
---|
457 | if (i) out.space();
|
---|
458 | out.print(prop);
|
---|
459 | });
|
---|
460 | });
|
---|
461 | }
|
---|
462 |
|
---|
463 | if (ctor.documentation) {
|
---|
464 | out.space();
|
---|
465 | out.print_string(ctor.documentation);
|
---|
466 | }
|
---|
467 |
|
---|
468 | if (ctor.SUBCLASSES.length > 0) {
|
---|
469 | out.space();
|
---|
470 | out.with_block(function() {
|
---|
471 | ctor.SUBCLASSES.forEach(function(ctor) {
|
---|
472 | out.indent();
|
---|
473 | doitem(ctor);
|
---|
474 | out.newline();
|
---|
475 | });
|
---|
476 | });
|
---|
477 | }
|
---|
478 | }
|
---|
479 | doitem(AST_Node);
|
---|
480 | return out + "\n";
|
---|
481 | }
|
---|
482 | }
|
---|