mirror of https://github.com/webpack/webpack.git
241 lines
7.2 KiB
JavaScript
241 lines
7.2 KiB
JavaScript
"use strict";
|
|
|
|
require("./helpers/warmup-webpack");
|
|
|
|
const path = require("path");
|
|
const fs = require("graceful-fs");
|
|
const rimraf = require("rimraf");
|
|
const webpack = require("..");
|
|
const captureStdio = require("./helpers/captureStdio");
|
|
|
|
/**
|
|
* Escapes regular expression metacharacters
|
|
* @param {string} str String to quote
|
|
* @returns {string} Escaped string
|
|
*/
|
|
const quoteMeta = str => str.replace(/[-[\]\\/{}()*+?.^$|]/g, "\\$&");
|
|
|
|
const base = path.join(__dirname, "statsCases");
|
|
const outputBase = path.join(__dirname, "js", "stats");
|
|
const tests = fs
|
|
.readdirSync(base)
|
|
.filter(
|
|
testName =>
|
|
fs.existsSync(path.join(base, testName, "index.js")) ||
|
|
fs.existsSync(path.join(base, testName, "webpack.config.js"))
|
|
)
|
|
.filter(testName => {
|
|
const testDirectory = path.join(base, testName);
|
|
const filterPath = path.join(testDirectory, "test.filter.js");
|
|
if (fs.existsSync(filterPath) && !require(filterPath)()) {
|
|
// eslint-disable-next-line jest/no-disabled-tests, jest/valid-describe-callback
|
|
describe.skip(testName, () => it("filtered", () => {}));
|
|
|
|
return false;
|
|
}
|
|
return true;
|
|
});
|
|
|
|
describe("StatsTestCases", () => {
|
|
jest.setTimeout(30000);
|
|
let stderr;
|
|
|
|
beforeEach(() => {
|
|
stderr = captureStdio(process.stderr, true);
|
|
});
|
|
|
|
afterEach(() => {
|
|
stderr.restore();
|
|
});
|
|
|
|
for (const testName of tests) {
|
|
// eslint-disable-next-line no-loop-func
|
|
it(`should print correct stats for ${testName}`, done => {
|
|
const outputDirectory = path.join(outputBase, testName);
|
|
rimraf.sync(outputDirectory);
|
|
fs.mkdirSync(outputDirectory, { recursive: true });
|
|
let options = {
|
|
mode: "development",
|
|
entry: "./index",
|
|
output: {
|
|
filename: "bundle.js"
|
|
}
|
|
};
|
|
if (fs.existsSync(path.join(base, testName, "webpack.config.js"))) {
|
|
options = require(path.join(base, testName, "webpack.config.js"));
|
|
}
|
|
let testConfig = {};
|
|
try {
|
|
// try to load a test file
|
|
testConfig = Object.assign(
|
|
testConfig,
|
|
require(path.join(base, testName, "test.config.js"))
|
|
);
|
|
} catch (_err) {
|
|
// ignored
|
|
}
|
|
|
|
const resolvedOptions = Array.isArray(options) ? options : [options];
|
|
for (const options of resolvedOptions) {
|
|
if (!options.context) options.context = path.join(base, testName);
|
|
if (!options.output) options.output = options.output || {};
|
|
if (!options.output.path) options.output.path = outputDirectory;
|
|
if (!options.plugins) options.plugins = [];
|
|
if (!options.optimization) options.optimization = {};
|
|
if (options.optimization.minimize === undefined) {
|
|
options.optimization.minimize = false;
|
|
}
|
|
if (
|
|
options.cache &&
|
|
options.cache !== true &&
|
|
options.cache.type === "filesystem"
|
|
) {
|
|
options.cache.cacheDirectory = path.resolve(
|
|
outputBase,
|
|
".cache",
|
|
testName
|
|
);
|
|
}
|
|
}
|
|
const c = webpack(options);
|
|
const compilers = c.compilers ? c.compilers : [c];
|
|
for (const c of compilers) {
|
|
const ifs = c.inputFileSystem;
|
|
c.inputFileSystem = Object.create(ifs);
|
|
c.inputFileSystem.readFile = function readFile() {
|
|
// eslint-disable-next-line prefer-rest-params
|
|
const args = Array.prototype.slice.call(arguments);
|
|
const callback = args.pop();
|
|
// eslint-disable-next-line no-useless-call
|
|
ifs.readFile.apply(ifs, [
|
|
...args,
|
|
(err, result) => {
|
|
if (err) return callback(err);
|
|
if (!/\.(js|json|txt)$/.test(args[0])) {
|
|
return callback(null, result);
|
|
}
|
|
callback(null, result.toString("utf8").replace(/\r/g, ""));
|
|
}
|
|
]);
|
|
};
|
|
c.hooks.compilation.tap("StatsTestCasesTest", compilation => {
|
|
for (const hook of [
|
|
"optimize",
|
|
"optimizeModules",
|
|
"optimizeChunks",
|
|
"afterOptimizeTree",
|
|
"afterOptimizeAssets",
|
|
"beforeHash"
|
|
]) {
|
|
compilation.hooks[hook].tap("TestCasesTest", () =>
|
|
compilation.checkConstraints()
|
|
);
|
|
}
|
|
});
|
|
}
|
|
c.run((err, stats) => {
|
|
if (err) return done(err);
|
|
for (const compilation of [
|
|
...(stats.stats ? stats.stats : [stats])
|
|
].map(s => s.compilation)) {
|
|
compilation.logging.delete("webpack.Compilation.ModuleProfile");
|
|
}
|
|
expect(stats.hasErrors()).toBe(testName.endsWith("error"));
|
|
if (!testName.endsWith("error") && stats.hasErrors()) {
|
|
return done(
|
|
new Error(
|
|
stats.toString({
|
|
all: false,
|
|
errors: true,
|
|
errorStack: true,
|
|
errorDetails: true
|
|
})
|
|
)
|
|
);
|
|
}
|
|
fs.writeFileSync(
|
|
path.join(outputBase, testName, "stats.txt"),
|
|
stats.toString({
|
|
preset: "verbose",
|
|
context: path.join(base, testName),
|
|
colors: false
|
|
}),
|
|
"utf8"
|
|
);
|
|
|
|
let toStringOptions = {
|
|
context: path.join(base, testName),
|
|
colors: false
|
|
};
|
|
let hasColorSetting = false;
|
|
if (typeof c.options.stats !== "undefined") {
|
|
toStringOptions = c.options.stats;
|
|
if (toStringOptions === null || typeof toStringOptions !== "object") {
|
|
toStringOptions = { preset: toStringOptions };
|
|
}
|
|
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 stats.stats ? 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;
|
|
}
|
|
let actual = stats.toString(toStringOptions);
|
|
expect(typeof actual).toBe("string");
|
|
if (!hasColorSetting) {
|
|
actual = stderr.toString() + actual;
|
|
actual = actual
|
|
.replace(/\u001B\[[0-9;]*m/g, "")
|
|
.replace(/[.0-9]+(\s?ms)/g, "X$1");
|
|
} else {
|
|
actual = stderr.toStringRaw() + actual;
|
|
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>")
|
|
.replace(/[.0-9]+(<\/CLR>)?(\s?ms)/g, "X$1$2");
|
|
}
|
|
// cspell:ignore Xdir
|
|
const testPath = path.join(base, testName);
|
|
actual = actual
|
|
.replace(/\r\n?/g, "\n")
|
|
.replace(/webpack [^ )]+(\)?) compiled/g, "webpack x.x.x$1 compiled")
|
|
.replace(new RegExp(quoteMeta(testPath), "g"), `Xdir/${testName}`)
|
|
.replace(/(\w)\\(\w)/g, "$1/$2")
|
|
.replace(/, additional resolving: X ms/g, "")
|
|
.replace(/Unexpected identifier '.+?'/g, "Unexpected identifier")
|
|
.replace(/[.0-9]+(\s?(bytes|KiB|MiB|GiB))/g, "X$1")
|
|
.replace(
|
|
/ms\s\([0-9a-f]{6,32}\)|(?![0-9]+-)[0-9a-f-]{6,32}\./g,
|
|
match => `${match.replace(/[0-9a-f]/g, "X")}`
|
|
)
|
|
// Normalize stack traces between Jest v27 and v30
|
|
// Jest v27: at Object.<anonymous>.module.exports
|
|
// Jest v30: at Object.module.exports
|
|
.replace(/Object\.<anonymous>\./g, "Object.");
|
|
expect(actual).toMatchSnapshot();
|
|
|
|
if (testConfig.validate) {
|
|
try {
|
|
testConfig.validate(stats, stderr.toString());
|
|
} catch (err) {
|
|
done(err);
|
|
return;
|
|
}
|
|
}
|
|
|
|
done();
|
|
});
|
|
});
|
|
}
|
|
});
|