You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
141 lines
4.3 KiB
JavaScript
141 lines
4.3 KiB
JavaScript
/**
|
|
* timeRequire - measure the time to load all the subsequnt modules by hoocking require() calls
|
|
*
|
|
* @author Ciprian Popa (cyparu)
|
|
* @since 0.0.1
|
|
* @version 0.0.1
|
|
*/
|
|
|
|
"use strict";
|
|
|
|
var // setup vars
|
|
requireData = [],
|
|
write = process.stdout.write.bind(process.stdout),
|
|
relative = require("path").relative,
|
|
cwd = process.cwd(),
|
|
// require hooker should be first module loaded so all the other requires should count as well
|
|
/* jshint -W003 */
|
|
hook = require("./requireHook")(_hooker),
|
|
table = require("text-table"),
|
|
dateTime = require("date-time"),
|
|
prettyMs = require("pretty-ms"),
|
|
chalk = require("chalk"),
|
|
// extra locals
|
|
DEFAULT_COLUMNS = 80,
|
|
BAR_CHAR = process.platform === "win32" ? "■" : "▇",
|
|
sorted = hasArg("--sorted") || hasArg("--s"),
|
|
treshold = (hasArg("--verbose") || hasArg("--V")) ? 0.0: 0.01, // TODO - configure treshold using CLI ?
|
|
EXTRA_COLUMNS = sorted ? 24 : 20;
|
|
|
|
function hasArg(arg) {
|
|
return process.argv.indexOf(arg) !== -1;
|
|
}
|
|
|
|
function pad(count, seq) {
|
|
return (count > 1) ? new Array(count).join(seq) : "";
|
|
}
|
|
|
|
function log(str) {
|
|
write(str + "\n", "utf8");
|
|
}
|
|
|
|
/**
|
|
* Callback/listener used by requireHook hook to collect all the modules in their loading order
|
|
*/
|
|
function _hooker(data) {
|
|
var filename = relative(cwd, data.filename);
|
|
// use the shortest name
|
|
if (filename.length > data.filename) {
|
|
filename = data.filename;
|
|
}
|
|
requireData.push({
|
|
order: requireData.length, // loading order
|
|
time: data.startedIn, // time
|
|
label: data.name + " (" + filename + ")"
|
|
// name: data.name,
|
|
// filename: filename
|
|
});
|
|
}
|
|
|
|
function formatTable(tableData, totalTime) {
|
|
var NAME_FILE_REX = /(.+)( \(.+\))/,
|
|
maxColumns = process.stdout.columns || DEFAULT_COLUMNS,
|
|
validCount = 0,
|
|
longestRequire = tableData.reduce(function(acc, data) {
|
|
var avg = data.time / totalTime;
|
|
if (avg < treshold) {
|
|
return acc;
|
|
}
|
|
validCount++;
|
|
return Math.max(acc, data.label.length);
|
|
}, 0),
|
|
maxBarWidth = (longestRequire > maxColumns / 2) ? ((maxColumns - EXTRA_COLUMNS) / 2) : (maxColumns - (longestRequire + EXTRA_COLUMNS)),
|
|
processedTableData = [],
|
|
counter, maxOrderChars;
|
|
|
|
function shorten(name) {
|
|
var nameLength = name.length,
|
|
partLength, start, end;
|
|
if (name.length < maxBarWidth) {
|
|
return name;
|
|
}
|
|
partLength = Math.floor((maxBarWidth - 3) / 2);
|
|
start = name.substr(0, partLength + 1);
|
|
end = name.substr(nameLength - partLength);
|
|
return start.trim() + "..." + end.trim();
|
|
}
|
|
|
|
function createBar(percentage) {
|
|
var rounded = Math.round(percentage * 100);
|
|
return ((rounded === 0) ? "0" : (pad(Math.ceil(maxBarWidth * percentage) + 1, BAR_CHAR) + " " + rounded)) + "%";
|
|
}
|
|
|
|
// sort the data if needed
|
|
if (sorted) {
|
|
tableData.sort(function(e1, e2) {
|
|
return e2.time - e1.time;
|
|
});
|
|
}
|
|
// initialize the counter
|
|
counter = 1;
|
|
// get num ber of chars for padding
|
|
maxOrderChars = tableData.length.toString().length;
|
|
// push the header
|
|
processedTableData.push(["#" + (sorted ? " [order]" : ""), "module", "time", "%"]);
|
|
tableData.forEach(function(data) {
|
|
var avg = data.time / totalTime,
|
|
counterLabel, label, match;
|
|
// slect just data over the threshold
|
|
if (avg >= treshold) {
|
|
counterLabel = counter++;
|
|
// for sorted collumns show the order loading with padding
|
|
if (sorted) {
|
|
counterLabel += pad(maxOrderChars - data.order.toString().length + 1, " ") + " [" + data.order + "]";
|
|
}
|
|
label = shorten(data.label);
|
|
match = label.match(NAME_FILE_REX);
|
|
if (match) {
|
|
label = chalk.green(match[1]) + match[2];
|
|
}
|
|
processedTableData.push([counterLabel, label, chalk.yellow(prettyMs(data.time)), chalk.blue(createBar(avg))]);
|
|
}
|
|
});
|
|
|
|
return table(processedTableData, {
|
|
align: ["r", "l", "r", "l"],
|
|
stringLength: function(str) {
|
|
return chalk.stripColor(str).length;
|
|
}
|
|
});
|
|
}
|
|
|
|
// hook process exit to display the report at the end
|
|
process.once("exit", function() {
|
|
var startTime = hook.hookedAt,
|
|
totalTime = Date.now() - startTime.getTime();
|
|
log("\n\n" + chalk.underline("Start time: " + chalk.yellow("(" + dateTime(startTime) + ")") + " [treshold=" + (treshold * 100) + "%" + (sorted ? ",sorted" : "") + "]"));
|
|
log(formatTable(requireData, totalTime));
|
|
log(chalk.bold.blue("Total require(): ") + chalk.yellow(requireData.length));
|
|
log(chalk.bold.blue("Total time: ") + chalk.yellow(prettyMs(totalTime)));
|
|
});
|