2019-08-13 19:31:07 +08:00
|
|
|
"use strict";
|
|
|
|
|
|
|
|
const path = require("path");
|
|
|
|
const child_process = require("child_process");
|
|
|
|
const fs = require("fs");
|
|
|
|
const rimraf = require("rimraf");
|
|
|
|
|
|
|
|
const cacheDirectory = path.resolve(__dirname, "js/buildDepsCache");
|
|
|
|
const outputDirectory = path.resolve(__dirname, "js/buildDeps");
|
|
|
|
const inputDirectory = path.resolve(__dirname, "js/buildDepsInput");
|
|
|
|
|
2020-12-15 23:58:40 +08:00
|
|
|
const exec = (n, options = {}) => {
|
2019-08-13 19:31:07 +08:00
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
const p = child_process.fork(
|
|
|
|
path.resolve(__dirname, "fixtures/buildDependencies/run.js"),
|
2020-12-15 23:58:40 +08:00
|
|
|
[n, JSON.stringify(options)],
|
|
|
|
{ stdio: ["ignore", "pipe", "pipe", "ipc"] }
|
2019-08-13 19:31:07 +08:00
|
|
|
);
|
|
|
|
const chunks = [];
|
2020-12-15 23:58:40 +08:00
|
|
|
p.stderr.on("data", chunk => chunks.push(chunk));
|
2019-08-13 19:31:07 +08:00
|
|
|
p.stdout.on("data", chunk => chunks.push(chunk));
|
|
|
|
p.once("exit", code => {
|
|
|
|
const stdout = Buffer.concat(chunks).toString("utf-8");
|
|
|
|
if (code === 0) {
|
|
|
|
resolve(stdout);
|
|
|
|
} else {
|
|
|
|
reject(new Error(`Code ${code}: ${stdout}`));
|
|
|
|
}
|
|
|
|
});
|
|
|
|
p.once("error", err => reject(err));
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
describe("BuildDependencies", () => {
|
|
|
|
beforeEach(done => {
|
|
|
|
rimraf(cacheDirectory, done);
|
|
|
|
});
|
|
|
|
beforeEach(done => {
|
|
|
|
rimraf(outputDirectory, done);
|
|
|
|
});
|
2019-08-20 18:31:36 +08:00
|
|
|
|
|
|
|
beforeEach(done => {
|
|
|
|
rimraf(inputDirectory, done);
|
|
|
|
});
|
2019-08-13 19:31:07 +08:00
|
|
|
beforeEach(done => {
|
|
|
|
fs.mkdir(inputDirectory, { recursive: true }, done);
|
|
|
|
});
|
2019-08-20 18:31:36 +08:00
|
|
|
it("should capture loader and config dependencies", async () => {
|
2020-12-15 23:58:40 +08:00
|
|
|
fs.writeFileSync(
|
|
|
|
path.resolve(inputDirectory, "loader-dependency.js"),
|
|
|
|
"module.exports = 0;"
|
|
|
|
);
|
|
|
|
fs.writeFileSync(
|
|
|
|
path.resolve(inputDirectory, "config-dependency.js"),
|
|
|
|
"module.exports = 0;"
|
|
|
|
);
|
|
|
|
await exec("0", { invalidBuildDepdencies: true, buildTwice: true });
|
2019-08-13 19:31:07 +08:00
|
|
|
fs.writeFileSync(
|
|
|
|
path.resolve(inputDirectory, "loader-dependency.js"),
|
|
|
|
"module.exports = 1;"
|
|
|
|
);
|
2019-08-20 15:07:51 +08:00
|
|
|
fs.writeFileSync(
|
|
|
|
path.resolve(inputDirectory, "config-dependency.js"),
|
|
|
|
"module.exports = 1;"
|
|
|
|
);
|
2020-12-15 23:58:40 +08:00
|
|
|
await exec("1");
|
2019-08-13 19:31:07 +08:00
|
|
|
fs.writeFileSync(
|
|
|
|
path.resolve(inputDirectory, "loader-dependency.js"),
|
2019-08-13 23:58:51 +08:00
|
|
|
"module.exports = Date.now();"
|
2019-08-13 19:31:07 +08:00
|
|
|
);
|
2019-08-20 15:07:51 +08:00
|
|
|
const now1 = Date.now();
|
2019-08-13 19:31:07 +08:00
|
|
|
await exec("2");
|
2020-12-15 23:58:40 +08:00
|
|
|
await exec("3");
|
2019-08-20 15:07:51 +08:00
|
|
|
fs.writeFileSync(
|
2019-08-20 19:50:14 +08:00
|
|
|
path.resolve(inputDirectory, "config-dependency"),
|
2019-08-20 15:07:51 +08:00
|
|
|
"module.exports = Date.now();"
|
|
|
|
);
|
|
|
|
const now2 = Date.now();
|
|
|
|
await exec("4");
|
2020-12-15 23:58:40 +08:00
|
|
|
const now3 = Date.now();
|
|
|
|
await exec("5");
|
|
|
|
const results = Array.from({ length: 6 }).map((_, i) =>
|
2019-08-20 15:07:51 +08:00
|
|
|
require(`./js/buildDeps/${i}/main.js`)
|
|
|
|
);
|
|
|
|
for (const r of results) {
|
|
|
|
expect(typeof r.loader).toBe("number");
|
|
|
|
expect(typeof r.config).toBe("number");
|
|
|
|
expect(typeof r.uncached).toBe("number");
|
|
|
|
}
|
2020-12-15 23:58:40 +08:00
|
|
|
expect(results[0].loader).toBe(0);
|
|
|
|
expect(results[0].config).toBe(0);
|
|
|
|
expect(results[0].uncached).toBe(0);
|
|
|
|
// 0 -> 1 should not cache at all because of invalid buildDeps
|
|
|
|
expect(results[1].loader).toBe(1);
|
2019-08-20 15:07:51 +08:00
|
|
|
expect(results[1].config).toBe(1);
|
|
|
|
expect(results[1].uncached).toBe(1);
|
2020-12-15 23:58:40 +08:00
|
|
|
// 1 -> 2 should be invalidated
|
|
|
|
expect(results[2].loader).toBeGreaterThan(now1);
|
2019-08-20 15:07:51 +08:00
|
|
|
expect(results[2].config).toBe(1);
|
|
|
|
expect(results[2].uncached).toBe(1);
|
2020-12-15 23:58:40 +08:00
|
|
|
// 2 -> 3 should stay cached
|
|
|
|
expect(results[3].loader).toBe(results[2].loader);
|
|
|
|
expect(results[3].config).toBe(1);
|
|
|
|
expect(results[3].uncached).toBe(1);
|
|
|
|
// 3 -> 4 should be invalidated
|
|
|
|
expect(results[4].loader).toBeGreaterThan(now2);
|
|
|
|
expect(results[4].config).toBeGreaterThan(now2);
|
|
|
|
expect(results[4].uncached).toBe(results[4].config);
|
|
|
|
// 4 -> 5 should stay cached, but uncacheable module still rebuilds
|
|
|
|
expect(results[5].loader).toBe(results[4].loader);
|
|
|
|
expect(results[5].config).toBe(results[4].config);
|
|
|
|
expect(results[5].uncached).toBeGreaterThan(now3);
|
2019-08-20 15:07:51 +08:00
|
|
|
}, 100000);
|
2019-08-13 19:31:07 +08:00
|
|
|
});
|