Skip to content

Commit 1504f4e

Browse files
committed
Add test with custom webpack cache plugin
1 parent 360afe9 commit 1504f4e

File tree

1 file changed

+106
-0
lines changed

1 file changed

+106
-0
lines changed

test/cache.test.js

+106
Original file line numberDiff line numberDiff line change
@@ -75,6 +75,10 @@ test("should output files to cache directory when cache type is filesystem", asy
7575
presets: ["@babel/preset-env"],
7676
},
7777
},
78+
// when cache.type is filesystem, webpack will try to cache the loader result if they
79+
// are cacheable (by default). The webpack cache will then be hit before the babel-loader
80+
// cache. To test the babel-loader cache behaviour, we have to mark the loader results
81+
// as uncacheable
7882
{
7983
loader: "./test/fixtures/uncacheable-passthrough-loader.cjs",
8084
},
@@ -595,3 +599,105 @@ test("should work with memory type webpack cache", async () => {
595599
});
596600
});
597601
});
602+
603+
test("it should work with custom webpack cache plugin", async () => {
604+
class CustomCachePlugin {
605+
/**
606+
* Apply the plugin
607+
* @param {import("webpack").Compiler} compiler the compiler instance
608+
*/
609+
apply(compiler) {
610+
let cache = Object.create(null);
611+
const pluginName = this.constructor.name;
612+
613+
compiler.cache.hooks.store.tap(pluginName, (identifier, etag, data) => {
614+
cache[identifier] = { etag, data };
615+
});
616+
617+
compiler.cache.hooks.get.tap(
618+
pluginName,
619+
(identifier, etag, gotHandlers) => {
620+
if (!(identifier in cache)) {
621+
return null;
622+
} else if (cache[identifier] != null) {
623+
const cacheEntry = cache[identifier];
624+
if (cacheEntry.etag === etag) {
625+
return cacheEntry.data;
626+
} else {
627+
return null;
628+
}
629+
}
630+
gotHandlers.push((result, callback) => {
631+
if (result === undefined) {
632+
cache[identifier] = null;
633+
} else {
634+
cache[identifier] = { etag, data: result };
635+
}
636+
return callback();
637+
});
638+
},
639+
);
640+
641+
compiler.cache.hooks.shutdown.tap(
642+
pluginName,
643+
() => (cache = Object.create(null)),
644+
);
645+
}
646+
}
647+
648+
const config = Object.assign({}, globalConfig, {
649+
// disable builtin webpack cache so that CustomCachePlugin can provide the cache backend
650+
cache: false,
651+
entry: path.join(__dirname, "fixtures/constant.js"),
652+
output: {
653+
path: context.directory,
654+
},
655+
module: {
656+
rules: [
657+
{
658+
test: /\.jsx?/,
659+
exclude: /node_modules/,
660+
use: [
661+
{
662+
loader: babelLoader,
663+
options: {
664+
cacheDirectory: true,
665+
configFile: false,
666+
presets: ["@babel/preset-env"],
667+
},
668+
},
669+
{
670+
loader: "./test/fixtures/uncacheable-passthrough-loader.cjs",
671+
},
672+
],
673+
},
674+
],
675+
},
676+
plugins: [new CustomCachePlugin()],
677+
stats: {
678+
loggingDebug: ["babel-loader", CustomCachePlugin.name],
679+
},
680+
});
681+
682+
return new Promise((resolve, reject) => {
683+
const compiler = webpack(config);
684+
compiler.run((err, stats) => {
685+
if (err) reject(err);
686+
assert.match(
687+
stats.toString(config.stats),
688+
/normalizing loader options\n\s+resolving Babel configs\n\s+cache is enabled\n\s+getting cache for.+\n\s+missed cache for.+\n\s+applying Babel transform\n\s+caching result for.+\n\s+cached result for.+/,
689+
"The first run stat does not match the snapshot regex",
690+
);
691+
compiler.run((err, newStats) => {
692+
if (err) reject(err);
693+
console.log(newStats.toString(config.stats));
694+
assert.match(
695+
newStats.toString(config.stats),
696+
/normalizing loader options\n\s+resolving Babel configs\n\s+cache is enabled\n\s+getting cache for.+\n\s+found cache for.+/,
697+
"The second run stat does not match the snapshot regex",
698+
);
699+
resolve();
700+
});
701+
});
702+
});
703+
});

0 commit comments

Comments
 (0)