webpack/test/WatchTestCases.template.js

567 lines
17 KiB
JavaScript
Raw Normal View History

2021-10-05 19:43:30 +08:00
"use strict";
require("./helpers/warmup-webpack");
2025-04-16 22:04:11 +08:00
/** @typedef {Record<string, EXPECTED_ANY>} Env */
/** @typedef {{ testPath: string, srcPath: string }} TestOptions */
2021-10-05 19:43:30 +08:00
const path = require("path");
const fs = require("graceful-fs");
const vm = require("vm");
const rimraf = require("rimraf");
const { pathToFileURL, fileURLToPath } = require("url");
2021-10-05 19:43:30 +08:00
const checkArrayExpectation = require("./checkArrayExpectation");
const createLazyTestEnv = require("./helpers/createLazyTestEnv");
const { remove } = require("./helpers/remove");
const prepareOptions = require("./helpers/prepareOptions");
const deprecationTracking = require("./helpers/deprecationTracking");
const FakeDocument = require("./helpers/FakeDocument");
const asModule = require("./helpers/asModule");
2021-10-05 19:43:30 +08:00
2025-04-22 20:42:33 +08:00
/**
* @param {string} src src
* @param {string} dest dest
* @param {boolean} initial is initial?
*/
2021-10-05 19:43:30 +08:00
function copyDiff(src, dest, initial) {
if (!fs.existsSync(dest)) fs.mkdirSync(dest);
const files = fs.readdirSync(src);
2024-08-02 02:36:27 +08:00
for (const filename of files) {
2021-10-05 19:43:30 +08:00
const srcFile = path.join(src, filename);
const destFile = path.join(dest, filename);
const directory = fs.statSync(srcFile).isDirectory();
if (directory) {
copyDiff(srcFile, destFile, initial);
} else {
2025-04-22 18:49:30 +08:00
const content = fs.readFileSync(srcFile);
2021-10-05 19:43:30 +08:00
if (/^DELETE\s*$/.test(content.toString("utf-8"))) {
fs.unlinkSync(destFile);
} else if (/^DELETE_DIRECTORY\s*$/.test(content.toString("utf-8"))) {
rimraf.sync(destFile);
} else {
fs.writeFileSync(destFile, content);
if (initial) {
const longTimeAgo = Date.now() - 1000 * 60 * 60 * 24;
fs.utimesSync(
destFile,
Date.now() - longTimeAgo,
Date.now() - longTimeAgo
);
}
}
}
2024-08-02 02:36:27 +08:00
}
2021-10-05 19:43:30 +08:00
}
const describeCases = config => {
describe(config.name, () => {
if (process.env.NO_WATCH_TESTS) {
2024-06-11 20:32:02 +08:00
// eslint-disable-next-line jest/no-disabled-tests
2021-10-05 19:43:30 +08:00
it.skip("long running tests excluded", () => {});
return;
}
const casesPath = path.join(__dirname, "watchCases");
2025-04-16 22:04:11 +08:00
const categories = fs.readdirSync(casesPath).map(cat => ({
2024-07-31 11:31:11 +08:00
name: cat,
tests: fs
.readdirSync(path.join(casesPath, cat))
2024-07-31 16:02:41 +08:00
.filter(folder => !folder.includes("_"))
2024-07-31 11:31:11 +08:00
.filter(testName => {
const testDirectory = path.join(casesPath, cat, testName);
const filterPath = path.join(testDirectory, "test.filter.js");
if (fs.existsSync(filterPath) && !require(filterPath)(config)) {
// eslint-disable-next-line jest/no-disabled-tests, jest/valid-describe-callback
describe.skip(testName, () => it("filtered", () => {}));
return false;
}
return true;
})
.sort()
}));
2021-10-05 19:43:30 +08:00
beforeAll(() => {
let dest = path.join(__dirname, "js");
if (!fs.existsSync(dest)) fs.mkdirSync(dest);
2024-07-31 10:39:30 +08:00
dest = path.join(__dirname, "js", `${config.name}-src`);
2021-10-05 19:43:30 +08:00
if (!fs.existsSync(dest)) fs.mkdirSync(dest);
});
2024-08-02 02:36:27 +08:00
for (const category of categories) {
2021-10-05 19:43:30 +08:00
beforeAll(() => {
const dest = path.join(
__dirname,
"js",
2024-07-31 10:39:30 +08:00
`${config.name}-src`,
2021-10-05 19:43:30 +08:00
category.name
);
if (!fs.existsSync(dest)) fs.mkdirSync(dest);
});
describe(category.name, () => {
2024-08-02 02:36:27 +08:00
for (const testName of category.tests) {
2021-10-05 19:43:30 +08:00
describe(testName, () => {
const tempDirectory = path.join(
__dirname,
"js",
2024-07-31 10:39:30 +08:00
`${config.name}-src`,
2021-10-05 19:43:30 +08:00
category.name,
testName
);
const testDirectory = path.join(casesPath, category.name, testName);
2025-04-16 22:04:11 +08:00
/** @type {TODO} */
2021-10-05 19:43:30 +08:00
const runs = fs
.readdirSync(testDirectory)
.sort()
2024-07-31 11:31:11 +08:00
.filter(name =>
fs.statSync(path.join(testDirectory, name)).isDirectory()
)
2021-10-05 19:43:30 +08:00
.map(name => ({ name }));
beforeAll(done => {
rimraf(tempDirectory, done);
});
2024-07-31 10:39:30 +08:00
it(`${testName} should compile`, done => {
const outputDirectory = path.join(
__dirname,
"js",
config.name,
category.name,
testName
);
2021-10-05 19:43:30 +08:00
2024-07-31 10:39:30 +08:00
rimraf.sync(outputDirectory);
2021-10-05 19:43:30 +08:00
2024-07-31 10:39:30 +08:00
let options = {};
const configPath = path.join(testDirectory, "webpack.config.js");
if (fs.existsSync(configPath)) {
options = prepareOptions(require(configPath), {
testPath: outputDirectory,
srcPath: tempDirectory
});
}
const applyConfig = (options, idx) => {
if (!options.mode) options.mode = "development";
if (!options.context) options.context = tempDirectory;
if (!options.entry) options.entry = "./index.js";
if (!options.target) options.target = "async-node";
if (!options.output) options.output = {};
if (!options.output.path) options.output.path = outputDirectory;
if (typeof options.output.pathinfo === "undefined")
options.output.pathinfo = true;
if (!options.output.filename)
options.output.filename = "bundle.js";
if (options.cache && options.cache.type === "filesystem") {
const cacheDirectory = path.join(tempDirectory, ".cache");
options.cache.cacheDirectory = cacheDirectory;
options.cache.name = `config-${idx}`;
2021-10-05 19:43:30 +08:00
}
2024-07-31 10:39:30 +08:00
if (config.experiments) {
if (!options.experiments) options.experiments = {};
for (const key of Object.keys(config.experiments)) {
if (options.experiments[key] === undefined)
options.experiments[key] = config.experiments[key];
2021-10-12 16:56:14 +08:00
}
2024-07-31 10:39:30 +08:00
}
if (config.optimization) {
if (!options.optimization) options.optimization = {};
for (const key of Object.keys(config.optimization)) {
if (options.optimization[key] === undefined)
options.optimization[key] = config.optimization[key];
2021-10-05 19:43:30 +08:00
}
}
2024-07-31 10:39:30 +08:00
};
if (Array.isArray(options)) {
2024-08-02 23:42:44 +08:00
for (const [idx, item] of options.entries()) {
applyConfig(item, idx);
}
2024-07-31 10:39:30 +08:00
} else {
applyConfig(options, 0);
}
2021-10-05 19:43:30 +08:00
2024-07-31 10:39:30 +08:00
const state = {};
let runIdx = 0;
let waitMode = false;
let run = runs[runIdx];
let triggeringFilename;
let lastHash = "";
const currentWatchStepModule = require("./helpers/currentWatchStep");
let compilationFinished = done;
currentWatchStepModule.step = run.name;
copyDiff(path.join(testDirectory, run.name), tempDirectory, true);
2021-10-05 19:43:30 +08:00
2024-07-31 10:39:30 +08:00
setTimeout(() => {
const deprecationTracker = deprecationTracking.start();
const webpack = require("..");
const compiler = webpack(options);
compiler.hooks.invalid.tap(
"WatchTestCasesTest",
(filename, mtime) => {
triggeringFilename = filename;
}
);
compiler.watch(
{
aggregateTimeout: 1000
},
async (err, stats) => {
2024-07-31 10:39:30 +08:00
if (err) return compilationFinished(err);
if (!stats) {
return compilationFinished(
new Error("No stats reported from Compiler")
2021-10-05 19:43:30 +08:00
);
2024-07-31 10:39:30 +08:00
}
if (stats.hash === lastHash) return;
lastHash = stats.hash;
if (run.done && lastHash !== stats.hash) {
return compilationFinished(
new Error(
`Compilation changed but no change was issued ${
lastHash
} != ${stats.hash} (run ${runIdx})\n` +
`Triggering change: ${triggeringFilename}`
2021-10-05 19:43:30 +08:00
)
2024-07-31 10:39:30 +08:00
);
}
if (waitMode) return;
run.done = true;
run.stats = stats;
if (err) return compilationFinished(err);
const statOptions = {
preset: "verbose",
cached: true,
cachedAssets: true,
cachedModules: true,
colors: false
};
fs.mkdirSync(outputDirectory, { recursive: true });
fs.writeFileSync(
path.join(
outputDirectory,
`stats.${runs[runIdx] && runs[runIdx].name}.txt`
),
stats.toString(statOptions),
"utf-8"
);
const jsonStats = stats.toJson({
errorDetails: true
});
if (
checkArrayExpectation(
path.join(testDirectory, run.name),
jsonStats,
"error",
"Error",
2024-11-01 21:55:49 +08:00
options,
2024-07-31 10:39:30 +08:00
compilationFinished
2021-10-05 19:43:30 +08:00
)
2024-07-31 10:39:30 +08:00
)
return;
if (
checkArrayExpectation(
path.join(testDirectory, run.name),
jsonStats,
"warning",
"Warning",
2024-11-01 21:55:49 +08:00
options,
2024-07-31 10:39:30 +08:00
compilationFinished
2021-10-05 19:43:30 +08:00
)
2024-07-31 10:39:30 +08:00
)
return;
2021-10-05 19:43:30 +08:00
2024-07-31 10:39:30 +08:00
const globalContext = {
2025-04-22 19:09:25 +08:00
console,
expect,
2024-07-31 10:39:30 +08:00
setTimeout,
clearTimeout,
document: new FakeDocument()
};
2021-10-05 19:43:30 +08:00
const baseModuleScope = {
console,
it: run.it,
beforeEach: _beforeEach,
afterEach: _afterEach,
expect,
jest,
STATS_JSON: jsonStats,
nsObj: m => {
Object.defineProperty(m, Symbol.toStringTag, {
value: "Module"
});
return m;
},
window: globalContext,
self: globalContext,
WATCH_STEP: run.name,
STATE: state
};
const esmCache = new Map();
const esmIdentifier = `${category.name}-${testName}`;
const esmContext = vm.createContext(baseModuleScope, {
name: "context for esm"
});
// ESM
const isModule =
options.experiments && options.experiments.outputModule;
2025-04-22 20:42:33 +08:00
/**
* @param {string} currentDirectory The directory to resolve relative paths from
* @param {string} module The module to require
* @param {("unlinked"|"evaluated")} esmMode The mode for ESM module handling
2025-04-22 20:42:33 +08:00
* @returns {EXPECTED_ANY} required module
* @private
*/
function _require(currentDirectory, module, esmMode) {
if (/^\.\.?\//.test(module) || path.isAbsolute(module)) {
2024-07-31 10:39:30 +08:00
let fn;
const p = path.isAbsolute(module)
? module
: path.join(currentDirectory, module);
const content = fs.readFileSync(p, "utf-8");
if (isModule) {
if (!vm.SourceTextModule)
throw new Error(
"Running this test requires '--experimental-vm-modules'.\nRun with 'node --experimental-vm-modules node_modules/jest-cli/bin/jest'."
);
let esm = esmCache.get(p);
if (!esm) {
esm = new vm.SourceTextModule(content, {
identifier: `${esmIdentifier}-${p}`,
url: `${pathToFileURL(p).href}?${esmIdentifier}`,
context: esmContext,
initializeImportMeta: (meta, module) => {
meta.url = pathToFileURL(p).href;
},
importModuleDynamically: async (
specifier,
module
) => {
const normalizedSpecifier =
specifier.startsWith("file:")
? `./${path.relative(
path.dirname(p),
fileURLToPath(specifier)
)}`
: specifier.replace(
/https:\/\/test.cases\/path\//,
"./"
);
const result = await _require(
currentDirectory,
normalizedSpecifier,
"evaluated"
);
return await asModule(result, module.context);
}
});
esmCache.set(p, esm);
}
if (esmMode === "unlinked") return esm;
return (async () => {
if (esmMode === "unlinked") return esm;
if (esm.status !== "evaluated") {
await esm.link(
async (specifier, referencingModule) =>
await asModule(
await _require(
path.dirname(
referencingModule.identifier
? referencingModule.identifier.slice(
esmIdentifier.length + 1
)
: fileURLToPath(referencingModule.url)
),
specifier,
"unlinked"
),
referencingModule.context,
true
)
);
// node.js 10 needs instantiate
if (esm.instantiate) esm.instantiate();
await esm.evaluate();
}
if (esmMode === "evaluated") return esm;
const ns = esm.namespace;
return ns.default && ns.default instanceof Promise
? ns.default
: ns;
})();
2024-07-31 10:39:30 +08:00
}
2024-07-31 10:39:30 +08:00
if (
options.target === "web" ||
options.target === "webworker"
) {
fn = vm.runInNewContext(
2024-07-31 12:23:44 +08:00
"(function(require, module, exports, __dirname, __filename, it, WATCH_STEP, STATS_JSON, STATE, expect, window, self) {" +
2024-07-31 10:39:30 +08:00
`function nsObj(m) { Object.defineProperty(m, Symbol.toStringTag, { value: "Module" }); return m; }${
content
}\n})`,
2021-10-05 19:43:30 +08:00
globalContext,
2024-07-31 10:39:30 +08:00
p
);
} else {
fn = vm.runInThisContext(
2024-07-31 12:23:44 +08:00
"(function(require, module, exports, __dirname, __filename, it, WATCH_STEP, STATS_JSON, STATE, expect) {" +
"global.expect = expect;" +
2024-07-31 10:39:30 +08:00
`function nsObj(m) { Object.defineProperty(m, Symbol.toStringTag, { value: "Module" }); return m; }${
content
}\n})`,
p
2021-10-05 19:43:30 +08:00
);
2024-07-31 04:21:27 +08:00
}
2024-07-31 10:39:30 +08:00
const m = {
exports: {}
};
fn.call(
m.exports,
_require.bind(null, path.dirname(p)),
m,
m.exports,
path.dirname(p),
p,
run.it,
run.name,
jsonStats,
state,
expect,
globalContext,
globalContext
2024-07-31 05:43:19 +08:00
);
2024-07-31 10:39:30 +08:00
return module.exports;
} else if (
testConfig.modules &&
module in testConfig.modules
) {
return testConfig.modules[module];
2021-10-05 19:43:30 +08:00
}
2024-07-31 10:39:30 +08:00
return jest.requireActual(module);
}
2021-10-05 19:43:30 +08:00
2024-07-31 10:39:30 +08:00
let testConfig = {};
try {
// try to load a test file
testConfig = require(
path.join(testDirectory, "test.config.js")
2021-10-05 19:43:30 +08:00
);
2024-07-31 15:37:05 +08:00
} catch (_err) {
2024-07-31 10:39:30 +08:00
// empty
}
2021-10-05 19:43:30 +08:00
2024-07-31 10:39:30 +08:00
if (testConfig.noTests)
return process.nextTick(compilationFinished);
const getBundle = (outputDirectory, module) => {
if (Array.isArray(module)) {
return module.map(arg =>
path.join(outputDirectory, arg)
);
} else if (module instanceof RegExp) {
return fs
.readdirSync(outputDirectory)
.filter(f => module.test(f))
.map(f => path.join(outputDirectory, f));
}
return [path.join(outputDirectory, module)];
};
const promises = [];
for (const p of getBundle(
2024-07-31 10:39:30 +08:00
outputDirectory,
testConfig.bundlePath || "./bundle.js"
)) {
promises.push(
Promise.resolve().then(() =>
_require(outputDirectory, p)
)
);
}
await Promise.all(promises);
2021-10-05 19:43:30 +08:00
2024-07-31 10:39:30 +08:00
if (run.getNumberOfTests() < 1)
return compilationFinished(
new Error("No tests exported by test case")
2021-10-05 19:43:30 +08:00
);
2024-07-31 10:39:30 +08:00
run.it(
"should compile the next step",
done => {
runIdx++;
if (runIdx < runs.length) {
run = runs[runIdx];
waitMode = true;
setTimeout(() => {
waitMode = false;
compilationFinished = done;
currentWatchStepModule.step = run.name;
copyDiff(
path.join(testDirectory, run.name),
tempDirectory,
false
);
}, 1500);
} else {
const deprecations = deprecationTracker();
if (
checkArrayExpectation(
testDirectory,
{ deprecations },
"deprecation",
"Deprecation",
2024-11-01 21:55:49 +08:00
options,
2024-07-31 10:39:30 +08:00
done
)
) {
compiler.close(() => {});
return;
}
compiler.close(done);
}
},
45000
);
compilationFinished();
}
);
}, 300);
}, 45000);
2021-10-05 19:43:30 +08:00
for (const run of runs) {
const { it: _it, getNumberOfTests } = createLazyTestEnv(
10000,
run.name
);
run.it = _it;
run.getNumberOfTests = getNumberOfTests;
it(`${run.name} should allow to read stats`, done => {
if (run.stats) {
run.stats.toString({ all: true });
run.stats = undefined;
}
done();
});
}
afterAll(() => {
remove(tempDirectory);
});
const {
it: _it,
beforeEach: _beforeEach,
afterEach: _afterEach
} = createLazyTestEnv(10000);
2021-10-05 19:43:30 +08:00
});
2024-08-02 02:36:27 +08:00
}
2021-10-05 19:43:30 +08:00
});
2024-08-02 02:36:27 +08:00
}
2021-10-05 19:43:30 +08:00
});
};
2024-07-31 04:54:55 +08:00
// eslint-disable-next-line jest/no-export
module.exports.describeCases = describeCases;