2018-11-28 20:07:40 +08:00
|
|
|
"use strict";
|
|
|
|
|
2021-07-15 20:51:52 +08:00
|
|
|
require("./helpers/warmup-webpack");
|
|
|
|
|
2018-11-28 20:07:40 +08:00
|
|
|
const path = require("path");
|
2019-06-11 19:09:42 +08:00
|
|
|
const fs = require("graceful-fs");
|
2018-11-28 20:07:40 +08:00
|
|
|
const vm = require("vm");
|
2019-12-28 04:25:56 +08:00
|
|
|
const rimraf = require("rimraf");
|
2018-11-28 20:07:40 +08:00
|
|
|
const checkArrayExpectation = require("./checkArrayExpectation");
|
|
|
|
const createLazyTestEnv = require("./helpers/createLazyTestEnv");
|
2025-04-06 20:53:42 +08:00
|
|
|
const FakeDocument = require("./helpers/FakeDocument");
|
2018-11-28 20:07:40 +08:00
|
|
|
|
|
|
|
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
|
|
|
}));
|
2018-11-28 20:07:40 +08:00
|
|
|
|
|
|
|
const describeCases = config => {
|
|
|
|
describe(config.name, () => {
|
2024-08-02 02:36:27 +08:00
|
|
|
for (const category of categories) {
|
2018-11-28 20:07:40 +08:00
|
|
|
describe(category.name, () => {
|
2024-08-02 02:36:27 +08:00
|
|
|
for (const testName of category.tests) {
|
2020-11-29 02:12:32 +08:00
|
|
|
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
|
2020-11-29 02:12:32 +08:00
|
|
|
describe.skip(testName, () => {
|
|
|
|
it("filtered", () => {});
|
|
|
|
});
|
2024-08-02 02:36:27 +08:00
|
|
|
continue;
|
2020-11-29 02:12:32 +08:00
|
|
|
}
|
2018-11-28 20:07:40 +08:00
|
|
|
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(
|
2018-11-28 20:07:40 +08:00
|
|
|
__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
|
|
|
|
}
|
|
|
|
|
2025-04-06 20:53:42 +08:00
|
|
|
const onCompiled = (err, stats) => {
|
2024-07-31 10:39:30 +08:00
|
|
|
if (err) return done(err);
|
|
|
|
const jsonStats = stats.toJson({
|
|
|
|
errorDetails: true
|
2018-11-28 20:07:40 +08:00
|
|
|
});
|
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;
|
|
|
|
}
|
2018-11-28 20:07:40 +08:00
|
|
|
|
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") {
|
2020-11-28 19:42:03 +08:00
|
|
|
return {
|
2024-07-31 10:39:30 +08:00
|
|
|
status: 404,
|
|
|
|
ok: false
|
2020-11-28 19:42:03 +08:00
|
|
|
};
|
|
|
|
}
|
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) {
|
2025-04-06 20:53:42 +08:00
|
|
|
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
|
|
|
|
);
|
2018-11-28 20:07:40 +08:00
|
|
|
}
|
|
|
|
}
|
2024-07-31 10:39:30 +08:00
|
|
|
};
|
2025-04-06 20:53:42 +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");
|
2018-11-28 20:07:40 +08:00
|
|
|
}
|
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";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
2018-11-28 20:07:40 +08:00
|
|
|
|
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
|
2018-11-28 20:07:40 +08:00
|
|
|
});
|
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);
|
|
|
|
});
|
|
|
|
}
|
2018-11-28 20:07:40 +08:00
|
|
|
|
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;" +
|
2025-05-07 00:54:23 +08:00
|
|
|
"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
|
2020-09-03 17:38:38 +08:00
|
|
|
);
|
2024-07-31 10:39:30 +08:00
|
|
|
return m.exports;
|
2020-04-25 06:28:27 +08:00
|
|
|
}
|
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") {
|
2025-04-06 20:53:42 +08:00
|
|
|
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}`
|
2020-04-25 06:28:27 +08:00
|
|
|
);
|
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);
|
|
|
|
}
|
|
|
|
);
|
2025-04-06 20:53:42 +08:00
|
|
|
};
|
|
|
|
compiler = webpack(options);
|
|
|
|
compiler.run(onCompiled);
|
2024-07-31 10:39:30 +08:00
|
|
|
}, 20000);
|
2018-11-28 20:07:40 +08:00
|
|
|
|
2020-09-26 09:42:37 +08:00
|
|
|
const {
|
|
|
|
it: _it,
|
|
|
|
beforeEach: _beforeEach,
|
|
|
|
afterEach: _afterEach,
|
|
|
|
getNumberOfTests
|
2021-06-21 18:06:41 +08:00
|
|
|
} = createLazyTestEnv(20000);
|
2018-11-28 20:07:40 +08:00
|
|
|
});
|
2024-08-02 02:36:27 +08:00
|
|
|
}
|
2018-11-28 20:07:40 +08:00
|
|
|
});
|
2024-08-02 02:36:27 +08:00
|
|
|
}
|
2018-11-28 20:07:40 +08:00
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2024-06-11 20:32:02 +08:00
|
|
|
// eslint-disable-next-line jest/no-export
|
2018-11-28 20:07:40 +08:00
|
|
|
module.exports.describeCases = describeCases;
|