webpack/test/StatsTestCases.basictest.js

200 lines
6.1 KiB
JavaScript
Raw Normal View History

2017-01-18 23:11:11 +08:00
"use strict";
const path = require("path");
const fs = require("graceful-fs");
2019-11-04 23:48:47 +08:00
const rimraf = require("rimraf");
2019-07-31 23:29:21 +08:00
const captureStdio = require("./helpers/captureStdio");
const webpack = require("..");
2015-04-27 03:19:39 +08:00
/**
* Escapes regular expression metacharacters
* @param {string} str String to quote
* @returns {string} Escaped string
*/
const quotemeta = str => {
return str.replace(/[-[\]\\/{}()*+?.^$|]/g, "\\$&");
};
2017-01-18 23:11:11 +08:00
const base = path.join(__dirname, "statsCases");
const outputBase = path.join(__dirname, "js", "stats");
2018-02-25 18:46:17 +08:00
const tests = fs
.readdirSync(base)
.filter(
testName =>
fs.existsSync(path.join(base, testName, "index.js")) ||
fs.existsSync(path.join(base, testName, "webpack.config.js"))
2018-07-02 23:44:47 +08:00
)
.filter(testName => {
const testDirectory = path.join(base, testName);
const filterPath = path.join(testDirectory, "test.filter.js");
if (fs.existsSync(filterPath) && !require(filterPath)()) {
describe.skip(testName, () => it("filtered"));
return false;
}
return true;
});
2015-04-27 03:19:39 +08:00
describe("StatsTestCases", () => {
2021-07-15 17:20:48 +08:00
jest.setTimeout(60000);
let stderr;
beforeEach(() => {
stderr = captureStdio(process.stderr, true);
});
afterEach(() => {
stderr.restore();
});
2017-01-18 23:11:11 +08:00
tests.forEach(testName => {
2018-02-25 18:46:17 +08:00
it("should print correct stats for " + testName, done => {
2019-11-04 23:48:47 +08:00
const outputDirectory = path.join(outputBase, testName);
rimraf.sync(outputDirectory);
fs.mkdirSync(outputDirectory, { recursive: true });
2017-01-18 23:11:11 +08:00
let options = {
mode: "development",
entry: "./index",
output: {
filename: "bundle.js"
}
2015-04-27 03:19:39 +08:00
};
2018-02-25 18:46:17 +08:00
if (fs.existsSync(path.join(base, testName, "webpack.config.js"))) {
2018-04-11 04:52:22 +08:00
options = require(path.join(base, testName, "webpack.config.js"));
2015-04-27 03:19:39 +08:00
}
let testConfig = {};
try {
// try to load a test file
testConfig = Object.assign(
testConfig,
require(path.join(base, testName, "test.config.js"))
);
} catch (e) {
// ignored
}
2018-02-25 18:46:17 +08:00
(Array.isArray(options) ? options : [options]).forEach(options => {
if (!options.context) options.context = path.join(base, testName);
if (!options.output) options.output = options.output || {};
2019-11-04 23:48:47 +08:00
if (!options.output.path) options.output.path = outputDirectory;
2018-02-25 18:46:17 +08:00
if (!options.plugins) options.plugins = [];
if (!options.optimization) options.optimization = {};
if (options.optimization.minimize === undefined)
options.optimization.minimize = false;
2015-09-03 20:17:11 +08:00
});
2017-01-18 23:11:11 +08:00
const c = webpack(options);
const compilers = c.compilers ? c.compilers : [c];
2018-02-25 18:46:17 +08:00
compilers.forEach(c => {
2017-01-18 23:11:11 +08:00
const ifs = c.inputFileSystem;
2015-09-03 20:17:11 +08:00
c.inputFileSystem = Object.create(ifs);
2020-03-29 06:10:15 +08:00
c.inputFileSystem.readFile = function () {
2017-01-18 23:11:11 +08:00
const args = Array.prototype.slice.call(arguments);
const callback = args.pop();
2018-02-25 18:46:17 +08:00
ifs.readFile.apply(
ifs,
args.concat([
(err, result) => {
if (err) return callback(err);
if (!/\.(js|json|txt)$/.test(args[0]))
return callback(null, result);
2018-02-25 18:46:17 +08:00
callback(null, result.toString("utf-8").replace(/\r/g, ""));
}
])
);
2015-09-03 20:17:11 +08:00
};
c.hooks.compilation.tap("StatsTestCasesTest", compilation => {
[
"optimize",
"optimizeModules",
"optimizeChunks",
"afterOptimizeTree",
"afterOptimizeAssets",
"beforeHash"
].forEach(hook => {
compilation.hooks[hook].tap("TestCasesTest", () =>
compilation.checkConstraints()
);
});
});
2015-09-03 20:17:11 +08:00
});
2017-01-18 23:11:11 +08:00
c.run((err, stats) => {
2018-02-25 18:46:17 +08:00
if (err) return done(err);
2021-03-08 16:13:41 +08:00
for (const compilation of []
.concat(stats.stats || stats)
.map(s => s.compilation)) {
compilation.logging.delete("webpack.Compilation.ModuleProfile");
}
2018-02-25 18:46:17 +08:00
if (/error$/.test(testName)) {
2018-01-24 23:00:43 +08:00
expect(stats.hasErrors()).toBe(true);
2018-02-25 18:46:17 +08:00
} else if (stats.hasErrors()) {
return done(
new Error(
stats.toString({
all: false,
errors: true,
errorStack: true,
errorDetails: true
})
)
);
2019-05-10 03:34:28 +08:00
} else {
fs.writeFileSync(
path.join(outputBase, testName, "stats.txt"),
stats.toString({
preset: "verbose",
context: path.join(base, testName),
colors: false
}),
"utf-8"
);
}
2017-01-18 23:11:11 +08:00
let toStringOptions = {
context: path.join(base, testName),
2015-08-09 18:42:43 +08:00
colors: false
};
2017-01-18 23:11:11 +08:00
let hasColorSetting = false;
if (typeof c.options.stats !== "undefined") {
toStringOptions = c.options.stats;
2018-02-25 18:46:17 +08:00
if (toStringOptions === null || typeof toStringOptions !== "object")
toStringOptions = { preset: toStringOptions };
2018-02-25 18:46:17 +08:00
if (!toStringOptions.context)
toStringOptions.context = path.join(base, testName);
hasColorSetting = typeof toStringOptions.colors !== "undefined";
}
if (Array.isArray(c.options) && !toStringOptions.children) {
toStringOptions.children = c.options.map(o => o.stats);
}
// mock timestamps
for (const { compilation: s } of [].concat(stats.stats || stats)) {
expect(s.startTime).toBeGreaterThan(0);
expect(s.endTime).toBeGreaterThan(0);
s.endTime = new Date("04/20/1970, 12:42:42 PM").getTime();
s.startTime = s.endTime - 1234;
}
2017-01-18 23:11:11 +08:00
let actual = stats.toString(toStringOptions);
2018-01-24 23:00:43 +08:00
expect(typeof actual).toBe("string");
2018-02-25 18:46:17 +08:00
if (!hasColorSetting) {
actual = stderr.toString() + actual;
2015-11-14 21:08:04 +08:00
actual = actual
.replace(/\u001b\[[0-9;]*m/g, "")
2019-07-24 16:51:04 +08:00
.replace(/[.0-9]+(\s?ms)/g, "X$1");
2015-11-14 21:08:04 +08:00
} else {
actual = stderr.toStringRaw() + actual;
2015-11-14 21:08:04 +08:00
actual = actual
.replace(/\u001b\[1m\u001b\[([0-9;]*)m/g, "<CLR=$1,BOLD>")
.replace(/\u001b\[1m/g, "<CLR=BOLD>")
.replace(/\u001b\[39m\u001b\[22m/g, "</CLR>")
.replace(/\u001b\[([0-9;]*)m/g, "<CLR=$1>")
2019-07-24 16:51:04 +08:00
.replace(/[.0-9]+(<\/CLR>)?(\s?ms)/g, "X$1$2");
2015-11-14 21:08:04 +08:00
}
2018-12-06 18:36:32 +08:00
const testPath = path.join(base, testName);
2015-11-14 21:08:04 +08:00
actual = actual
2015-09-03 20:42:00 +08:00
.replace(/\r\n?/g, "\n")
.replace(/webpack [^ )]+(\)?) compiled/g, "webpack x.x.x$1 compiled")
2019-07-24 16:51:04 +08:00
.replace(new RegExp(quotemeta(testPath), "g"), "Xdir/" + testName)
.replace(/(\w)\\(\w)/g, "$1/$2")
.replace(/, additional resolving: X ms/g, "");
2018-05-01 22:06:18 +08:00
expect(actual).toMatchSnapshot();
if (testConfig.validate) testConfig.validate(stats, stderr.toString());
2015-04-27 03:19:39 +08:00
done();
});
});
});
});