webpack/test/HotTestCases.template.js

340 lines
10 KiB
JavaScript

"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());
categories = categories.map(cat => {
return {
name: cat,
tests: fs
.readdirSync(path.join(casesPath, cat))
.filter(folder => folder.indexOf("_") < 0)
};
});
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");
if (fs.existsSync(filterPath) && !require(filterPath)(config)) {
describe.skip(testName, () => {
it("filtered", () => {});
});
return;
}
describe(testName, () => {
let compiler;
afterAll(callback => {
compiler.close(callback);
compiler = undefined;
});
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,
"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
});
if (
checkArrayExpectation(
testDirectory,
jsonStats,
"error",
"Error",
done
)
) {
return;
}
if (
checkArrayExpectation(
testDirectory,
jsonStats,
"warning",
"Warning",
done
)
) {
return;
}
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 {
status: 404,
ok: false
};
}
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
}
}
};
},
head: {
appendChild(element) {
if (element._type === "script") {
// run it
Promise.resolve().then(() => {
_require(urlToRelativePath(element.src));
});
}
}
},
getElementsByTagName(name) {
if (name === "head") return [this.head];
if (name === "script") return [];
throw new Error("Not supported");
}
},
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";
}
}
};
function _next(callback) {
fakeUpdateLoaderOptions.updateIndex++;
compiler.run((err, stats) => {
if (err) return callback(err);
const jsonStats = stats.toJson({
errorDetails: true
});
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);
});
}
function _require(module) {
if (module.substr(0, 2) === "./") {
const p = path.join(outputDirectory, module);
if (module.endsWith(".json")) {
return JSON.parse(fs.readFileSync(p, "utf-8"));
} else {
const fn = vm.runInThisContext(
"(function(require, module, exports, __dirname, __filename, it, beforeEach, afterEach, expect, jest, self, window, fetch, document, importScripts, Worker, EventSource, NEXT, STATS) {" +
"global.expect = expect;" +
'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
);
return m.exports;
}
} else 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}`
);
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);
});
});
});
});
});
};
module.exports.describeCases = describeCases;