webpack/bin/webpack.js

282 lines
8.5 KiB
JavaScript
Raw Normal View History

2012-03-10 20:11:23 +08:00
#!/usr/bin/env node
2012-03-12 04:50:55 +08:00
/*
MIT License http://www.opensource.org/licenses/mit-license.php
Author Tobias Koppers @sokra
*/
2012-03-10 20:11:23 +08:00
var path = require("path");
var fs = require("fs");
2012-03-15 07:05:29 +08:00
var util = require("util");
var sprintf = require("sprintf").sprintf;
2012-03-10 20:11:23 +08:00
var argv = require("optimist")
.usage("Usage: $0 <input> <output>")
2012-03-12 04:50:55 +08:00
2012-03-10 20:11:23 +08:00
.boolean("single")
.describe("single", "Disable Code Splitting")
.default("single", false)
2012-03-12 04:50:55 +08:00
2012-03-10 20:11:23 +08:00
.boolean("min")
.describe("min", "Minimize it with uglifyjs")
.default("min", false)
2012-03-12 04:50:55 +08:00
2012-03-10 20:11:23 +08:00
.boolean("filenames")
.describe("filenames", "Output Filenames Into File")
.default("filenames", false)
2012-03-12 04:50:55 +08:00
2012-03-10 20:11:23 +08:00
.string("options")
.describe("options", "Options JSON File")
2012-03-12 04:50:55 +08:00
2012-03-10 20:11:23 +08:00
.string("script-src-prefix")
.describe("script-src-prefix", "Path Prefix For JavaScript Loading")
2012-03-12 04:50:55 +08:00
2012-03-10 20:11:23 +08:00
.string("libary")
.describe("libary", "Stores the exports into this variable")
2012-03-12 04:50:55 +08:00
.boolean("colors")
.describe("colors", "Output Stats with colors")
.default("colors", false)
2012-03-15 07:05:29 +08:00
.boolean("json")
.describe("json", "Output Stats as JSON")
.default("json", false)
2012-04-03 22:26:08 +08:00
.boolean("by-size")
.describe("by-size", "Sort modules by size in Stats")
.default("by-size", false)
2012-04-05 20:59:01 +08:00
.boolean("verbose")
.describe("verbose", "Output dependencies in Stats")
.default("verbose", false)
.string("alias")
.describe("alias", "Set a alias name for a module. ex. http=http-browserify")
2012-03-15 07:05:29 +08:00
2012-04-03 22:26:08 +08:00
.boolean("debug")
.describe("debug", "Prints debug info to output files")
.default("debug", false)
2012-05-02 03:33:59 +08:00
.boolean("watch")
.describe("watch", "Recompiles on changes (except loaders)")
.default("watch", false)
.boolean("progress")
.describe("progress", "Displays a progress while compiling")
.default("progress", false)
2012-03-10 20:11:23 +08:00
.demand(1)
.argv;
var input = argv._[0],
output = argv._[1];
if (input && input[0] !== '/' && input[1] !== ':') {
input = path.join(process.cwd(), input);
}
if (output && output[0] !== '/' && input[1] !== ':') {
output = path.join(process.cwd(), output);
}
var options = {};
if(argv.options) {
options = JSON.parse(fs.readFileSync(argv.options, "utf-8"));
}
if(argv["script-src-prefix"]) {
options.scriptSrcPrefix = argv["script-src-prefix"];
}
if(argv.min) {
options.minimize = true;
}
2012-04-03 22:26:08 +08:00
if(argv.debug) {
options.debug = true;
}
2012-05-02 03:33:59 +08:00
if(argv.watch) {
options.watch = true;
}
2012-03-10 20:11:23 +08:00
if(argv.filenames) {
options.includeFilenames = true;
}
if(argv.libary) {
options.libary = argv.libary;
}
if(argv.alias) {
if(typeof argv.alias === "string")
argv.alias = [argv.alias];
options.resolve = options.resolve || {};
options.resolve.alias = options.resolve.alias || {};
var aliasObj = options.resolve.alias;
argv.alias.forEach(function(alias) {
alias = alias.split("=");
aliasObj[alias[0]] = alias[1];
});
}
2012-03-10 20:11:23 +08:00
var webpack = require("../lib/webpack.js");
2012-05-02 03:33:59 +08:00
function c(str) {
return argv.colors ? str : "";
}
if(!output) {
2012-03-10 20:11:23 +08:00
webpack(input, options, function(err, source) {
if(err) {
console.error(err);
return;
}
if(output) {
fs.writeFileSync(output, source, "utf-8");
} else {
process.stdout.write(source);
}
});
} else {
output = output || path.join(process.cwd(), "js", "web.js");
if(!options.outputDirectory) options.outputDirectory = path.dirname(output);
if(!options.output) options.output = path.basename(output);
if(!options.outputPostfix) options.outputPostfix = "." + path.basename(output);
2012-05-02 20:06:42 +08:00
if(argv.progress) {
if(!options.events) options.events = new (require("events").EventEmitter)();
var events = options.events;
var sum = 0;
var finished = 0;
var chars = 0;
function print() {
var msg = "";
if(sum > 0) {
msg += "compiling... (" + c("\033[1m\033[33m");
msg += sprintf("%4s", finished+"") + "/" + sprintf("%4s", sum+"");
msg += " " + sprintf("%4s", Math.floor(finished*100/sum)+"%");
msg += c("\033[39m\033[22m") + ")";
}
for(var i = 0; i < chars; i++)
process.stderr.write("\b");
process.stderr.write(msg);
chars = msg.length;
}
events.on("task", function(name) {
sum++;
print();
});
events.on("task-end", function(name) {
finished++;
if(name) {
for(var i = 0; i < chars; i++)
process.stderr.write("\b \b");
process.stderr.write(name + " " + c("\033[1m\033[32m") + "done" + c("\033[39m\033[22m") + "\n");
chars = 0;
}
print();
});
events.on("bundle", function(name) {
sum = 0;
finished = 0;
for(var i = 0; i < chars; i++)
process.stderr.write("\b \b");
chars = 0;
});
}
webpack(input, options, function(err, stats) {
2012-03-10 20:11:23 +08:00
if(err) {
console.error(err);
return;
}
2012-03-15 07:05:29 +08:00
if(argv.json)
console.log(util.inspect(stats, false, 10, argv.colors));
else {
2012-05-01 23:46:26 +08:00
console.log("Hash: "+c("\033[1m") + stats.hash + c("\033[22m"));
2012-04-05 20:59:01 +08:00
console.log("Chunks: "+c("\033[1m") + stats.chunkCount + c("\033[22m"));
console.log("Modules: "+c("\033[1m") + stats.modulesCount + c("\033[22m"));
console.log("Modules including duplicates: "+c("\033[1m") + stats.modulesIncludingDuplicates + c("\033[22m"));
console.log("Modules pre chunk: "+c("\033[1m") + stats.modulesPerChunk + c("\033[22m"));
console.log("Modules first chunk: "+c("\033[1m") + stats.modulesFirstChunk + c("\033[22m"));
2012-03-15 07:05:29 +08:00
if(stats.fileSizes)
2012-05-01 23:46:26 +08:00
Object.keys(stats.fileSizes).reverse().forEach(function(file) {
2012-04-05 20:59:01 +08:00
console.log(c("\033[1m") + sprintf("%" + (5 + options.output.length) + "s", file) + c("\033[22m")+": "+c("\033[1m") + sprintf("%8d", stats.fileSizes[file]) + c("\033[22m") + " characters");
2012-05-01 23:46:26 +08:00
});
2012-04-05 20:59:01 +08:00
var cwd = process.cwd();
var cwdParent = path.dirname(cwd);
var buildins = path.join(__dirname, "..");
cwd = cwd.replace(/[-[\]{}()*+?.,\\^$|#\s]/g, "\\$&");
cwd = new RegExp("^" + cwd + "|(!)" + cwd, "g");
2012-04-09 20:44:47 +08:00
var buildinsAsModule = cwd.test(buildins);
2012-04-05 20:59:01 +08:00
cwdParent = cwdParent.replace(/[-[\]{}()*+?.,\\^$|#\s]/g, "\\$&");
cwdParent = new RegExp("^" + cwdParent + "|(!)" + cwdParent, "g");
buildins = buildins.replace(/[-[\]{}()*+?.,\\^$|#\s]/g, "\\$&");
buildins = new RegExp("^" + buildins + "|(!)" + buildins, "g");
2012-05-03 06:30:27 +08:00
var node_modulesRegExpA = /\/node_modules\//g;
var node_modulesRegExpB = /\\node_modules\\/g;
var index_jsRegExp = /[\\\/]index.js!/g;
2012-04-05 20:59:01 +08:00
function compressFilename(filename) {
if(!filename)
return filename;
2012-04-09 20:44:47 +08:00
if(buildinsAsModule)
filename = filename.replace(buildins, "!(webpack)");
2012-04-05 20:59:01 +08:00
filename = filename.replace(cwd, "!.");
2012-04-09 20:44:47 +08:00
if(!buildinsAsModule)
filename = filename.replace(buildins, "!(webpack)");
2012-05-03 06:30:27 +08:00
filename = filename.replace(node_modulesRegExpA, "/~/");
filename = filename.replace(node_modulesRegExpB, "\\~\\");
filename = filename.replace(index_jsRegExp, "!");
return filename.replace(/^!|!$/, "");
2012-04-05 20:59:01 +08:00
}
2012-03-15 07:05:29 +08:00
if(stats.fileModules) {
console.log();
console.log(" <id> <size> <filename>");
if(argv.verbose)
console.log(" <reason> from <filename>");
2012-05-01 23:46:26 +08:00
Object.keys(stats.fileModules).reverse().forEach(function(file) {
2012-04-05 20:59:01 +08:00
console.log(c("\033[1m\033[32m") + file + c("\033[39m\033[22m"));
2012-03-15 07:05:29 +08:00
var modules = stats.fileModules[file];
if(argv["by-size"])
modules.sort(function(a, b) {
return b.size - a.size;
});
2012-03-15 07:05:29 +08:00
modules.forEach(function(module) {
console.log(" "+c("\033[1m") + sprintf("%3s", module.id+"") + " " + (typeof module.size === "number" ? sprintf("%9s", Math.round(module.size)+"") : " ") + " " +
2012-04-05 20:59:01 +08:00
(compressFilename(module.filename) ||
(module.dirname && ("[context] " + compressFilename(module.dirname))) ||
"[unknown]") + c("\033[22m"));
if(argv.verbose) {
module.reasons.forEach(function(reason) {
switch(reason.type) {
case "require":
console.log(" "+c("\033[36m")+"require (" + reason.count + "x) from " + compressFilename(reason.filename) + c("\033[39m"));
break;
case "context":
console.log(" "+c("\033[90m")+"context from " + compressFilename(reason.filename) + c("\033[39m"));
break;
case "async require":
console.log(" "+c("\033[35m")+"async require (" + reason.count + "x) from " + compressFilename(reason.filename) + c("\033[39m"));
break;
case "async context":
console.log(" "+c("\033[35ma")+"sync context from " + compressFilename(reason.filename) + c("\033[39m"));
break;
default:
console.log(" "+c("\033[31m") + reason.type + c("\033[39m"));
}
});
}
2012-03-15 07:05:29 +08:00
});
2012-05-01 23:46:26 +08:00
});
2012-03-15 07:05:29 +08:00
}
if(stats.warnings) {
stats.warnings.forEach(function(warning) {
2012-04-05 20:59:01 +08:00
console.log(c("\033[1m\033[33m")+"WARNING: " + warning + c("\033[39m\033[22m"));
2012-03-15 07:05:29 +08:00
});
}
2012-04-03 22:26:08 +08:00
if(stats.errors) {
stats.errors.forEach(function(error) {
2012-04-05 20:59:01 +08:00
console.log(c("\033[1m\033[31m")+"ERROR: " + error + c("\033[39m\033[22m"));
2012-04-03 22:26:08 +08:00
});
}
2012-03-15 07:05:29 +08:00
}
2012-03-10 20:11:23 +08:00
});
}