webpack/test/HotTestCases.template.js

415 lines
12 KiB
JavaScript
Raw Normal View History

"use strict";
require("./helpers/warmup-webpack");
const path = require("path");
const fs = require("graceful-fs");
const vm = require("vm");
const rimraf = require("rimraf");
const checkArrayExpectation = require("./checkArrayExpectation");
const createLazyTestEnv = require("./helpers/createLazyTestEnv");
const FakeDocument = require("./helpers/FakeDocument");
const casesPath = path.join(__dirname, "hotCases");
let categories = fs
.readdirSync(casesPath)
.filter(dir => fs.statSync(path.join(casesPath, dir)).isDirectory());
2024-07-31 11:31:11 +08:00
categories = categories.map(cat => ({
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
}));
const describeCases = config => {
describe(config.name, () => {
2024-08-02 02:36:27 +08:00
for (const category of categories) {
describe(category.name, () => {
2024-08-02 02:36:27 +08:00
for (const testName of category.tests) {
const testDirectory = path.join(casesPath, category.name, testName);
const filterPath = path.join(testDirectory, "test.filter.js");
2021-05-27 04:49:07 +08:00
if (fs.existsSync(filterPath) && !require(filterPath)(config)) {
2024-06-11 20:32:02 +08:00
// eslint-disable-next-line jest/no-disabled-tests
describe.skip(testName, () => {
it("filtered", () => {});
});
2024-08-02 02:36:27 +08:00
continue;
}
describe(testName, () => {
2021-01-21 23:32:49 +08:00
let compiler;
afterAll(callback => {
compiler.close(callback);
2021-06-25 19:38:38 +08:00
compiler = undefined;
2021-01-21 23:32:49 +08:00
});
2024-07-31 10:39:30 +08:00
it(`${testName} should compile`, done => {
const webpack = require("..");
const outputDirectory = path.join(
__dirname,
"js",
`hot-cases-${config.name}`,
category.name,
testName
);
rimraf.sync(outputDirectory);
const recordsPath = path.join(outputDirectory, "records.json");
const fakeUpdateLoaderOptions = {
updateIndex: 0
};
const configPath = path.join(testDirectory, "webpack.config.js");
let options = {};
if (fs.existsSync(configPath)) options = require(configPath);
if (typeof options === "function") {
options = options({ config });
}
if (!options.mode) options.mode = "development";
if (!options.devtool) options.devtool = false;
if (!options.context) options.context = testDirectory;
if (!options.entry) options.entry = "./index.js";
if (!options.output) options.output = {};
if (!options.output.path) options.output.path = outputDirectory;
if (!options.output.filename)
options.output.filename = "bundle.js";
if (!options.output.chunkFilename)
options.output.chunkFilename = "[name].chunk.[fullhash].js";
if (options.output.pathinfo === undefined)
options.output.pathinfo = true;
if (options.output.publicPath === undefined)
options.output.publicPath = "https://test.cases/path/";
if (options.output.library === undefined)
options.output.library = { type: "commonjs2" };
if (!options.optimization) options.optimization = {};
if (!options.optimization.moduleIds)
options.optimization.moduleIds = "named";
if (!options.module) options.module = {};
if (!options.module.rules) options.module.rules = [];
options.module.rules.push({
loader: path.join(
__dirname,
2024-07-31 10:39:30 +08:00
"hotCases",
"fake-update-loader.js"
),
enforce: "pre"
});
if (!options.target) options.target = config.target;
if (!options.plugins) options.plugins = [];
options.plugins.push(
new webpack.HotModuleReplacementPlugin(),
new webpack.LoaderOptionsPlugin(fakeUpdateLoaderOptions)
);
if (!options.recordsPath) options.recordsPath = recordsPath;
2024-11-30 01:35:09 +08:00
let testConfig = {};
try {
// try to load a test file
testConfig = Object.assign(
testConfig,
require(path.join(testDirectory, "test.config.js"))
);
} catch (_err) {
// ignored
}
const onCompiled = (err, stats) => {
2024-07-31 10:39:30 +08:00
if (err) return done(err);
const jsonStats = stats.toJson({
errorDetails: true
});
2024-07-31 10:39:30 +08:00
if (
checkArrayExpectation(
testDirectory,
jsonStats,
"error",
"Error",
2024-11-01 21:55:49 +08:00
options,
2024-07-31 10:39:30 +08:00
done
)
) {
return;
}
if (
checkArrayExpectation(
testDirectory,
jsonStats,
"warning",
"Warning",
2024-11-01 21:55:49 +08:00
options,
2024-07-31 10:39:30 +08:00
done
)
) {
return;
}
2024-07-31 10:39:30 +08:00
const urlToPath = url => {
if (url.startsWith("https://test.cases/path/"))
url = url.slice(24);
return path.resolve(outputDirectory, `./${url}`);
};
const urlToRelativePath = url => {
if (url.startsWith("https://test.cases/path/"))
url = url.slice(24);
return `./${url}`;
};
const window = {
2024-11-30 01:35:09 +08:00
_elements: [],
2024-07-31 10:39:30 +08:00
fetch: async url => {
try {
const buffer = await new Promise((resolve, reject) => {
fs.readFile(urlToPath(url), (err, b) =>
err ? reject(err) : resolve(b)
);
});
return {
status: 200,
ok: true,
json: async () => JSON.parse(buffer.toString("utf-8"))
};
} catch (err) {
if (err.code === "ENOENT") {
return {
2024-07-31 10:39:30 +08:00
status: 404,
ok: false
};
}
2024-07-31 10:39:30 +08:00
throw err;
}
},
importScripts: url => {
expect(url).toMatch(/^https:\/\/test\.cases\/path\//);
_require(urlToRelativePath(url));
},
document: {
createElement(type) {
const ele = {
2024-07-31 10:39:30 +08:00
_type: type,
2024-11-30 01:35:09 +08:00
getAttribute(name) {
return this[name];
},
2024-07-31 10:39:30 +08:00
setAttribute(name, value) {
2024-11-30 01:35:09 +08:00
this[name] = value;
},
removeAttribute(name) {
delete this[name];
2024-07-31 10:39:30 +08:00
},
parentNode: {
removeChild(node) {
2024-11-30 01:35:09 +08:00
window._elements = window._elements.filter(
item => item !== node
);
}
}
2024-07-31 10:39:30 +08:00
};
ele.sheet =
type === "link"
? new FakeDocument.FakeSheet(ele, outputDirectory)
: {};
return ele;
2020-10-27 18:53:17 +08:00
},
2024-07-31 10:39:30 +08:00
head: {
appendChild(element) {
2024-11-30 01:35:09 +08:00
window._elements.push(element);
2024-07-31 10:39:30 +08:00
if (element._type === "script") {
// run it
Promise.resolve().then(() => {
_require(urlToRelativePath(element.src));
});
2024-11-30 01:35:09 +08:00
} else if (element._type === "link") {
Promise.resolve().then(() => {
if (element.onload) {
// run it
element.onload({ type: "load" });
}
});
}
},
insertBefore(element, before) {
window._elements.push(element);
if (element._type === "script") {
// run it
Promise.resolve().then(() => {
_require(urlToRelativePath(element.src));
});
} else if (element._type === "link") {
// run it
Promise.resolve().then(() => {
element.onload({ type: "load" });
});
2024-07-31 10:39:30 +08:00
}
2020-10-27 18:53:17 +08:00
}
2024-07-31 10:39:30 +08:00
},
getElementsByTagName(name) {
if (name === "head") return [this.head];
2024-11-30 01:35:09 +08:00
if (name === "script" || name === "link") {
return window._elements.filter(
item => item._type === name
);
}
2024-07-31 10:39:30 +08:00
throw new Error("Not supported");
}
2024-07-31 10:39:30 +08:00
},
Worker: require("./helpers/createFakeWorker")({
outputDirectory
}),
EventSource: require("./helpers/EventSourceForNode"),
location: {
href: "https://test.cases/path/index.html",
origin: "https://test.cases",
toString() {
return "https://test.cases/path/index.html";
}
}
};
2024-11-30 01:35:09 +08:00
const moduleScope = {
window
};
if (testConfig.moduleScope) {
testConfig.moduleScope(moduleScope, options);
}
2024-07-31 10:39:30 +08:00
function _next(callback) {
fakeUpdateLoaderOptions.updateIndex++;
compiler.run((err, stats) => {
if (err) return callback(err);
const jsonStats = stats.toJson({
errorDetails: true
});
2024-07-31 10:39:30 +08:00
if (
checkArrayExpectation(
testDirectory,
jsonStats,
"error",
`errors${fakeUpdateLoaderOptions.updateIndex}`,
"Error",
2024-11-01 21:55:49 +08:00
options,
2024-07-31 10:39:30 +08:00
callback
)
) {
return;
}
if (
checkArrayExpectation(
testDirectory,
jsonStats,
"warning",
`warnings${fakeUpdateLoaderOptions.updateIndex}`,
"Warning",
2024-11-01 21:55:49 +08:00
options,
2024-07-31 10:39:30 +08:00
callback
)
) {
return;
}
callback(null, jsonStats);
});
}
2025-04-22 20:42:33 +08:00
/**
* @private
* @param {string} module module
* @returns {EXPECTED_ANY} required module
*/
2024-07-31 10:39:30 +08:00
function _require(module) {
if (module.startsWith("./")) {
const p = path.join(outputDirectory, module);
2024-11-30 01:35:09 +08:00
if (module.endsWith(".css")) {
return fs.readFileSync(p, "utf-8");
}
2024-07-31 10:39:30 +08:00
if (module.endsWith(".json")) {
return JSON.parse(fs.readFileSync(p, "utf-8"));
2024-07-31 04:21:27 +08:00
}
2024-07-31 10:39:30 +08:00
const fn = vm.runInThisContext(
2024-07-31 12:23:44 +08:00
"(function(require, module, exports, __dirname, __filename, it, beforeEach, afterEach, expect, jest, self, window, fetch, document, importScripts, Worker, EventSource, NEXT, STATS) {" +
"global.expect = expect;" +
"global.it = it;" +
2024-07-31 10:39:30 +08:00
`function nsObj(m) { Object.defineProperty(m, Symbol.toStringTag, { value: "Module" }); return m; }${fs.readFileSync(
p,
"utf-8"
)}\n})`,
p
);
const m = {
exports: {}
};
fn.call(
m.exports,
_require,
m,
m.exports,
outputDirectory,
p,
_it,
_beforeEach,
_afterEach,
expect,
jest,
window,
window,
window.fetch,
window.document,
window.importScripts,
window.Worker,
window.EventSource,
_next,
jsonStats
);
2024-07-31 10:39:30 +08:00
return m.exports;
}
2024-07-31 10:39:30 +08:00
return require(module);
}
let promise = Promise.resolve();
const info = stats.toJson({ all: false, entrypoints: true });
if (config.target === "web") {
for (const file of info.entrypoints.main.assets) {
if (file.name.endsWith(".css")) {
const link = window.document.createElement("link");
link.href = path.join(outputDirectory, file.name);
window.document.head.appendChild(link);
} else {
_require(`./${file.name}`);
}
}
2024-07-31 10:39:30 +08:00
} else {
const assets = info.entrypoints.main.assets;
const result = _require(
`./${assets[assets.length - 1].name}`
);
2024-07-31 10:39:30 +08:00
if (typeof result === "object" && "then" in result)
promise = promise.then(() => result);
}
promise.then(
() => {
if (getNumberOfTests() < 1)
return done(new Error("No tests exported by test case"));
done();
},
err => {
console.log(err);
done(err);
}
);
};
compiler = webpack(options);
compiler.run(onCompiled);
2024-07-31 10:39:30 +08:00
}, 20000);
const {
it: _it,
beforeEach: _beforeEach,
afterEach: _afterEach,
getNumberOfTests
} = createLazyTestEnv(20000);
});
2024-08-02 02:36:27 +08:00
}
});
2024-08-02 02:36:27 +08:00
}
});
};
2024-06-11 20:32:02 +08:00
// eslint-disable-next-line jest/no-export
module.exports.describeCases = describeCases;