webpack/test/HotTestCases.template.js

332 lines
9.7 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 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, () => {
categories.forEach(category => {
describe(category.name, () => {
category.tests.forEach(testName => {
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", () => {});
});
return;
}
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;
compiler = webpack(options);
compiler.run((err, stats) => {
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",
done
)
) {
return;
}
if (
checkArrayExpectation(
testDirectory,
jsonStats,
"warning",
"Warning",
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 = {
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) {
return {
_type: type,
_attrs: {},
setAttribute(name, value) {
this._attrs[name] = value;
},
parentNode: {
removeChild(node) {
// ok
}
}
2024-07-31 10:39:30 +08:00
};
2020-10-27 18:53:17 +08:00
},
2024-07-31 10:39:30 +08:00
head: {
appendChild(element) {
if (element._type === "script") {
// run it
Promise.resolve().then(() => {
_require(urlToRelativePath(element.src));
});
}
2020-10-27 18:53:17 +08:00
}
2024-07-31 10:39:30 +08:00
},
getElementsByTagName(name) {
if (name === "head") return [this.head];
if (name === "script") return [];
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-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",
callback
)
) {
return;
}
if (
checkArrayExpectation(
testDirectory,
jsonStats,
"warning",
`warnings${fakeUpdateLoaderOptions.updateIndex}`,
"Warning",
callback
)
) {
return;
}
callback(null, jsonStats);
});
}
2024-07-31 10:39:30 +08:00
function _require(module) {
if (module.startsWith("./")) {
const p = path.join(outputDirectory, module);
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;" +
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)
_require(`./${file.name}`);
} 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);
}
);
});
}, 20000);
const {
it: _it,
beforeEach: _beforeEach,
afterEach: _afterEach,
getNumberOfTests
} = createLazyTestEnv(20000);
});
});
});
});
});
};
2024-06-11 20:32:02 +08:00
// eslint-disable-next-line jest/no-export
module.exports.describeCases = describeCases;